1 /*****************************************************************************
2 * vout.m: MacOS X video output module
3 *****************************************************************************
4 * Copyright (C) 2001-2006 the VideoLAN team
7 * Authors: Colin Delacroix <colin@zoy.org>
8 * Florian G. Pflug <fgp@phlo.org>
9 * Jon Lech Johansen <jon-vl@nanocrew.net>
10 * Derk-Jan Hartman <hartman at videolan dot org>
11 * Eric Petit <titer@m0k.org>
12 * Benjamin Pracht <bigben at videolan dot org>
13 * Felix K
\9fhne <fkuehne at videolan dot org>
15 * This program is free software; you can redistribute it and/or modify
16 * it under the terms of the GNU General Public License as published by
17 * the Free Software Foundation; either version 2 of the License, or
18 * (at your option) any later version.
20 * This program is distributed in the hope that it will be useful,
21 * but WITHOUT ANY WARRANTY; without even the implied warranty of
22 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
23 * GNU General Public License for more details.
25 * You should have received a copy of the GNU General Public License
26 * along with this program; if not, write to the Free Software
27 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
28 *****************************************************************************/
30 /*****************************************************************************
32 *****************************************************************************/
33 #include <errno.h> /* ENOMEM */
34 #include <stdlib.h> /* free() */
35 #include <string.h> /* strerror() */
37 /* BeginFullScreen, EndFullScreen */
38 #include <QuickTime/QuickTime.h>
45 /*****************************************************************************
46 * DeviceCallback: Callback triggered when the video-device variable is changed
47 *****************************************************************************/
48 int DeviceCallback( vlc_object_t *p_this, const char *psz_variable,
49 vlc_value_t old_val, vlc_value_t new_val, void *param )
52 vout_thread_t *p_vout = (vout_thread_t *)p_this;
54 msg_Dbg( p_vout, "set %d", new_val.i_int );
55 var_Create( p_vout->p_vlc, "video-device", VLC_VAR_INTEGER );
56 var_Set( p_vout->p_vlc, "video-device", new_val );
58 val.b_bool = VLC_TRUE;
59 var_Set( p_vout, "intf-change", val );
64 /*****************************************************************************
65 * VLCEmbeddedList implementation
66 *****************************************************************************/
67 @implementation VLCEmbeddedList
72 o_embedded_array = [NSMutableArray array];
80 for( i = 0; i < [o_embedded_array count]; i++ )
82 id o_vout_view = [o_embedded_array objectAtIndex: i];
83 if( ![o_vout_view isUsed] )
85 [o_vout_view setUsed: YES];
92 - (void)releaseEmbeddedVout: (id)o_vout_view
94 if( [o_embedded_array containsObject: o_vout_view] )
96 [o_vout_view setUsed: NO];
100 msg_Warn( VLCIntf, "cannot find Video Output");
104 - (void)addEmbeddedVout: (id)o_vout_view
106 if( ![o_embedded_array containsObject: o_vout_view] )
108 [o_embedded_array addObject: o_vout_view];
112 - (BOOL)windowContainsEmbedded: (id)o_window
114 /* if( ![[o_window className] isEqualToString: @"VLCWindow"] )
116 NSLog( @"We were not given a VLCWindow" );
118 return ([self getViewForWindow: o_window] == nil ? NO : YES );
121 - (id)getViewForWindow: (id)o_window
123 id o_enumerator = [o_embedded_array objectEnumerator];
124 id o_current_embedded;
126 while( (o_current_embedded = [o_enumerator nextObject]) )
128 if( [o_current_embedded getWindow] == o_window )
130 return o_current_embedded;
138 /*****************************************************************************
139 * VLCVoutView implementation
140 *****************************************************************************/
141 @implementation VLCVoutView
143 - (id)initWithFrame:(NSRect)frameRect
145 [super initWithFrame: frameRect];
148 s_frame = &frameRect;
155 - (BOOL)setVout: (vout_thread_t *) vout subView: (NSView *) view
156 frame: (NSRect *) frame
159 NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
160 NSArray *o_screens = [NSScreen screens];
166 if( [o_screens count] <= 0 )
168 msg_Err( p_vout, "no OSX screens available" );
172 p_real_vout = [VLCVoutView getRealVout: p_vout];
174 /* Get the pref value when this is the first time, otherwise retrieve the device from the top level video-device var */
175 if( var_Type( p_real_vout->p_vlc, "video-device" ) == 0 )
177 i_device = var_GetInteger( p_vout, "macosx-vdev" );
181 i_device = var_GetInteger( p_real_vout->p_vlc, "video-device" );
184 /* Setup the menuitem for the multiple displays. */
185 if( var_Type( p_real_vout, "video-device" ) == 0 )
188 vlc_value_t val2, text;
191 var_Create( p_real_vout, "video-device", VLC_VAR_INTEGER |
193 text.psz_string = _("Video Device");
194 var_Change( p_real_vout, "video-device", VLC_VAR_SETTEXT, &text, NULL );
196 NSEnumerator * o_enumerator = [o_screens objectEnumerator];
199 text.psz_string = _("Default");
200 var_Change( p_real_vout, "video-device",
201 VLC_VAR_ADDCHOICE, &val2, &text );
202 var_Set( p_real_vout, "video-device", val2 );
204 while( (o_screen = [o_enumerator nextObject]) != NULL )
207 NSRect s_rect = [o_screen frame];
209 snprintf( psz_temp, sizeof(psz_temp)/sizeof(psz_temp[0])-1,
210 "%s %d (%dx%d)", _("Screen"), i,
211 (int)s_rect.size.width, (int)s_rect.size.height );
213 text.psz_string = psz_temp;
215 var_Change( p_real_vout, "video-device",
216 VLC_VAR_ADDCHOICE, &val2, &text );
219 var_Set( p_real_vout, "video-device", val2 );
224 var_AddCallback( p_real_vout, "video-device", DeviceCallback,
227 val2.b_bool = VLC_TRUE;
228 var_Set( p_real_vout, "intf-change", val2 );
231 /* Add the view. It's automatically resized to fit the window */
232 [self addSubview: o_view];
233 [self setAutoresizesSubviews: YES];
239 - (void)resizeSubviewsWithOldSize:(NSSize)oldBoundsSize
241 [super resizeSubviewsWithOldSize: oldBoundsSize];
242 [o_view setFrameSize: [self frame].size];
247 [o_view removeFromSuperview];
257 NSMutableString * o_title = nil, * o_mrl = nil;
258 input_thread_t * p_input;
265 p_input = vlc_object_find( p_vout, VLC_OBJECT_INPUT, FIND_PARENT );
267 if( p_input == NULL )
272 if( p_input->input.p_item->psz_name != NULL )
273 o_title = [NSMutableString stringWithUTF8String:
274 p_input->input.p_item->psz_name];
275 if( p_input->input.p_item->psz_uri != NULL )
276 o_mrl = [NSMutableString stringWithUTF8String:
277 p_input->input.p_item->psz_uri];
283 if( p_input->input.p_access && !strcmp( p_input->input.p_access->p_module->psz_shortname, "File" ) )
285 NSRange prefix_range = [o_mrl rangeOfString: @"file:"];
286 if( prefix_range.location != NSNotFound )
287 [o_mrl deleteCharactersInRange: prefix_range];
288 [o_window setRepresentedFilename: o_mrl];
290 [o_window setTitle: o_title];
294 [o_window setTitle: [NSString stringWithCString: VOUT_TITLE]];
296 vlc_object_release( p_input );
300 - (void)setOnTop:(BOOL)b_on_top
304 [o_window setLevel: NSStatusWindowLevel];
308 [o_window setLevel: NSNormalWindowLevel];
312 - (void)scaleWindowWithFactor: (float)factor
315 int i_corrected_height, i_corrected_width;
317 NSPoint topleftscreen;
319 if ( !p_vout->b_fullscreen )
323 topleftbase.y = [o_window frame].size.height;
324 topleftscreen = [o_window convertBaseToScreen: topleftbase];
326 if( p_vout->render.i_height * p_vout->render.i_aspect >
327 p_vout->render.i_width * VOUT_ASPECT_FACTOR )
329 i_corrected_width = p_vout->render.i_height * p_vout->render.i_aspect /
331 newsize.width = (int) ( i_corrected_width * factor );
332 newsize.height = (int) ( p_vout->render.i_height * factor );
336 i_corrected_height = p_vout->render.i_width * VOUT_ASPECT_FACTOR /
337 p_vout->render.i_aspect;
338 newsize.width = (int) ( p_vout->render.i_width * factor );
339 newsize.height = (int) ( i_corrected_height * factor );
342 /* Calculate the window's new size */
343 new_frame.size.width = [o_window frame].size.width -
344 [self frame].size.width + newsize.width;
345 new_frame.size.height = [o_window frame].size.height -
346 [self frame].size.height + newsize.height;
348 new_frame.origin.x = topleftscreen.x;
349 new_frame.origin.y = topleftscreen.y - new_frame.size.height;
351 [o_window setFrame: new_frame display: YES];
353 p_vout->i_changes |= VOUT_SIZE_CHANGE;
357 - (void)toggleFloatOnTop
361 if( !p_real_vout ) return;
362 if( var_Get( p_real_vout, "video-on-top", &val )>=0 && val.b_bool)
364 val.b_bool = VLC_FALSE;
368 val.b_bool = VLC_TRUE;
370 var_Set( p_real_vout, "video-on-top", val );
373 - (void)toggleFullscreen
376 if( !p_real_vout ) return;
377 var_Get( p_real_vout, "fullscreen", &val );
378 val.b_bool = !val.b_bool;
379 var_Set( p_real_vout, "fullscreen", val );
385 var_Get( p_real_vout, "fullscreen", &val );
386 return( val.b_bool );
391 vout_Control( p_real_vout, VOUT_SNAPSHOT );
396 /* Disable Screensaver */
397 UpdateSystemActivity( UsrActivity );
405 - (void)keyDown:(NSEvent *)o_event
409 unsigned int i_pressed_modifiers = 0;
412 i_pressed_modifiers = [o_event modifierFlags];
414 if( i_pressed_modifiers & NSShiftKeyMask )
415 val.i_int |= KEY_MODIFIER_SHIFT;
416 if( i_pressed_modifiers & NSControlKeyMask )
417 val.i_int |= KEY_MODIFIER_CTRL;
418 if( i_pressed_modifiers & NSAlternateKeyMask )
419 val.i_int |= KEY_MODIFIER_ALT;
420 if( i_pressed_modifiers & NSCommandKeyMask )
421 val.i_int |= KEY_MODIFIER_COMMAND;
423 key = [[o_event charactersIgnoringModifiers] characterAtIndex: 0];
427 /* Escape should always get you out of fullscreen */
428 if( key == (unichar) 0x1b )
430 if( p_real_vout && [self isFullscreen] )
432 [self toggleFullscreen];
435 else if ( key == ' ' )
438 val.i_int = config_GetInt( p_vout, "key-play-pause" );
439 var_Set( p_vout->p_vlc, "key-pressed", val );
443 val.i_int |= CocoaKeyToVLC( key );
444 var_Set( p_vout->p_vlc, "key-pressed", val );
449 [super keyDown: o_event];
453 - (void)mouseDown:(NSEvent *)o_event
459 switch( [o_event type] )
461 case NSLeftMouseDown:
463 if( [o_event clickCount] <= 1 )
465 /* single clicking */
466 var_Get( p_vout, "mouse-button-down", &val );
468 var_Set( p_vout, "mouse-button-down", val );
472 /* multiple clicking */
473 [self toggleFullscreen];
477 case NSRightMouseDown:
479 msg_Dbg( p_vout, "received NSRightMouseDown (generic method)" );
480 [NSMenu popUpContextMenu: [[VLCMain sharedInstance] getVoutMenu] withEvent: o_event forView: [[[VLCMain sharedInstance] getControls] getVoutView]];
485 [super mouseDown: o_event];
491 - (void)otherMouseDown:(NSEvent *)o_event
497 switch( [o_event type] )
499 case NSOtherMouseDown:
501 var_Get( p_vout, "mouse-button-down", &val );
503 var_Set( p_vout, "mouse-button-down", val );
508 [super mouseDown: o_event];
514 - (void)rightMouseDown:(NSEvent *)o_event
518 switch( [o_event type] )
520 case NSRightMouseDown:
522 msg_Dbg( p_vout, "received NSRightMouseDown (specific method)" );
523 [NSMenu popUpContextMenu: [[VLCMain sharedInstance] getVoutMenu] withEvent: o_event forView: [[[VLCMain sharedInstance] getControls] getVoutView]];
528 [super mouseDown: o_event];
534 - (void)mouseUp:(NSEvent *)o_event
540 switch( [o_event type] )
545 b_val.b_bool = VLC_TRUE;
546 var_Set( p_vout, "mouse-clicked", b_val );
548 var_Get( p_vout, "mouse-button-down", &val );
550 var_Set( p_vout, "mouse-button-down", val );
555 [super mouseUp: o_event];
561 - (void)otherMouseUp:(NSEvent *)o_event
567 switch( [o_event type] )
571 var_Get( p_vout, "mouse-button-down", &val );
573 var_Set( p_vout, "mouse-button-down", val );
578 [super mouseUp: o_event];
584 - (void)rightMouseUp:(NSEvent *)o_event
588 switch( [o_event type] )
592 /* FIXME: this isn't the appropriate place, but we can't receive
593 * NSRightMouseDown some how */
594 msg_Dbg( p_vout, "received NSRightMouseUp" );
595 [NSMenu popUpContextMenu: [[VLCMain sharedInstance] getVoutMenu] withEvent: o_event forView: [[[VLCMain sharedInstance] getControls] getVoutView]];
600 [super mouseUp: o_event];
606 - (void)mouseDragged:(NSEvent *)o_event
608 [self mouseMoved: o_event];
611 - (void)otherMouseDragged:(NSEvent *)o_event
613 [self mouseMoved: o_event];
616 - (void)rightMouseDragged:(NSEvent *)o_event
618 [self mouseMoved: o_event];
621 - (void)mouseMoved:(NSEvent *)o_event
629 s_rect = [o_view bounds];
630 ml = [o_view convertPoint: [o_event locationInWindow] fromView: nil];
631 b_inside = [o_view mouse: ml inRect: s_rect];
636 unsigned int i_width, i_height, i_x, i_y;
638 vout_PlacePicture( p_vout, (unsigned int)s_rect.size.width,
639 (unsigned int)s_rect.size.height,
640 &i_x, &i_y, &i_width, &i_height );
642 val.i_int = ( ((int)ml.x) - i_x ) *
643 p_vout->render.i_width / i_width;
644 var_Set( p_vout, "mouse-x", val );
646 if( [[o_view className] isEqualToString: @"VLCGLView"] )
648 val.i_int = ( ((int)(s_rect.size.height - ml.y)) - i_y ) *
649 p_vout->render.i_height / i_height;
653 val.i_int = ( ((int)ml.y) - i_y ) *
654 p_vout->render.i_height / i_height;
656 var_Set( p_vout, "mouse-y", val );
658 val.b_bool = VLC_TRUE;
659 var_Set( p_vout, "mouse-moved", val );
662 [super mouseMoved: o_event];
665 - (BOOL)acceptsFirstResponder
670 - (BOOL)becomeFirstResponder
675 - (BOOL)resignFirstResponder
677 /* We need to stay the first responder or we'll miss some
682 /* Class methods used by the different vout modules */
684 + (vout_thread_t *)getRealVout: (vout_thread_t *)p_vout
686 /* p_real_vout: the vout we have to use to check for video-on-top
687 and a few other things. If we are the QuickTime output, it's us.
688 It we are the OpenGL provider, it is our parent. */
689 if( p_vout->i_object_type == VLC_OBJECT_OPENGL )
691 return (vout_thread_t *) p_vout->p_parent;
700 + (id)getVoutView: (vout_thread_t *)p_vout subView: (NSView *)view
701 frame: (NSRect *)s_frame
703 vlc_value_t value_drawable;
706 vout_thread_t * p_real_vout = [VLCVoutView getRealVout: p_vout];
708 var_Get( p_vout->p_vlc, "drawable", &value_drawable );
710 var_Create( p_vout, "macosx-vdev", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
711 var_Create( p_vout, "macosx-fill", VLC_VAR_BOOL | VLC_VAR_DOINHERIT );
712 var_Create( p_vout, "macosx-stretch", VLC_VAR_BOOL | VLC_VAR_DOINHERIT );
713 var_Create( p_vout, "macosx-opaqueness", VLC_VAR_FLOAT | VLC_VAR_DOINHERIT );
714 var_Create( p_vout, "macosx-background", VLC_VAR_BOOL | VLC_VAR_DOINHERIT );
715 var_Create( p_vout, "macosx-black", VLC_VAR_BOOL | VLC_VAR_DOINHERIT );
716 var_Create( p_vout, "macosx-embedded", VLC_VAR_BOOL | VLC_VAR_DOINHERIT );
719 /* We only wait for NSApp to initialise if we're not embedded (as in the
720 * case of the Mozilla plugin). We can tell whether we're embedded or not
721 * by examining the "drawable" value: if it's zero, we're running in the
722 * main Mac intf; if it's non-zero, we're embedded. */
723 if( value_drawable.i_int == 0 )
725 /* Wait for a MacOS X interface to appear. Timeout is 2 seconds. */
726 for( i_timeout = 20 ; i_timeout-- ; )
730 msleep( INTF_IDLE_SLEEP );
736 /* No MacOS X intf, unable to communicate with MT */
737 msg_Err( p_vout, "no MacOS X interface present" );
742 if ( VLCIntf && !(p_vout->b_fullscreen) &&
743 !(var_GetBool( p_real_vout, "macosx-background" )) &&
744 var_GetBool( p_vout, "macosx-embedded") )
746 o_return = [[[VLCMain sharedInstance] getEmbeddedList]
752 /* No embedded vout is available */
753 if( o_return == nil )
756 bzero( &null_rect, sizeof( NSRect ) );
757 o_return = [[VLCDetachedVoutView alloc] initWithFrame: null_rect ];
759 [o_return setVout: p_vout subView: view frame: s_frame];
765 /*****************************************************************************
766 * VLCDetachedVoutView implementation
767 *****************************************************************************/
768 @implementation VLCDetachedVoutView
770 - (id)initWithFrame: (NSRect)frameRect
772 [super initWithFrame: frameRect];
773 i_time_mouse_last_moved = 0;
777 - (bool)setVout: (vout_thread_t *) p_arg_vout subView: (NSView *) view
778 frame: (NSRect *) s_arg_frame
780 BOOL b_return = [super setVout: p_arg_vout subView: view frame:s_arg_frame];
781 i_time_mouse_last_moved = mdate();
782 o_window = [[VLCWindow alloc] initWithVout: p_arg_vout view: self
785 [view setFrame: [self frame]];
787 if( var_GetBool( p_real_vout, "video-on-top" ) )
789 [o_window setLevel: NSStatusWindowLevel];
793 [o_window setAcceptsMouseMovedEvents: TRUE];
799 [o_window closeWindow];
800 [o_window setAcceptsMouseMovedEvents: NO];
801 i_time_mouse_last_moved = 0;
805 - (void)mouseMoved:(NSEvent *)o_event
807 i_time_mouse_last_moved = mdate();
808 [super mouseMoved: o_event];
811 - (void)hideMouse:(BOOL)b_hide
815 NSView *o_contents = [o_window contentView];
817 ml = [o_window convertScreenToBase:[NSEvent mouseLocation]];
818 ml = [o_contents convertPoint:ml fromView:nil];
819 b_inside = [o_contents mouse: ml inRect: [o_contents bounds]];
821 if( b_hide && b_inside )
823 [NSCursor setHiddenUntilMouseMoves: YES];
827 [NSCursor setHiddenUntilMouseMoves: NO];
834 if( p_vout->b_fullscreen )
836 if( mdate() - i_time_mouse_last_moved > 3000000 )
838 [self hideMouse: YES];
843 [self hideMouse: NO];
849 /*****************************************************************************
850 * VLCEmbeddedVoutView implementation
851 *****************************************************************************/
853 @implementation VLCEmbeddedVoutView
855 - (id)initWithFrame: (NSRect)frameRect
857 [super initWithFrame: frameRect];
859 [[[VLCMain sharedInstance] getEmbeddedList] addEmbeddedVout: self];
863 - (BOOL)setVout: (vout_thread_t *) p_arg_vout subView: (NSView *) view
864 frame: (NSRect *) s_arg_frame
868 b_return = [super setVout: p_arg_vout subView: view frame: s_arg_frame];
871 o_window = [self window];
872 [o_window makeKeyAndOrderFront: self];
873 [o_window setAcceptsMouseMovedEvents: TRUE];
875 if( var_GetBool( p_real_vout, "video-on-top" ) )
877 [o_window setLevel: NSStatusWindowLevel];
880 [view setFrameSize: [self frame].size];
885 - (void)setUsed: (BOOL)b_new_used
898 [o_window setAcceptsMouseMovedEvents: NO];
899 [[[VLCMain sharedInstance] getEmbeddedList] releaseEmbeddedVout: self];
905 @implementation VLCDetachedEmbeddedVoutView
907 - (BOOL)setVout: (vout_thread_t *) p_arg_vout subView: (NSView *) view
908 frame: (NSRect *) s_arg_frame
910 BOOL b_return = [super setVout: p_arg_vout subView: view frame: s_arg_frame];
914 [o_window setAlphaValue: var_GetFloat( p_vout, "macosx-opaqueness" )];
916 [self scaleWindowWithFactor: 1.0];
917 [o_window makeKeyAndOrderFront: self];
924 [o_window orderOut: self];
930 /*****************************************************************************
931 * VLCWindow implementation
932 *****************************************************************************/
933 @implementation VLCWindow
935 - (id) initWithVout: (vout_thread_t *) vout view: (VLCVoutView *) view
936 frame: (NSRect *) frame
942 [self performSelectorOnMainThread: @selector(initReal:)
943 withObject: NULL waitUntilDone: YES];
953 - (id)initReal: (id) sender
955 NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
956 NSArray *o_screens = [NSScreen screens];
958 vlc_bool_t b_menubar_screen = VLC_FALSE;
961 b_init_ok = VLC_FALSE;
963 p_real_vout = [VLCVoutView getRealVout: p_vout];
964 i_device = var_GetInteger( p_real_vout->p_vlc, "video-device" );
965 b_black = var_GetBool( p_real_vout->p_vlc, "macosx-black" );
967 /* Find out on which screen to open the window */
968 if( i_device <= 0 || i_device > (int)[o_screens count] )
970 /* No preference specified. Use the main screen */
971 o_screen = [NSScreen mainScreen];
972 i_device = [o_screens indexOfObject: o_screen];
973 if( o_screen == [o_screens objectAtIndex: 0] )
974 b_menubar_screen = VLC_TRUE;
979 o_screen = [o_screens objectAtIndex: i_device];
980 b_menubar_screen = ( i_device == 0 );
983 if( p_vout->b_fullscreen )
985 CGDisplayFadeReservationToken token;
986 NSRect screen_rect = [o_screen frame];
987 screen_rect.origin.x = screen_rect.origin.y = 0;
989 /* Creates a window with size: screen_rect on o_screen */
990 [self initWithContentRect: screen_rect
991 styleMask: NSBorderlessWindowMask
992 backing: NSBackingStoreBuffered
993 defer: YES screen: o_screen];
995 if( var_GetBool( p_real_vout, "macosx-black" ) )
996 if( b_black == VLC_TRUE )
998 CGAcquireDisplayFadeReservation(kCGMaxDisplayReservationInterval, &token);
999 CGDisplayFade( token, 0.5, kCGDisplayBlendNormal, kCGDisplayBlendSolidColor, 0, 0, 0, true );
1000 CGReleaseDisplayFadeReservation( token );
1002 for( i = 0 ; i < [o_screens count]; i++)
1006 CGDirectDisplayID displayID;
1007 CGGammaValue redMin, redMax, redGamma,
1008 greenMin, greenMax, greenGamma,
1009 blueMin, blueMax, blueGamma;
1011 CGDisplayCount dspyCnt;
1014 if( i == (unsigned int)i_device ) continue;
1016 screen_rect = [[o_screens objectAtIndex: i] frame];
1018 gPoint.x = screen_rect.origin.x;
1019 gPoint.y = screen_rect.origin.y;
1020 CGGetDisplaysWithPoint( gPoint, 1, &(dispSettings.displayID), &dspyCnt);
1021 CGGetDisplayTransferByFormula(
1022 dispSettings.displayID,
1023 &dispSettings.redMin, &dispSettings.redMax, &dispSettings.redGamma,
1024 &dispSettings.greenMin, &dispSettings.greenMax, &dispSettings.greenGamma,
1025 &dispSettings.blueMin, &dispSettings.blueMax, &dispSettings.blueGamma );
1026 CGSetDisplayTransferByFormula(
1027 dispSettings.displayID,
1028 dispSettings.redMin, 0, dispSettings.redGamma,
1029 dispSettings.greenMin, 0, dispSettings.greenGamma,
1030 dispSettings.blueMin, 0, dispSettings.blueGamma );
1033 if( b_menubar_screen )
1035 SetSystemUIMode( kUIModeAllHidden, kUIOptionAutoShowMenuBar);
1037 if( b_black == VLC_TRUE )
1039 CGAcquireDisplayFadeReservation(kCGMaxDisplayReservationInterval, &token);
1040 CGDisplayFade( token, 2 , kCGDisplayBlendSolidColor, kCGDisplayBlendNormal, 0, 0, 0, false );
1041 CGReleaseDisplayFadeReservation( token);
1044 else if( var_GetBool( p_real_vout, "macosx-background" ) )
1046 NSRect screen_rect = [o_screen frame];
1047 screen_rect.origin.x = screen_rect.origin.y = 0;
1049 /* Creates a window with size: screen_rect on o_screen */
1050 [self initWithContentRect: screen_rect
1051 styleMask: NSBorderlessWindowMask
1052 backing: NSBackingStoreBuffered
1053 defer: YES screen: o_screen];
1055 [self setLevel: CGWindowLevelForKey(kCGDesktopWindowLevelKey)];
1059 unsigned int i_stylemask = NSTitledWindowMask |
1060 NSMiniaturizableWindowMask |
1061 NSClosableWindowMask |
1062 NSResizableWindowMask;
1067 s_rect.size.width = p_vout->i_window_width;
1068 s_rect.size.height = p_vout->i_window_height;
1075 [self initWithContentRect: s_rect
1076 styleMask: i_stylemask
1077 backing: NSBackingStoreBuffered
1078 defer: YES screen: o_screen];
1080 [self setAlphaValue: var_GetFloat( p_vout, "macosx-opaqueness" )];
1088 [self makeKeyAndOrderFront: nil];
1089 [self setReleasedWhenClosed: YES];
1091 /* We'll catch mouse events */
1092 [self makeFirstResponder: o_view];
1094 /* Add the view. It's automatically resized to fit the window */
1095 [self setContentView: o_view];
1099 b_init_ok = VLC_TRUE;
1108 - (void) closeWindow
1110 /* XXX waitUntilDone = NO to avoid a possible deadlock when hitting
1112 [self setContentView: NULL];
1113 [self performSelectorOnMainThread: @selector(closeReal:)
1114 withObject: NULL waitUntilDone: NO];
1117 - (id) closeReal: (id) sender
1119 if( b_black == VLC_TRUE )
1121 CGDisplayFadeReservationToken token;
1122 CGAcquireDisplayFadeReservation(kCGMaxDisplayReservationInterval, &token);
1123 CGDisplayFade( token, 2, kCGDisplayBlendSolidColor, kCGDisplayBlendNormal, 0, 0, 0, false );
1124 CGReleaseDisplayFadeReservation( token);
1125 CGDisplayRestoreColorSyncSettings();
1127 SetSystemUIMode( kUIModeNormal, 0);
1137 - (BOOL)canBecomeKeyWindow
1142 /* Sometimes crashes VLC....
1143 - (BOOL)performKeyEquivalent:(NSEvent *)o_event
1145 return [[VLCMain sharedInstance] hasDefinedShortcutKey:o_event];
1148 /* This is actually the same as VLCControls::stop. */
1150 - (BOOL)windowShouldClose:(id)sender
1152 playlist_t * p_playlist = vlc_object_find( p_vout, VLC_OBJECT_PLAYLIST,
1154 if( p_playlist == NULL )
1159 playlist_Stop( p_playlist );
1160 vlc_object_release( p_playlist );
1162 /* The window will be closed by the intf later. */