1 /*****************************************************************************
2 * Windows.m: MacOS X interface module
3 *****************************************************************************
4 * Copyright (C) 2012-2014 VLC authors and VideoLAN
7 * Authors: Felix Paul Kühne <fkuehne -at- videolan -dot- org>
8 * David Fuhrmann <david dot fuhrmann at googlemail dot com>
10 * This program is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License as published by
12 * the Free Software Foundation; either version 2 of the License, or
13 * (at your option) any later version.
15 * This program is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU General Public License for more details.
20 * You should have received a copy of the GNU General Public License
21 * along with this program; if not, write to the Free Software
22 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
23 *****************************************************************************/
27 #import "CoreInteraction.h"
28 #import "ControlsBar.h"
30 #import "CompatibilityFixes.h"
32 /*****************************************************************************
35 * Missing extension to NSWindow
36 *****************************************************************************/
38 @implementation VLCWindow
40 @synthesize hasActiveVideo=b_has_active_video;
41 @synthesize fullscreen=b_fullscreen;
43 - (id)initWithContentRect:(NSRect)contentRect styleMask:(NSUInteger)styleMask
44 backing:(NSBackingStoreType)backingType defer:(BOOL)flag
46 self = [super initWithContentRect:contentRect styleMask:styleMask backing:backingType defer:flag];
48 /* we don't want this window to be restored on relaunch */
49 if (!OSX_SNOW_LEOPARD)
50 [self setRestorable:NO];
55 - (void)setCanBecomeKeyWindow: (BOOL)canBecomeKey
57 b_isset_canBecomeKeyWindow = YES;
58 b_canBecomeKeyWindow = canBecomeKey;
61 - (BOOL)canBecomeKeyWindow
63 if (b_isset_canBecomeKeyWindow)
64 return b_canBecomeKeyWindow;
66 return [super canBecomeKeyWindow];
69 - (void)setCanBecomeMainWindow: (BOOL)canBecomeMain
71 b_isset_canBecomeMainWindow = YES;
72 b_canBecomeMainWindow = canBecomeMain;
75 - (BOOL)canBecomeMainWindow
77 if (b_isset_canBecomeMainWindow)
78 return b_canBecomeMainWindow;
80 return [super canBecomeMainWindow];
83 - (void)closeAndAnimate: (BOOL)animate
92 // TODO this callback stuff does not work and is not needed
93 invoc = [[[NSInvocation alloc] init] autorelease];
94 [invoc setSelector:@selector(close)];
95 [invoc setTarget: self];
97 if (![self isVisible] || [self alphaValue] == 0.0) {
102 [self orderOut: self animate: YES callback: invoc];
105 - (void)orderOut: (id)sender animate: (BOOL)animate
107 NSInvocation *invoc = [[[NSInvocation alloc] init] autorelease];
108 [invoc setSelector:@selector(orderOut:)];
109 [invoc setTarget: self];
110 [invoc setArgument: sender atIndex: 2];
111 [self orderOut: sender animate: animate callback: invoc];
114 - (void)orderOut: (id)sender animate: (BOOL)animate callback:(NSInvocation *)callback
116 NSViewAnimation *anim;
117 NSViewAnimation *current_anim;
118 NSMutableDictionary *dict;
121 [self orderOut: sender];
125 dict = [[NSMutableDictionary alloc] initWithCapacity:2];
127 [dict setObject:self forKey:NSViewAnimationTargetKey];
129 [dict setObject:NSViewAnimationFadeOutEffect forKey:NSViewAnimationEffectKey];
130 anim = [[NSViewAnimation alloc] initWithViewAnimations:[NSArray arrayWithObject:dict]];
133 [anim setAnimationBlockingMode:NSAnimationNonblocking];
134 [anim setDuration:0.9];
135 [anim setFrameRate:30];
136 [anim setUserInfo:callback];
137 [anim setDelegate:self];
139 @synchronized(self) {
140 current_anim = self->o_current_animation;
142 if ([[[current_anim viewAnimations] objectAtIndex:0] objectForKey: NSViewAnimationEffectKey] == NSViewAnimationFadeOutEffect && [current_anim isAnimating]) {
146 [current_anim stopAnimation];
147 [anim setCurrentProgress:1.0 - [current_anim currentProgress]];
148 [current_anim release];
151 [anim setCurrentProgress:1.0 - [self alphaValue]];
152 self->o_current_animation = anim;
153 [anim startAnimation];
158 - (void)orderFront: (id)sender animate: (BOOL)animate
160 NSViewAnimation *anim;
161 NSViewAnimation *current_anim;
162 NSMutableDictionary *dict;
165 [super orderFront: sender];
166 [self setAlphaValue: 1.0];
170 if (![self isVisible]) {
171 [self setAlphaValue: 0.0];
172 [super orderFront: sender];
174 else if ([self alphaValue] == 1.0) {
175 [super orderFront: self];
179 dict = [[NSMutableDictionary alloc] initWithCapacity:2];
181 [dict setObject:self forKey:NSViewAnimationTargetKey];
183 [dict setObject:NSViewAnimationFadeInEffect forKey:NSViewAnimationEffectKey];
184 anim = [[NSViewAnimation alloc] initWithViewAnimations:[NSArray arrayWithObject:dict]];
187 [anim setAnimationBlockingMode:NSAnimationNonblocking];
188 [anim setDuration:0.5];
189 [anim setFrameRate:30];
190 [anim setDelegate:self];
192 @synchronized(self) {
193 current_anim = self->o_current_animation;
195 if ([[[current_anim viewAnimations] objectAtIndex:0] objectForKey: NSViewAnimationEffectKey] == NSViewAnimationFadeInEffect && [current_anim isAnimating]) {
199 [current_anim stopAnimation];
200 [anim setCurrentProgress:1.0 - [current_anim currentProgress]];
201 [current_anim release];
204 [anim setCurrentProgress:[self alphaValue]];
205 self->o_current_animation = anim;
206 [self orderFront: sender];
207 [anim startAnimation];
212 - (void)animationDidEnd:(NSAnimation*)anim
214 if ([self alphaValue] <= 0.0) {
215 NSInvocation * invoc;
216 [super orderOut: nil];
217 [self setAlphaValue: 1.0];
218 if ((invoc = [anim userInfo])) {
224 - (VLCVoutView *)videoView
226 NSArray *o_subViews = [[self contentView] subviews];
227 if ([o_subViews count] > 0) {
228 id o_vout_view = [o_subViews objectAtIndex:0];
230 if ([o_vout_view class] == [VLCVoutView class])
231 return (VLCVoutView *)o_vout_view;
237 - (NSRect)constrainFrameRect:(NSRect)frameRect toScreen:(NSScreen *)screen
240 screen = [self screen];
241 NSRect screenRect = [screen frame];
242 NSRect constrainedRect = [super constrainFrameRect:frameRect toScreen:screen];
246 * With Mavericks, there is a nasty bug resulting in grey bars on top in fullscreen mode.
247 * It looks like this is enforced by the os because the window is in the way for the menu bar.
249 * According to the documentation, this constraining can be changed by overwriting this
250 * method. But in this situation, even the received frameRect is already contrained with the
251 * menu bars height substracted. This case is detected here, and the full height is
254 * See #9469 and radar://15583566
257 BOOL b_inFullscreen = [self fullscreen] || ([self respondsToSelector:@selector(inFullscreenTransition)] && [(VLCVideoWindowCommon *)self inFullscreenTransition]);
259 if((OSX_MAVERICKS || OSX_YOSEMITE) && b_inFullscreen && constrainedRect.size.width == screenRect.size.width
260 && constrainedRect.size.height != screenRect.size.height
261 && abs(screenRect.size.height - constrainedRect.size.height) <= 25.) {
263 msg_Dbg(VLCIntf, "Contrain window height %.1f to screen height %.1f",
264 constrainedRect.size.height, screenRect.size.height);
265 constrainedRect.size.height = screenRect.size.height;
268 return constrainedRect;
274 /*****************************************************************************
275 * VLCVideoWindowCommon
277 * Common code for main window, detached window and extra video window
278 *****************************************************************************/
280 @interface VLCVideoWindowCommon (Internal)
281 - (void)customZoom:(id)sender;
282 - (void)hasBecomeFullscreen;
283 - (void)hasEndedFullscreen;
286 @implementation VLCVideoWindowCommon
288 @synthesize videoView=o_video_view;
289 @synthesize controlsBar=o_controls_bar;
290 @synthesize inFullscreenTransition=b_in_fullscreen_transition;
291 @synthesize windowShouldExitFullscreenWhenFinished=b_windowShouldExitFullscreenWhenFinished;
296 - (id)initWithContentRect:(NSRect)contentRect styleMask:(NSUInteger)styleMask
297 backing:(NSBackingStoreType)backingType defer:(BOOL)flag
299 b_dark_interface = config_GetInt(VLCIntf, "macosx-interfacestyle");
301 if (b_dark_interface) {
302 styleMask = NSBorderlessWindowMask;
303 #ifdef MAC_OS_X_VERSION_10_7
304 if (!OSX_SNOW_LEOPARD)
305 styleMask |= NSResizableWindowMask;
309 self = [super initWithContentRect:contentRect styleMask:styleMask
310 backing:backingType defer:flag];
312 /* we want to be moveable regardless of our style */
313 [self setMovableByWindowBackground: YES];
314 [self setCanBecomeKeyWindow:YES];
316 o_temp_view = [[NSView alloc] init];
317 [o_temp_view setAutoresizingMask:NSViewHeightSizable | NSViewWidthSizable];
324 [o_temp_view release];
330 BOOL b_nativeFullscreenMode = NO;
331 #ifdef MAC_OS_X_VERSION_10_7
332 if (!OSX_SNOW_LEOPARD)
333 b_nativeFullscreenMode = var_InheritBool(VLCIntf, "macosx-nativefullscreenmode");
336 if (b_nativeFullscreenMode) {
337 [self setCollectionBehavior: NSWindowCollectionBehaviorFullScreenPrimary];
341 [super awakeFromNib];
344 - (void)setTitle:(NSString *)title
346 if (!title || [title length] < 1)
349 if (b_dark_interface && o_titlebar_view)
350 [o_titlebar_view setWindowTitle: title];
352 [super setTitle: title];
356 #pragma mark zoom / minimize / close
358 - (BOOL)validateMenuItem:(NSMenuItem *)menuItem
360 SEL s_menuAction = [menuItem action];
362 if ((s_menuAction == @selector(performClose:)) || (s_menuAction == @selector(performMiniaturize:)) || (s_menuAction == @selector(performZoom:)))
365 return [super validateMenuItem:menuItem];
368 - (BOOL)windowShouldClose:(id)sender
373 - (void)performClose:(id)sender
375 if (!([self styleMask] & NSTitledWindowMask)) {
376 [[NSNotificationCenter defaultCenter] postNotificationName:NSWindowWillCloseNotification object:self];
380 [super performClose: sender];
383 - (void)performMiniaturize:(id)sender
385 if (!([self styleMask] & NSTitledWindowMask))
386 [self miniaturize: sender];
388 [super performMiniaturize: sender];
391 - (void)performZoom:(id)sender
393 if (!([self styleMask] & NSTitledWindowMask))
394 [self customZoom: sender];
396 [super performZoom: sender];
399 - (void)zoom:(id)sender
401 if (!([self styleMask] & NSTitledWindowMask))
402 [self customZoom: sender];
404 [super zoom: sender];
408 * Given a proposed frame rectangle, return a modified version
409 * which will fit inside the screen.
411 * This method is based upon NSWindow.m, part of the GNUstep GUI Library, licensed under LGPLv2+.
412 * Authors: Scott Christley <scottc@net-community.com>, Venkat Ajjanagadde <venkat@ocbi.com>,
413 * Felipe A. Rodriguez <far@ix.netcom.com>, Richard Frith-Macdonald <richard@brainstorm.co.uk>
414 * Copyright (C) 1996 Free Software Foundation, Inc.
416 - (NSRect) customConstrainFrameRect: (NSRect)frameRect toScreen: (NSScreen*)screen
418 NSRect screenRect = [screen visibleFrame];
421 /* Move top edge of the window inside the screen */
422 difference = NSMaxY (frameRect) - NSMaxY (screenRect);
423 if (difference > 0) {
424 frameRect.origin.y -= difference;
427 /* If the window is resizable, resize it (if needed) so that the
428 bottom edge is on the screen or can be on the screen when the user moves
430 difference = NSMaxY (screenRect) - NSMaxY (frameRect);
431 if (_styleMask & NSResizableWindowMask) {
434 difference2 = screenRect.origin.y - frameRect.origin.y;
435 difference2 -= difference;
436 // Take in account the space between the top of window and the top of the
437 // screen which can be used to move the bottom of the window on the screen
438 if (difference2 > 0) {
439 frameRect.size.height -= difference2;
440 frameRect.origin.y += difference2;
443 /* Ensure that resizing doesn't makewindow smaller than minimum */
444 difference2 = [self minSize].height - frameRect.size.height;
445 if (difference2 > 0) {
446 frameRect.size.height += difference2;
447 frameRect.origin.y -= difference2;
457 Zooms the receiver. This method calls the delegate method
458 windowShouldZoom:toFrame: to determine if the window should
459 be allowed to zoom to full screen.
461 * This method is based upon NSWindow.m, part of the GNUstep GUI Library, licensed under LGPLv2+.
462 * Authors: Scott Christley <scottc@net-community.com>, Venkat Ajjanagadde <venkat@ocbi.com>,
463 * Felipe A. Rodriguez <far@ix.netcom.com>, Richard Frith-Macdonald <richard@brainstorm.co.uk>
464 * Copyright (C) 1996 Free Software Foundation, Inc.
466 - (void) customZoom: (id)sender
468 NSRect maxRect = [[self screen] visibleFrame];
469 NSRect currentFrame = [self frame];
471 if ([[self delegate] respondsToSelector: @selector(windowWillUseStandardFrame:defaultFrame:)]) {
472 maxRect = [[self delegate] windowWillUseStandardFrame: self defaultFrame: maxRect];
475 maxRect = [self customConstrainFrameRect: maxRect toScreen: [self screen]];
477 // Compare the new frame with the current one
478 if ((abs(NSMaxX(maxRect) - NSMaxX(currentFrame)) < DIST)
479 && (abs(NSMaxY(maxRect) - NSMaxY(currentFrame)) < DIST)
480 && (abs(NSMinX(maxRect) - NSMinX(currentFrame)) < DIST)
481 && (abs(NSMinY(maxRect) - NSMinY(currentFrame)) < DIST)) {
482 // Already in zoomed mode, reset user frame, if stored
483 if ([self frameAutosaveName] != nil) {
484 [self setFrame: previousSavedFrame display: YES animate: YES];
485 [self saveFrameUsingName: [self frameAutosaveName]];
490 if ([self frameAutosaveName] != nil) {
491 [self saveFrameUsingName: [self frameAutosaveName]];
492 previousSavedFrame = [self frame];
495 [self setFrame: maxRect display: YES animate: YES];
499 #pragma mark Video window resizing logic
501 - (void)setWindowLevel:(NSInteger)i_state
503 if (var_InheritBool(VLCIntf, "video-wallpaper") || [self level] < NSNormalWindowLevel)
506 if (!b_fullscreen && !b_in_fullscreen_transition)
507 [self setLevel: i_state];
509 // save it for restore if window is currently minimized or in fullscreen
510 i_originalLevel = i_state;
513 - (NSRect)getWindowRectForProposedVideoViewSize:(NSSize)size
515 NSSize windowMinSize = [self minSize];
516 NSRect screenFrame = [[self screen] visibleFrame];
518 NSPoint topleftbase = NSMakePoint(0, [self frame].size.height);
519 NSPoint topleftscreen = [self convertBaseToScreen: topleftbase];
521 CGFloat f_width = size.width;
522 CGFloat f_height = size.height;
523 if (f_width < windowMinSize.width)
524 f_width = windowMinSize.width;
525 if (f_height < f_min_video_height)
526 f_height = f_min_video_height;
528 /* Calculate the window's new size */
530 new_frame.size.width = [self frame].size.width - [o_video_view frame].size.width + f_width;
531 new_frame.size.height = [self frame].size.height - [o_video_view frame].size.height + f_height;
532 new_frame.origin.x = topleftscreen.x;
533 new_frame.origin.y = topleftscreen.y - new_frame.size.height;
535 /* make sure the window doesn't exceed the screen size the window is on */
536 if (new_frame.size.width > screenFrame.size.width) {
537 new_frame.size.width = screenFrame.size.width;
538 new_frame.origin.x = screenFrame.origin.x;
540 if (new_frame.size.height > screenFrame.size.height) {
541 new_frame.size.height = screenFrame.size.height;
542 new_frame.origin.y = screenFrame.origin.y;
544 if (new_frame.origin.y < screenFrame.origin.y)
545 new_frame.origin.y = screenFrame.origin.y;
547 CGFloat right_screen_point = screenFrame.origin.x + screenFrame.size.width;
548 CGFloat right_window_point = new_frame.origin.x + new_frame.size.width;
549 if (right_window_point > right_screen_point)
550 new_frame.origin.x -= (right_window_point - right_screen_point);
557 // VOUT_WINDOW_SET_SIZE is triggered when exiting fullscreen. This event is ignored here
558 // to avoid interference with the animation.
559 if ([self fullscreen] || b_in_fullscreen_transition)
562 NSRect window_rect = [self getWindowRectForProposedVideoViewSize:nativeVideoSize];
563 [[self animator] setFrame:window_rect display:YES];
566 - (void)setNativeVideoSize:(NSSize)size
568 nativeVideoSize = size;
570 if (var_InheritBool(VLCIntf, "macosx-video-autoresize") && !var_InheritBool(VLCIntf, "video-wallpaper"))
574 - (NSSize)windowWillResize:(NSWindow *)window toSize:(NSSize)proposedFrameSize
576 if (![[VLCMain sharedInstance] activeVideoPlayback] || nativeVideoSize.width == 0. || nativeVideoSize.height == 0. || window != self)
577 return proposedFrameSize;
579 // needed when entering lion fullscreen mode
580 if (b_in_fullscreen_transition || [self fullscreen])
581 return proposedFrameSize;
583 if ([o_video_view isHidden])
584 return proposedFrameSize;
586 if ([[VLCCoreInteraction sharedInstance] aspectRatioIsLocked]) {
587 NSRect videoWindowFrame = [self frame];
588 NSRect viewRect = [o_video_view convertRect:[o_video_view bounds] toView: nil];
589 NSRect contentRect = [self contentRectForFrameRect:videoWindowFrame];
590 CGFloat marginy = viewRect.origin.y + videoWindowFrame.size.height - contentRect.size.height;
591 CGFloat marginx = contentRect.size.width - viewRect.size.width;
592 if (o_titlebar_view && b_dark_interface)
593 marginy += [o_titlebar_view frame].size.height;
595 proposedFrameSize.height = (proposedFrameSize.width - marginx) * nativeVideoSize.height / nativeVideoSize.width + marginy;
598 return proposedFrameSize;
601 - (void)windowWillMiniaturize:(NSNotification *)notification
603 // Set level to normal as a workaround for Mavericks bug causing window
604 // to vanish from screen, see radar://15473716
605 i_originalLevel = [self level];
606 [self setLevel: NSNormalWindowLevel];
609 - (void)windowDidDeminiaturize:(NSNotification *)notification
611 [self setLevel: i_originalLevel];
615 #pragma mark Mouse cursor handling
617 // NSTimer selectors require this function signature as per Apple's docs
618 - (void)hideMouseCursor:(NSTimer *)timer
620 [NSCursor setHiddenUntilMouseMoves: YES];
623 - (void)recreateHideMouseTimer
625 if (t_hide_mouse_timer != nil) {
626 [t_hide_mouse_timer invalidate];
627 [t_hide_mouse_timer release];
630 t_hide_mouse_timer = [NSTimer scheduledTimerWithTimeInterval:2
632 selector:@selector(hideMouseCursor:)
635 [t_hide_mouse_timer retain];
638 // Called automatically if window's acceptsMouseMovedEvents property is true
639 - (void)mouseMoved:(NSEvent *)theEvent
642 [self recreateHideMouseTimer];
644 [super mouseMoved: theEvent];
648 #pragma mark Key events
650 - (void)flagsChanged:(NSEvent *)theEvent
652 BOOL b_alt_pressed = ([theEvent modifierFlags] & NSAlternateKeyMask) != 0;
653 [o_titlebar_view informModifierPressed: b_alt_pressed];
655 [super flagsChanged:theEvent];
659 #pragma mark Lion native fullscreen handling
661 - (void)becomeKeyWindow
663 [super becomeKeyWindow];
665 // change fspanel state for the case when multiple windows are in fullscreen
666 if ([self hasActiveVideo] && [self fullscreen])
667 [[[VLCMainWindow sharedInstance] fsPanel] setActive:nil];
669 [[[VLCMainWindow sharedInstance] fsPanel] setNonActive:nil];
672 - (void)resignKeyWindow
674 [super resignKeyWindow];
676 [[[VLCMainWindow sharedInstance] fsPanel] setNonActive:nil];
679 -(NSArray*)customWindowsToEnterFullScreenForWindow:(NSWindow *)window
681 if (window == self) {
682 return [NSArray arrayWithObject:window];
688 - (NSArray*)customWindowsToExitFullScreenForWindow:(NSWindow*)window
690 if (window == self) {
691 return [NSArray arrayWithObject:window];
697 - (void)window:window startCustomAnimationToEnterFullScreenWithDuration:(NSTimeInterval)duration
699 [window setStyleMask:([window styleMask] | NSFullScreenWindowMask)];
701 NSScreen *screen = [window screen];
702 NSRect screenFrame = [screen frame];
704 [NSAnimationContext runAnimationGroup:^(NSAnimationContext *context) {
705 [context setDuration:0.5 * duration];
706 [[window animator] setFrame:screenFrame display:YES];
707 } completionHandler:nil];
710 - (void)window:window startCustomAnimationToExitFullScreenWithDuration:(NSTimeInterval)duration
712 [window setStyleMask:([window styleMask] & ~NSFullScreenWindowMask)];
713 [[window animator] setFrame:frameBeforeLionFullscreen display:YES animate:YES];
715 [NSAnimationContext runAnimationGroup:^(NSAnimationContext *context) {
716 [context setDuration:0.5 * duration];
717 [[window animator] setFrame:frameBeforeLionFullscreen display:YES animate:YES];
718 } completionHandler:nil];
721 - (void)windowWillEnterFullScreen:(NSNotification *)notification
723 i_originalLevel = [self level];
724 b_windowShouldExitFullscreenWhenFinished = [[VLCMain sharedInstance] activeVideoPlayback];
726 // b_fullscreen and b_in_fullscreen_transition must not be true yet
727 [[[VLCMain sharedInstance] voutController] updateWindowLevelForHelperWindows: NSNormalWindowLevel];
728 [self setLevel:NSNormalWindowLevel];
730 b_in_fullscreen_transition = YES;
732 var_SetBool(pl_Get(VLCIntf), "fullscreen", true);
734 frameBeforeLionFullscreen = [self frame];
736 if ([self hasActiveVideo]) {
737 vout_thread_t *p_vout = getVoutForActiveWindow();
739 var_SetBool(p_vout, "fullscreen", true);
740 vlc_object_release(p_vout);
744 if ([self hasActiveVideo])
745 [[VLCMainWindow sharedInstance] recreateHideMouseTimer];
747 if (b_dark_interface) {
748 [o_titlebar_view removeFromSuperviewWithoutNeedingDisplay];
751 CGFloat f_titleBarHeight = [o_titlebar_view frame].size.height;
752 winrect = [self frame];
754 winrect.size.height = winrect.size.height - f_titleBarHeight;
755 [self setFrame: winrect display:NO animate:NO];
758 [o_video_view setFrame: [[self contentView] frame]];
759 if (![o_video_view isHidden]) {
760 [[o_controls_bar bottomBarView] setHidden: YES];
763 [self setMovableByWindowBackground: NO];
766 - (void)windowDidEnterFullScreen:(NSNotification *)notification
768 // Indeed, we somehow can have an "inactive" fullscreen (but a visible window!).
769 // But this creates some problems when leaving fs over remote intfs, so activate app here.
770 [NSApp activateIgnoringOtherApps:YES];
772 [self setFullscreen: YES];
773 b_in_fullscreen_transition = NO;
775 if ([self hasActiveVideo]) {
776 [[[VLCMainWindow sharedInstance] fsPanel] setVoutWasUpdated: self];
777 if (![o_video_view isHidden])
778 [[[VLCMainWindow sharedInstance] fsPanel] setActive: nil];
781 NSArray *subviews = [[self videoView] subviews];
782 NSUInteger count = [subviews count];
784 for (NSUInteger x = 0; x < count; x++) {
785 if ([[subviews objectAtIndex:x] respondsToSelector:@selector(reshape)])
786 [[subviews objectAtIndex:x] reshape];
790 - (void)windowWillExitFullScreen:(NSNotification *)notification
792 b_in_fullscreen_transition = YES;
793 [self setFullscreen: NO];
795 if ([self hasActiveVideo]) {
796 var_SetBool(pl_Get(VLCIntf), "fullscreen", false);
798 vout_thread_t *p_vout = getVoutForActiveWindow();
800 var_SetBool(p_vout, "fullscreen", false);
801 vlc_object_release(p_vout);
805 [NSCursor setHiddenUntilMouseMoves: NO];
806 [[[VLCMainWindow sharedInstance] fsPanel] setNonActive: nil];
809 if (b_dark_interface) {
811 CGFloat f_titleBarHeight = [o_titlebar_view frame].size.height;
813 winrect = [o_video_view frame];
814 winrect.size.height -= f_titleBarHeight;
815 [o_video_view setFrame: winrect];
817 winrect = [self frame];
818 [o_titlebar_view setFrame: NSMakeRect(0, winrect.size.height - f_titleBarHeight,
819 winrect.size.width, f_titleBarHeight)];
820 [[self contentView] addSubview: o_titlebar_view];
822 winrect.size.height = winrect.size.height + f_titleBarHeight;
823 [self setFrame: winrect display:NO animate:NO];
826 NSRect videoViewFrame = [o_video_view frame];
827 videoViewFrame.origin.y += [o_controls_bar height];
828 videoViewFrame.size.height -= [o_controls_bar height];
829 [o_video_view setFrame: videoViewFrame];
831 if (![o_video_view isHidden]) {
832 [[o_controls_bar bottomBarView] setHidden: NO];
835 [self setMovableByWindowBackground: YES];
838 - (void)windowDidExitFullScreen:(NSNotification *)notification
840 b_in_fullscreen_transition = NO;
842 [[[VLCMain sharedInstance] voutController] updateWindowLevelForHelperWindows: i_originalLevel];
843 [self setLevel:i_originalLevel];
847 #pragma mark Fullscreen Logic
849 - (void)enterFullscreenWithAnimation:(BOOL)b_animation
851 NSMutableDictionary *dict1, *dict2;
855 BOOL blackout_other_displays = var_InheritBool(VLCIntf, "macosx-black");
857 screen = [NSScreen screenWithDisplayID:(CGDirectDisplayID)var_InheritInteger(VLCIntf, "macosx-vdev")];
860 msg_Dbg(VLCIntf, "chosen screen isn't present, using current screen for fullscreen mode");
861 screen = [self screen];
864 msg_Dbg(VLCIntf, "Using deepest screen");
865 screen = [NSScreen deepestScreen];
868 screen_rect = [screen frame];
871 [o_controls_bar setFullscreenState:YES];
872 [[[VLCMainWindow sharedInstance] controlsBar] setFullscreenState:YES];
874 [[VLCMainWindow sharedInstance] recreateHideMouseTimer];
876 if (blackout_other_displays)
877 [screen blackoutOtherScreens];
879 /* Make sure we don't see the window flashes in float-on-top mode */
880 i_originalLevel = [self level];
881 // b_fullscreen must not be true yet
882 [[[VLCMain sharedInstance] voutController] updateWindowLevelForHelperWindows: NSNormalWindowLevel];
883 [self setLevel:NSNormalWindowLevel];
885 /* Only create the o_fullscreen_window if we are not in the middle of the zooming animation */
886 if (!o_fullscreen_window) {
887 /* We can't change the styleMask of an already created NSWindow, so we create another window, and do eye catching stuff */
889 rect = [[o_video_view superview] convertRect: [o_video_view frame] toView: nil]; /* Convert to Window base coord */
890 rect.origin.x += [self frame].origin.x;
891 rect.origin.y += [self frame].origin.y;
892 o_fullscreen_window = [[VLCWindow alloc] initWithContentRect:rect styleMask: NSBorderlessWindowMask backing:NSBackingStoreBuffered defer:YES];
893 [o_fullscreen_window setBackgroundColor: [NSColor blackColor]];
894 [o_fullscreen_window setCanBecomeKeyWindow: YES];
895 [o_fullscreen_window setCanBecomeMainWindow: YES];
896 [o_fullscreen_window setHasActiveVideo: YES];
897 [o_fullscreen_window setFullscreen: YES];
899 /* Make sure video view gets visible in case the playlist was visible before */
900 b_video_view_was_hidden = [o_video_view isHidden];
901 [o_video_view setHidden: NO];
904 /* We don't animate if we are not visible, instead we
905 * simply fade the display */
906 CGDisplayFadeReservationToken token;
908 if (blackout_other_displays) {
909 CGAcquireDisplayFadeReservation(kCGMaxDisplayReservationInterval, &token);
910 CGDisplayFade(token, 0.5, kCGDisplayBlendNormal, kCGDisplayBlendSolidColor, 0, 0, 0, YES);
913 NSDisableScreenUpdates();
914 [o_video_view retain];
915 [[o_video_view superview] replaceSubview:o_video_view with:o_temp_view];
916 [o_temp_view setFrame:[o_video_view frame]];
917 [[o_fullscreen_window contentView] addSubview:o_video_view];
918 [o_video_view setFrame: [[o_fullscreen_window contentView] frame]];
919 [o_video_view release];
920 NSEnableScreenUpdates();
922 [screen setFullscreenPresentationOptions];
924 [o_fullscreen_window setFrame:screen_rect display:YES animate:NO];
926 [o_fullscreen_window orderFront:self animate:YES];
928 [o_fullscreen_window setLevel:NSNormalWindowLevel];
930 if (blackout_other_displays) {
931 CGDisplayFade(token, 0.3, kCGDisplayBlendSolidColor, kCGDisplayBlendNormal, 0, 0, 0, NO);
932 CGReleaseDisplayFadeReservation(token);
935 /* Will release the lock */
936 [self hasBecomeFullscreen];
941 /* Make sure we don't see the o_video_view disappearing of the screen during this operation */
942 NSDisableScreenUpdates();
943 [o_video_view retain];
944 [[o_video_view superview] replaceSubview:o_video_view with:o_temp_view];
945 [o_temp_view setFrame:[o_video_view frame]];
946 [[o_fullscreen_window contentView] addSubview:o_video_view];
947 [o_video_view setFrame: [[o_fullscreen_window contentView] frame]];
948 [o_video_view release];
949 [o_fullscreen_window makeKeyAndOrderFront:self];
950 NSEnableScreenUpdates();
953 /* We are in fullscreen (and no animation is running) */
954 if ([self fullscreen]) {
955 /* Make sure we are hidden */
956 [self orderOut: self];
961 if (o_fullscreen_anim1) {
962 [o_fullscreen_anim1 stopAnimation];
963 [o_fullscreen_anim1 release];
965 if (o_fullscreen_anim2) {
966 [o_fullscreen_anim2 stopAnimation];
967 [o_fullscreen_anim2 release];
970 [screen setFullscreenPresentationOptions];
972 dict1 = [[NSMutableDictionary alloc] initWithCapacity:2];
973 dict2 = [[NSMutableDictionary alloc] initWithCapacity:3];
975 [dict1 setObject:self forKey:NSViewAnimationTargetKey];
976 [dict1 setObject:NSViewAnimationFadeOutEffect forKey:NSViewAnimationEffectKey];
978 [dict2 setObject:o_fullscreen_window forKey:NSViewAnimationTargetKey];
979 [dict2 setObject:[NSValue valueWithRect:[o_fullscreen_window frame]] forKey:NSViewAnimationStartFrameKey];
980 [dict2 setObject:[NSValue valueWithRect:screen_rect] forKey:NSViewAnimationEndFrameKey];
982 /* Strategy with NSAnimation allocation:
983 - Keep at most 2 animation at a time
984 - leaveFullscreen/enterFullscreen are the only responsible for releasing and alloc-ing
986 o_fullscreen_anim1 = [[NSViewAnimation alloc] initWithViewAnimations:[NSArray arrayWithObject:dict1]];
987 o_fullscreen_anim2 = [[NSViewAnimation alloc] initWithViewAnimations:[NSArray arrayWithObject:dict2]];
992 [o_fullscreen_anim1 setAnimationBlockingMode: NSAnimationNonblocking];
993 [o_fullscreen_anim1 setDuration: 0.3];
994 [o_fullscreen_anim1 setFrameRate: 30];
995 [o_fullscreen_anim2 setAnimationBlockingMode: NSAnimationNonblocking];
996 [o_fullscreen_anim2 setDuration: 0.2];
997 [o_fullscreen_anim2 setFrameRate: 30];
999 [o_fullscreen_anim2 setDelegate: self];
1000 [o_fullscreen_anim2 startWhenAnimation: o_fullscreen_anim1 reachesProgress: 1.0];
1002 [o_fullscreen_anim1 startAnimation];
1003 /* fullscreenAnimation will be unlocked when animation ends */
1005 b_in_fullscreen_transition = YES;
1008 - (void)hasBecomeFullscreen
1010 if ([[o_video_view subviews] count] > 0)
1011 [o_fullscreen_window makeFirstResponder: [[o_video_view subviews] objectAtIndex:0]];
1013 [o_fullscreen_window makeKeyWindow];
1014 [o_fullscreen_window setAcceptsMouseMovedEvents: YES];
1016 /* tell the fspanel to move itself to front next time it's triggered */
1017 [[[VLCMainWindow sharedInstance] fsPanel] setVoutWasUpdated: o_fullscreen_window];
1018 [[[VLCMainWindow sharedInstance] fsPanel] setActive: nil];
1020 if ([self isVisible])
1021 [self orderOut: self];
1023 b_in_fullscreen_transition = NO;
1024 [self setFullscreen:YES];
1027 - (void)leaveFullscreenWithAnimation:(BOOL)b_animation
1029 NSMutableDictionary *dict1, *dict2;
1031 BOOL blackout_other_displays = var_InheritBool(VLCIntf, "macosx-black");
1034 [o_controls_bar setFullscreenState:NO];
1035 [[[VLCMainWindow sharedInstance] controlsBar] setFullscreenState:NO];
1037 /* We always try to do so */
1038 [NSScreen unblackoutScreens];
1040 [[o_video_view window] makeKeyAndOrderFront: nil];
1042 /* Don't do anything if o_fullscreen_window is already closed */
1043 if (!o_fullscreen_window) {
1047 [[[VLCMainWindow sharedInstance] fsPanel] setNonActive: nil];
1048 [[o_fullscreen_window screen] setNonFullscreenPresentationOptions];
1050 if (o_fullscreen_anim1) {
1051 [o_fullscreen_anim1 stopAnimation];
1052 [o_fullscreen_anim1 release];
1053 o_fullscreen_anim1 = nil;
1055 if (o_fullscreen_anim2) {
1056 [o_fullscreen_anim2 stopAnimation];
1057 [o_fullscreen_anim2 release];
1058 o_fullscreen_anim2 = nil;
1061 b_in_fullscreen_transition = YES;
1062 [self setFullscreen:NO];
1065 /* We don't animate if we are not visible, instead we
1066 * simply fade the display */
1067 CGDisplayFadeReservationToken token;
1069 if (blackout_other_displays) {
1070 CGAcquireDisplayFadeReservation(kCGMaxDisplayReservationInterval, &token);
1071 CGDisplayFade(token, 0.3, kCGDisplayBlendNormal, kCGDisplayBlendSolidColor, 0, 0, 0, YES);
1074 [self setAlphaValue:1.0];
1075 [self orderFront: self];
1077 /* Will release the lock */
1078 [self hasEndedFullscreen];
1080 if (blackout_other_displays) {
1081 CGDisplayFade(token, 0.5, kCGDisplayBlendSolidColor, kCGDisplayBlendNormal, 0, 0, 0, NO);
1082 CGReleaseDisplayFadeReservation(token);
1088 [self setAlphaValue: 0.0];
1089 [self orderFront: self];
1090 [[o_video_view window] orderFront: self];
1092 frame = [[o_temp_view superview] convertRect: [o_temp_view frame] toView: nil]; /* Convert to Window base coord */
1093 frame.origin.x += [self frame].origin.x;
1094 frame.origin.y += [self frame].origin.y;
1096 dict2 = [[NSMutableDictionary alloc] initWithCapacity:2];
1097 [dict2 setObject:self forKey:NSViewAnimationTargetKey];
1098 [dict2 setObject:NSViewAnimationFadeInEffect forKey:NSViewAnimationEffectKey];
1100 o_fullscreen_anim2 = [[NSViewAnimation alloc] initWithViewAnimations:[NSArray arrayWithObject:dict2]];
1103 [o_fullscreen_anim2 setAnimationBlockingMode: NSAnimationNonblocking];
1104 [o_fullscreen_anim2 setDuration: 0.3];
1105 [o_fullscreen_anim2 setFrameRate: 30];
1107 [o_fullscreen_anim2 setDelegate: self];
1109 dict1 = [[NSMutableDictionary alloc] initWithCapacity:3];
1111 [dict1 setObject:o_fullscreen_window forKey:NSViewAnimationTargetKey];
1112 [dict1 setObject:[NSValue valueWithRect:[o_fullscreen_window frame]] forKey:NSViewAnimationStartFrameKey];
1113 [dict1 setObject:[NSValue valueWithRect:frame] forKey:NSViewAnimationEndFrameKey];
1115 o_fullscreen_anim1 = [[NSViewAnimation alloc] initWithViewAnimations:[NSArray arrayWithObject:dict1]];
1118 [o_fullscreen_anim1 setAnimationBlockingMode: NSAnimationNonblocking];
1119 [o_fullscreen_anim1 setDuration: 0.2];
1120 [o_fullscreen_anim1 setFrameRate: 30];
1121 [o_fullscreen_anim2 startWhenAnimation: o_fullscreen_anim1 reachesProgress: 1.0];
1123 /* Make sure o_fullscreen_window is the frontmost window */
1124 [o_fullscreen_window orderFront: self];
1126 [o_fullscreen_anim1 startAnimation];
1127 /* fullscreenAnimation will be unlocked when animation ends */
1130 - (void)hasEndedFullscreen
1132 b_in_fullscreen_transition = NO;
1134 /* This function is private and should be only triggered at the end of the fullscreen change animation */
1135 /* Make sure we don't see the o_video_view disappearing of the screen during this operation */
1136 NSDisableScreenUpdates();
1137 [o_video_view retain];
1138 [o_video_view removeFromSuperviewWithoutNeedingDisplay];
1139 [[o_temp_view superview] replaceSubview:o_temp_view with:o_video_view];
1140 [o_video_view release];
1141 [o_video_view setFrame:[o_temp_view frame]];
1142 if ([[o_video_view subviews] count] > 0)
1143 [self makeFirstResponder: [[o_video_view subviews] objectAtIndex:0]];
1145 [o_video_view setHidden: b_video_view_was_hidden];
1147 [self makeKeyAndOrderFront:self];
1149 [o_fullscreen_window orderOut: self];
1150 NSEnableScreenUpdates();
1152 [o_fullscreen_window release];
1153 o_fullscreen_window = nil;
1155 [[[VLCMain sharedInstance] voutController] updateWindowLevelForHelperWindows: i_originalLevel];
1156 [self setLevel:i_originalLevel];
1158 [self setAlphaValue: config_GetFloat(VLCIntf, "macosx-opaqueness")];
1161 - (void)animationDidEnd:(NSAnimation*)animation
1163 NSArray *viewAnimations;
1164 if ([animation currentValue] < 1.0)
1167 /* Fullscreen ended or started (we are a delegate only for leaveFullscreen's/enterFullscren's anim2) */
1168 viewAnimations = [o_fullscreen_anim2 viewAnimations];
1169 if ([viewAnimations count] >=1 &&
1170 [[[viewAnimations objectAtIndex:0] objectForKey: NSViewAnimationEffectKey] isEqualToString:NSViewAnimationFadeInEffect]) {
1171 /* Fullscreen ended */
1172 [self hasEndedFullscreen];
1174 /* Fullscreen started */
1175 [self hasBecomeFullscreen];
1179 #pragma mark Accessibility stuff
1181 - (NSArray *)accessibilityAttributeNames
1183 if (!b_dark_interface || !o_titlebar_view)
1184 return [super accessibilityAttributeNames];
1186 static NSMutableArray *attributes = nil;
1187 if (attributes == nil) {
1188 attributes = [[super accessibilityAttributeNames] mutableCopy];
1189 NSArray *appendAttributes = [NSArray arrayWithObjects:NSAccessibilitySubroleAttribute,
1190 NSAccessibilityCloseButtonAttribute,
1191 NSAccessibilityMinimizeButtonAttribute,
1192 NSAccessibilityZoomButtonAttribute, nil];
1194 for(NSString *attribute in appendAttributes) {
1195 if (![attributes containsObject:attribute])
1196 [attributes addObject:attribute];
1202 - (id)accessibilityAttributeValue: (NSString*)o_attribute_name
1204 if (b_dark_interface && o_titlebar_view) {
1205 VLCMainWindowTitleView *o_tbv = o_titlebar_view;
1207 if ([o_attribute_name isEqualTo: NSAccessibilitySubroleAttribute])
1208 return NSAccessibilityStandardWindowSubrole;
1210 if ([o_attribute_name isEqualTo: NSAccessibilityCloseButtonAttribute])
1211 return [[o_tbv closeButton] cell];
1213 if ([o_attribute_name isEqualTo: NSAccessibilityMinimizeButtonAttribute])
1214 return [[o_tbv minimizeButton] cell];
1216 if ([o_attribute_name isEqualTo: NSAccessibilityZoomButtonAttribute])
1217 return [[o_tbv zoomButton] cell];
1220 return [super accessibilityAttributeValue: o_attribute_name];