1 /*****************************************************************************
2 * MainWindow.m: MacOS X interface module
3 *****************************************************************************
4 * Copyright (C) 2002-2012 VLC authors and VideoLAN
7 * Authors: Felix Paul Kühne <fkuehne -at- videolan -dot- org>
8 * Jon Lech Johansen <jon-vl@nanocrew.net>
9 * Christophe Massiot <massiot@via.ecp.fr>
10 * Derk-Jan Hartman <hartman at videolan.org>
12 * This program is free software; you can redistribute it and/or modify
13 * it under the terms of the GNU General Public License as published by
14 * the Free Software Foundation; either version 2 of the License, or
15 * (at your option) any later version.
17 * This program is distributed in the hope that it will be useful,
18 * but WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20 * GNU General Public License for more details.
22 * You should have received a copy of the GNU General Public License
23 * along with this program; if not, write to the Free Software
24 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
25 *****************************************************************************/
27 #import "CompatibilityFixes.h"
28 #import "MainWindow.h"
30 #import "CoreInteraction.h"
31 #import "AudioEffects.h"
34 #import "controls.h" // TODO: remove me
35 #import "SideBarItem.h"
36 #import <vlc_playlist.h>
37 #import <vlc_aout_intf.h>
39 #import <vlc_strings.h>
40 #import <vlc_services_discovery.h>
41 #import <vlc_aout_intf.h>
43 @implementation VLCMainWindow
44 static VLCMainWindow *_o_sharedInstance = nil;
46 + (VLCMainWindow *)sharedInstance
48 return _o_sharedInstance ? _o_sharedInstance : [[self alloc] init];
52 #pragma mark Initialization
56 if( _o_sharedInstance)
59 return _o_sharedInstance;
63 o_fspanel = [[VLCFSPanel alloc] init];
64 _o_sharedInstance = [super init];
67 return _o_sharedInstance;
70 - (id)initWithContentRect:(NSRect)contentRect styleMask:(NSUInteger)styleMask
71 backing:(NSBackingStoreType)backingType defer:(BOOL)flag
73 b_dark_interface = config_GetInt( VLCIntf, "macosx-interfacestyle" );
77 #ifdef MAC_OS_X_VERSION_10_7
78 styleMask = NSBorderlessWindowMask | NSResizableWindowMask;
80 styleMask = NSBorderlessWindowMask;
84 self = [super initWithContentRect:contentRect styleMask:styleMask
85 backing:backingType defer:flag];
87 [[VLCMain sharedInstance] updateTogglePlaylistState];
89 /* we want to be moveable regardless of our style */
90 [self setMovableByWindowBackground: YES];
92 /* we don't want this window to be restored on relaunch */
94 [self setRestorable:NO];
99 - (BOOL)performKeyEquivalent:(NSEvent *)o_event
101 /* We indeed want to prioritize Cocoa key equivalent against libvlc,
102 so we perform the menu equivalent now. */
103 if([[NSApp mainMenu] performKeyEquivalent:o_event])
106 return [[VLCMain sharedInstance] hasDefinedShortcutKey:o_event] || [(VLCControls *)[[VLCMain sharedInstance] controls] keyEvent:o_event];
111 if (b_dark_interface)
112 [o_color_backdrop release];
114 [[NSNotificationCenter defaultCenter] removeObserver: self];
115 [o_sidebaritems release];
121 /* setup the styled interface */
122 #ifndef MAC_OS_X_VERSION_10_7
123 b_nativeFullscreenMode = NO;
125 b_nativeFullscreenMode = config_GetInt( VLCIntf, "macosx-nativefullscreenmode" );
127 i_lastShownVolume = -1;
128 t_hide_mouse_timer = nil;
130 [o_play_btn setToolTip: _NS("Play/Pause")];
131 [o_bwd_btn setToolTip: _NS("Backward")];
132 [o_fwd_btn setToolTip: _NS("Forward")];
133 [o_stop_btn setToolTip: _NS("Stop")];
134 [o_playlist_btn setToolTip: _NS("Show/Hide Playlist")];
135 [o_repeat_btn setToolTip: _NS("Repeat")];
136 [o_shuffle_btn setToolTip: _NS("Shuffle")];
137 [o_effects_btn setToolTip: _NS("Effects")];
138 [o_fullscreen_btn setToolTip: _NS("Toggle Fullscreen mode")];
139 [[o_search_fld cell] setPlaceholderString: _NS("Search")];
140 [o_volume_sld setToolTip: _NS("Volume")];
141 [o_volume_down_btn setToolTip: _NS("Mute")];
142 [o_volume_up_btn setToolTip: _NS("Full Volume")];
143 [o_time_sld setToolTip: _NS("Position")];
144 [o_dropzone_btn setTitle: _NS("Open media...")];
145 [o_dropzone_lbl setStringValue: _NS("Drop media here")];
147 if (!b_dark_interface) {
148 [o_bottombar_view setImage: [NSImage imageNamed:@"bottom-background"]];
149 [o_bwd_btn setImage: [NSImage imageNamed:@"back"]];
150 [o_bwd_btn setAlternateImage: [NSImage imageNamed:@"back-pressed"]];
151 o_play_img = [[NSImage imageNamed:@"play"] retain];
152 o_play_pressed_img = [[NSImage imageNamed:@"play-pressed"] retain];
153 o_pause_img = [[NSImage imageNamed:@"pause"] retain];
154 o_pause_pressed_img = [[NSImage imageNamed:@"pause-pressed"] retain];
155 [o_fwd_btn setImage: [NSImage imageNamed:@"forward"]];
156 [o_fwd_btn setAlternateImage: [NSImage imageNamed:@"forward-pressed"]];
157 [o_stop_btn setImage: [NSImage imageNamed:@"stop"]];
158 [o_stop_btn setAlternateImage: [NSImage imageNamed:@"stop-pressed"]];
159 [o_playlist_btn setImage: [NSImage imageNamed:@"playlist"]];
160 [o_playlist_btn setAlternateImage: [NSImage imageNamed:@"playlist-pressed"]];
161 o_repeat_img = [[NSImage imageNamed:@"repeat"] retain];
162 o_repeat_pressed_img = [[NSImage imageNamed:@"repeat-pressed"] retain];
163 o_repeat_all_img = [[NSImage imageNamed:@"repeat-all"] retain];
164 o_repeat_all_pressed_img = [[NSImage imageNamed:@"repeat-all-pressed"] retain];
165 o_repeat_one_img = [[NSImage imageNamed:@"repeat-one"] retain];
166 o_repeat_one_pressed_img = [[NSImage imageNamed:@"repeat-one-pressed"] retain];
167 o_shuffle_img = [[NSImage imageNamed:@"shuffle"] retain];
168 o_shuffle_pressed_img = [[NSImage imageNamed:@"shuffle-pressed"] retain];
169 o_shuffle_on_img = [[NSImage imageNamed:@"shuffle-blue"] retain];
170 o_shuffle_on_pressed_img = [[NSImage imageNamed:@"shuffle-blue-pressed"] retain];
171 [o_time_sld_background setImagesLeft: [NSImage imageNamed:@"progression-track-wrapper-left"] middle: [NSImage imageNamed:@"progression-track-wrapper-middle"] right: [NSImage imageNamed:@"progression-track-wrapper-right"]];
172 [o_volume_down_btn setImage: [NSImage imageNamed:@"volume-low"]];
173 [o_volume_track_view setImage: [NSImage imageNamed:@"volume-slider-track"]];
174 [o_volume_up_btn setImage: [NSImage imageNamed:@"volume-high"]];
175 if (OSX_LION && b_nativeFullscreenMode)
177 [o_effects_btn setImage: [NSImage imageNamed:@"effects-one-button"]];
178 [o_effects_btn setAlternateImage: [NSImage imageNamed:@"effects-one-button-blue"]];
182 [o_effects_btn setImage: [NSImage imageNamed:@"effects-double-buttons"]];
183 [o_effects_btn setAlternateImage: [NSImage imageNamed:@"effects-double-buttons-pressed"]];
185 [o_fullscreen_btn setImage: [NSImage imageNamed:@"fullscreen-double-buttons"]];
186 [o_fullscreen_btn setAlternateImage: [NSImage imageNamed:@"fullscreen-double-buttons-pressed"]];
187 [o_time_sld_fancygradient_view setImagesLeft:[NSImage imageNamed:@"progression-fill-left"] middle:[NSImage imageNamed:@"progression-fill-middle"] right:[NSImage imageNamed:@"progression-fill-right"]];
191 [o_bottombar_view setImage: [NSImage imageNamed:@"bottom-background_dark"]];
192 [o_bwd_btn setImage: [NSImage imageNamed:@"back_dark"]];
193 [o_bwd_btn setAlternateImage: [NSImage imageNamed:@"back-pressed_dark"]];
194 o_play_img = [[NSImage imageNamed:@"play_dark"] retain];
195 o_play_pressed_img = [[NSImage imageNamed:@"play-pressed_dark"] retain];
196 o_pause_img = [[NSImage imageNamed:@"pause_dark"] retain];
197 o_pause_pressed_img = [[NSImage imageNamed:@"pause-pressed_dark"] retain];
198 [o_fwd_btn setImage: [NSImage imageNamed:@"forward_dark"]];
199 [o_fwd_btn setAlternateImage: [NSImage imageNamed:@"forward-pressed_dark"]];
200 [o_stop_btn setImage: [NSImage imageNamed:@"stop_dark"]];
201 [o_stop_btn setAlternateImage: [NSImage imageNamed:@"stop-pressed_dark"]];
202 [o_playlist_btn setImage: [NSImage imageNamed:@"playlist_dark"]];
203 [o_playlist_btn setAlternateImage: [NSImage imageNamed:@"playlist-pressed_dark"]];
204 o_repeat_img = [[NSImage imageNamed:@"repeat_dark"] retain];
205 o_repeat_pressed_img = [[NSImage imageNamed:@"repeat-pressed_dark"] retain];
206 o_repeat_all_img = [[NSImage imageNamed:@"repeat-all-blue_dark"] retain];
207 o_repeat_all_pressed_img = [[NSImage imageNamed:@"repeat-all-blue-pressed_dark"] retain];
208 o_repeat_one_img = [[NSImage imageNamed:@"repeat-one-blue_dark"] retain];
209 o_repeat_one_pressed_img = [[NSImage imageNamed:@"repeat-one-blue-pressed_dark"] retain];
210 o_shuffle_img = [[NSImage imageNamed:@"shuffle_dark"] retain];
211 o_shuffle_pressed_img = [[NSImage imageNamed:@"shuffle-pressed_dark"] retain];
212 o_shuffle_on_img = [[NSImage imageNamed:@"shuffle-blue_dark"] retain];
213 o_shuffle_on_pressed_img = [[NSImage imageNamed:@"shuffle-blue-pressed_dark"] retain];
214 [o_time_fld setTextColor: [NSColor colorWithCalibratedRed:229.0 green:229.0 blue:229.0 alpha:100.0]];
215 [o_time_sld_background setImagesLeft: [NSImage imageNamed:@"progression-track-wrapper-left_dark"] middle: [NSImage imageNamed:@"progression-track-wrapper-middle_dark"] right: [NSImage imageNamed:@"progression-track-wrapper-right_dark"]];
216 [o_volume_down_btn setImage: [NSImage imageNamed:@"volume-low_dark"]];
217 [o_volume_track_view setImage: [NSImage imageNamed:@"volume-slider-track_dark"]];
218 [o_volume_up_btn setImage: [NSImage imageNamed:@"volume-high_dark"]];
219 if (OSX_LION && b_nativeFullscreenMode)
221 [o_effects_btn setImage: [NSImage imageNamed:@"effects-one-button_dark"]];
222 [o_effects_btn setAlternateImage: [NSImage imageNamed:@"effects-one-button-blue_dark"]];
226 [o_effects_btn setImage: [NSImage imageNamed:@"effects-double-buttons_dark"]];
227 [o_effects_btn setAlternateImage: [NSImage imageNamed:@"effects-double-buttons-pressed_dark"]];
229 [o_fullscreen_btn setImage: [NSImage imageNamed:@"fullscreen-double-buttons_dark"]];
230 [o_fullscreen_btn setAlternateImage: [NSImage imageNamed:@"fullscreen-double-buttons-pressed_dark"]];
231 [o_time_sld_fancygradient_view setImagesLeft:[NSImage imageNamed:@"progressbar-fill-left_dark"] middle:[NSImage imageNamed:@"progressbar-fill-middle_dark"] right:[NSImage imageNamed:@"progressbar-fill-right_dark"]];
233 [o_repeat_btn setImage: o_repeat_img];
234 [o_repeat_btn setAlternateImage: o_repeat_pressed_img];
235 [o_shuffle_btn setImage: o_shuffle_img];
236 [o_shuffle_btn setAlternateImage: o_shuffle_pressed_img];
237 [o_play_btn setImage: o_play_img];
238 [o_play_btn setAlternateImage: o_play_pressed_img];
239 BOOL b_mute = ![[VLCCoreInteraction sharedInstance] isMuted];
240 [o_volume_sld setEnabled: b_mute];
241 [o_volume_up_btn setEnabled: b_mute];
243 /* interface builder action */
244 [self setDelegate: self];
245 [self setExcludedFromWindowsMenu: YES];
246 [self setAcceptsMouseMovedEvents: YES];
247 // Set that here as IB seems to be buggy
248 if (b_dark_interface)
249 [self setContentMinSize:NSMakeSize(500., (288. + [o_titlebar_view frame].size.height))];
251 [self setContentMinSize:NSMakeSize(500., 288.)];
252 [self setTitle: _NS("VLC media player")];
253 [o_playlist_btn setEnabled:NO];
254 o_temp_view = [[NSView alloc] init];
255 [o_temp_view setAutoresizingMask:NSViewHeightSizable | NSViewWidthSizable];
256 [o_dropzone_view setFrame: [o_playlist_table frame]];
257 [o_left_split_view setFrame: [o_sidebar_view frame]];
258 if (OSX_LION && b_nativeFullscreenMode)
260 [self setCollectionBehavior: NSWindowCollectionBehaviorFullScreenPrimary];
262 float f_width = [o_fullscreen_btn frame].size.width;
264 #define moveItem( item ) \
265 frame = [item frame]; \
266 frame.origin.x = f_width + frame.origin.x; \
267 [item setFrame: frame]
269 moveItem( o_effects_btn );
270 moveItem( o_volume_up_btn );
271 moveItem( o_volume_sld );
272 moveItem( o_volume_track_view );
273 moveItem( o_volume_down_btn );
274 moveItem( o_time_fld );
277 #define enlargeItem( item ) \
278 frame = [item frame]; \
279 frame.size.width = f_width + frame.size.width; \
280 [item setFrame: frame]
282 enlargeItem( o_time_sld );
283 enlargeItem( o_progress_bar );
284 enlargeItem( o_time_sld_background );
285 enlargeItem( o_time_sld_fancygradient_view );
288 [o_fullscreen_btn removeFromSuperviewWithoutNeedingDisplay];
291 [o_titlebar_view setFullscreenButtonHidden: YES];
295 /* the default small size of the search field is slightly different on Lion, let's work-around that */
297 frame = [o_search_fld frame];
298 frame.origin.y = frame.origin.y + 2.0;
299 frame.size.height = frame.size.height - 1.0;
300 [o_search_fld setFrame: frame];
303 /* create the sidebar */
304 o_sidebaritems = [[NSMutableArray alloc] init];
305 SideBarItem *libraryItem = [SideBarItem itemWithTitle:_NS("LIBRARY") identifier:@"library"];
306 SideBarItem *playlistItem = [SideBarItem itemWithTitle:_NS("Playlist") identifier:@"playlist"];
307 [playlistItem setIcon: [NSImage imageNamed:@"sidebar-playlist"]];
308 SideBarItem *mycompItem = [SideBarItem itemWithTitle:_NS("MY COMPUTER") identifier:@"mycomputer"];
309 SideBarItem *devicesItem = [SideBarItem itemWithTitle:_NS("DEVICES") identifier:@"devices"];
310 SideBarItem *lanItem = [SideBarItem itemWithTitle:_NS("LOCAL NETWORK") identifier:@"localnetwork"];
311 SideBarItem *internetItem = [SideBarItem itemWithTitle:_NS("INTERNET") identifier:@"internet"];
313 /* SD subnodes, inspired by the Qt4 intf */
314 char **ppsz_longnames;
316 char **ppsz_names = vlc_sd_GetNames( pl_Get( VLCIntf ), &ppsz_longnames, &p_categories );
318 msg_Err( VLCIntf, "no sd item found" ); //TODO
319 char **ppsz_name = ppsz_names, **ppsz_longname = ppsz_longnames;
320 int *p_category = p_categories;
321 NSMutableArray *internetItems = [[NSMutableArray alloc] init];
322 NSMutableArray *devicesItems = [[NSMutableArray alloc] init];
323 NSMutableArray *lanItems = [[NSMutableArray alloc] init];
324 NSMutableArray *mycompItems = [[NSMutableArray alloc] init];
325 NSString *o_identifier;
326 for (; *ppsz_name; ppsz_name++, ppsz_longname++, p_category++)
328 o_identifier = [NSString stringWithCString: *ppsz_name encoding: NSUTF8StringEncoding];
329 switch (*p_category) {
330 case SD_CAT_INTERNET:
332 [internetItems addObject: [SideBarItem itemWithTitle: [NSString stringWithCString: *ppsz_longname encoding: NSUTF8StringEncoding] identifier: o_identifier]];
333 if (!strncmp( *ppsz_name, "podcast", 7 ))
334 [[internetItems lastObject] setIcon: [NSImage imageNamed:@"sidebar-podcast"]];
336 [[internetItems lastObject] setIcon: [NSImage imageNamed:@"NSApplicationIcon"]];
337 [[internetItems lastObject] setSdtype: SD_CAT_INTERNET];
342 [devicesItems addObject: [SideBarItem itemWithTitle: [NSString stringWithCString: *ppsz_longname encoding: NSUTF8StringEncoding] identifier: o_identifier]];
343 [[devicesItems lastObject] setIcon: [NSImage imageNamed:@"NSApplicationIcon"]];
344 [[devicesItems lastObject] setSdtype: SD_CAT_DEVICES];
349 [lanItems addObject: [SideBarItem itemWithTitle: [NSString stringWithCString: *ppsz_longname encoding: NSUTF8StringEncoding] identifier: o_identifier]];
350 [[lanItems lastObject] setIcon: [NSImage imageNamed:@"sidebar-local"]];
351 [[lanItems lastObject] setSdtype: SD_CAT_LAN];
354 case SD_CAT_MYCOMPUTER:
356 [mycompItems addObject: [SideBarItem itemWithTitle: [NSString stringWithCString: *ppsz_longname encoding: NSUTF8StringEncoding] identifier: o_identifier]];
357 if (!strncmp( *ppsz_name, "video_dir", 9 ))
358 [[mycompItems lastObject] setIcon: [NSImage imageNamed:@"sidebar-movie"]];
359 else if (!strncmp( *ppsz_name, "audio_dir", 9 ))
360 [[mycompItems lastObject] setIcon: [NSImage imageNamed:@"sidebar-music"]];
361 else if (!strncmp( *ppsz_name, "picture_dir", 11 ))
362 [[mycompItems lastObject] setIcon: [NSImage imageNamed:@"sidebar-pictures"]];
364 [[mycompItems lastObject] setIcon: [NSImage imageNamed:@"NSApplicationIcon"]];
365 [[mycompItems lastObject] setSdtype: SD_CAT_MYCOMPUTER];
369 msg_Warn( VLCIntf, "unknown SD type found, skipping (%s)", *ppsz_name );
374 free( *ppsz_longname );
376 [mycompItem setChildren: [NSArray arrayWithArray: mycompItems]];
377 [devicesItem setChildren: [NSArray arrayWithArray: devicesItems]];
378 [lanItem setChildren: [NSArray arrayWithArray: lanItems]];
379 [internetItem setChildren: [NSArray arrayWithArray: internetItems]];
380 [mycompItems release];
381 [devicesItems release];
383 [internetItems release];
385 free( ppsz_longnames );
386 free( p_categories );
388 [libraryItem setChildren: [NSArray arrayWithObject: playlistItem]];
389 [o_sidebaritems addObject: libraryItem];
390 if ([mycompItem hasChildren])
391 [o_sidebaritems addObject: mycompItem];
392 if ([devicesItem hasChildren])
393 [o_sidebaritems addObject: devicesItem];
394 if ([lanItem hasChildren])
395 [o_sidebaritems addObject: lanItem];
396 if ([internetItem hasChildren])
397 [o_sidebaritems addObject: internetItem];
399 [o_sidebar_view reloadData];
400 [o_sidebar_view selectRowIndexes:[NSIndexSet indexSetWithIndex:0] byExtendingSelection:YES];
401 NSUInteger i_sidebaritem_count = [o_sidebaritems count];
402 for (NSUInteger x = 0; x < i_sidebaritem_count; x++)
403 [o_sidebar_view expandItem: [o_sidebaritems objectAtIndex: x] expandChildren: YES];
405 if( b_dark_interface )
407 [[NSNotificationCenter defaultCenter] addObserver: self selector: @selector(windowResizedOrMoved:) name: NSWindowDidResizeNotification object: nil];
408 [[NSNotificationCenter defaultCenter] addObserver: self selector: @selector(windowResizedOrMoved:) name: NSWindowDidMoveNotification object: nil];
410 [self setBackgroundColor: [NSColor clearColor]];
411 [self setOpaque: NO];
412 [self setHasShadow:YES];
415 CGFloat f_titleBarHeight = [o_titlebar_view frame].size.height;
416 winrect = [self frame];
418 [o_titlebar_view setFrame: NSMakeRect( 0, winrect.size.height - f_titleBarHeight,
419 winrect.size.width, f_titleBarHeight )];
420 [[self contentView] addSubview: o_titlebar_view];
422 winrect.size.height = winrect.size.height + f_titleBarHeight;
423 [self setFrame: winrect display:NO animate:NO];
424 winrect = [o_split_view frame];
425 winrect.size.height = winrect.size.height - f_titleBarHeight;
426 [o_split_view setFrame: winrect];
427 [o_video_view setFrame: winrect];
429 o_color_backdrop = [[VLCColorView alloc] initWithFrame: [o_split_view frame]];
430 [[self contentView] addSubview: o_color_backdrop positioned: NSWindowBelow relativeTo: o_split_view];
431 [o_color_backdrop setAutoresizingMask:NSViewHeightSizable | NSViewWidthSizable];
433 previousSavedFrame = winrect;
440 frame = [o_time_sld_fancygradient_view frame];
441 frame.size.height = frame.size.height - 1;
442 frame.origin.y = frame.origin.y + 1;
443 [o_time_sld_fancygradient_view setFrame: frame];
445 [o_video_view setFrame: [o_split_view frame]];
446 [o_playlist_table setBorderType: NSNoBorder];
447 [o_sidebar_scrollview setBorderType: NSNoBorder];
451 [o_resize_view setImage: NULL];
453 if ([self styleMask] & NSResizableWindowMask)
454 [o_resize_view removeFromSuperviewWithoutNeedingDisplay];
457 [o_time_sld_fancygradient_view removeFromSuperviewWithoutNeedingDisplay];
459 [[NSNotificationCenter defaultCenter] addObserver: self selector: @selector(someWindowWillClose:) name: NSWindowWillCloseNotification object: nil];
460 [[NSNotificationCenter defaultCenter] addObserver: self selector: @selector(someWindowWillMiniaturize:) name: NSWindowWillMiniaturizeNotification object:nil];
461 [[NSNotificationCenter defaultCenter] addObserver: self selector: @selector(applicationWillTerminate:) name: NSApplicationWillTerminateNotification object: nil];
465 #pragma mark Button Actions
467 - (IBAction)play:(id)sender
469 [[VLCCoreInteraction sharedInstance] play];
472 - (void)resetPreviousButton
474 if (([NSDate timeIntervalSinceReferenceDate] - last_bwd_event) >= 0.35) {
475 // seems like no further event occured, so let's switch the playback item
476 [[VLCCoreInteraction sharedInstance] previous];
477 just_triggered_previous = NO;
481 - (void)resetBackwardSkip
483 // the user stopped skipping, so let's allow him to change the item
484 if (([NSDate timeIntervalSinceReferenceDate] - last_bwd_event) >= 0.35)
485 just_triggered_previous = NO;
488 - (IBAction)bwd:(id)sender
490 if(!just_triggered_previous)
492 just_triggered_previous = YES;
493 [self performSelector:@selector(resetPreviousButton)
499 if (([NSDate timeIntervalSinceReferenceDate] - last_fwd_event) > 0.12 )
501 // we just skipped 3 "continous" events, otherwise we are too fast
502 [[VLCCoreInteraction sharedInstance] backward];
503 last_bwd_event = [NSDate timeIntervalSinceReferenceDate];
504 [self performSelector:@selector(resetBackwardSkip)
511 - (void)resetNextButton
513 if (([NSDate timeIntervalSinceReferenceDate] - last_fwd_event) >= 0.35) {
514 // seems like no further event occured, so let's switch the playback item
515 [[VLCCoreInteraction sharedInstance] next];
516 just_triggered_next = NO;
520 - (void)resetForwardSkip
522 // the user stopped skipping, so let's allow him to change the item
523 if (([NSDate timeIntervalSinceReferenceDate] - last_fwd_event) >= 0.35)
524 just_triggered_next = NO;
527 - (IBAction)fwd:(id)sender
529 if(!just_triggered_next)
531 just_triggered_next = YES;
532 [self performSelector:@selector(resetNextButton)
538 if (([NSDate timeIntervalSinceReferenceDate] - last_fwd_event) > 0.12 )
540 // we just skipped 3 "continous" events, otherwise we are too fast
541 [[VLCCoreInteraction sharedInstance] forward];
542 last_fwd_event = [NSDate timeIntervalSinceReferenceDate];
543 [self performSelector:@selector(resetForwardSkip)
550 - (IBAction)stop:(id)sender
552 [[VLCCoreInteraction sharedInstance] stop];
555 - (IBAction)togglePlaylist:(id)sender
559 if ([o_video_view isHidden] && [o_playlist_btn isEnabled]) {
560 [o_split_view setHidden: YES];
561 [o_video_view setHidden: NO];
562 [self makeFirstResponder: o_video_view];
566 [o_video_view setHidden: YES];
567 [o_split_view setHidden: NO];
568 [self makeFirstResponder: nil];
573 [o_split_view setHidden: NO];
574 [o_playlist_table setHidden: NO];
575 [o_video_view setHidden: ![[VLCMain sharedInstance] activeVideoPlayback]];
581 [o_repeat_btn setImage: o_repeat_one_img];
582 [o_repeat_btn setAlternateImage: o_repeat_one_pressed_img];
587 [o_repeat_btn setImage: o_repeat_all_img];
588 [o_repeat_btn setAlternateImage: o_repeat_all_pressed_img];
593 [o_repeat_btn setImage: o_repeat_img];
594 [o_repeat_btn setAlternateImage: o_repeat_pressed_img];
597 - (IBAction)repeat:(id)sender
599 vlc_value_t looping,repeating;
600 intf_thread_t * p_intf = VLCIntf;
601 playlist_t * p_playlist = pl_Get( p_intf );
603 var_Get( p_playlist, "repeat", &repeating );
604 var_Get( p_playlist, "loop", &looping );
606 if( !repeating.b_bool && !looping.b_bool )
608 /* was: no repeating at all, switching to Repeat One */
609 [[VLCCoreInteraction sharedInstance] repeatOne];
612 else if( repeating.b_bool && !looping.b_bool )
614 /* was: Repeat One, switching to Repeat All */
615 [[VLCCoreInteraction sharedInstance] repeatAll];
620 /* was: Repeat All or bug in VLC, switching to Repeat Off */
621 [[VLCCoreInteraction sharedInstance] repeatOff];
629 playlist_t *p_playlist = pl_Get( VLCIntf );
630 b_value = var_GetBool( p_playlist, "random" );
632 [o_shuffle_btn setImage: o_shuffle_on_img];
633 [o_shuffle_btn setAlternateImage: o_shuffle_on_pressed_img];
637 [o_shuffle_btn setImage: o_shuffle_img];
638 [o_shuffle_btn setAlternateImage: o_shuffle_pressed_img];
642 - (IBAction)shuffle:(id)sender
644 [[VLCCoreInteraction sharedInstance] shuffle];
648 - (IBAction)timeSliderAction:(id)sender
651 input_thread_t * p_input;
653 switch( [[NSApp currentEvent] type] )
656 case NSLeftMouseDown:
657 case NSLeftMouseDragged:
658 f_updated = [sender floatValue];
664 p_input = pl_CurrentInput( VLCIntf );
665 if( p_input != NULL )
670 char psz_time[MSTRTIME_MAX_SIZE];
672 pos.f_float = f_updated / 10000.;
673 var_Set( p_input, "position", pos );
674 [o_time_sld setFloatValue: f_updated];
676 var_Get( p_input, "time", &time );
678 mtime_t dur = input_item_GetDuration( input_GetItem( p_input ) );
679 if( [o_time_fld timeRemaining] && dur != -1 )
681 o_time = [NSString stringWithFormat: @"-%s", secstotimestr( psz_time, ((dur - time.i_time) / 1000000) )];
684 o_time = [NSString stringWithUTF8String: secstotimestr( psz_time, (time.i_time / 1000000) )];
686 [o_time_fld setStringValue: o_time];
687 [o_fspanel setStreamPos: f_updated andTime: o_time];
688 vlc_object_release( p_input );
690 [self drawFancyGradientEffectForTimeSlider];
693 - (IBAction)volumeAction:(id)sender
695 if (sender == o_volume_sld)
696 [[VLCCoreInteraction sharedInstance] setVolume: [sender intValue]];
697 else if (sender == o_volume_down_btn)
699 [[VLCCoreInteraction sharedInstance] mute];
700 [o_volume_sld setIntValue: 0];
701 BOOL b_mute = ![[VLCCoreInteraction sharedInstance] isMuted];
702 [o_volume_sld setEnabled: b_mute];
703 [o_volume_up_btn setEnabled: b_mute];
706 [[VLCCoreInteraction sharedInstance] setVolume: AOUT_VOLUME_MAX];
709 - (IBAction)effects:(id)sender
711 [[VLCMainMenu sharedInstance] showAudioEffects: sender];
714 - (IBAction)fullscreen:(id)sender
716 [[VLCCoreInteraction sharedInstance] toggleFullscreen];
719 - (IBAction)dropzoneButtonAction:(id)sender
721 [[[VLCMain sharedInstance] open] openFileGeneric];
725 #pragma mark overwritten default functionality
726 - (BOOL)canBecomeKeyWindow
731 - (BOOL)validateMenuItem:(NSMenuItem *)menuItem
733 SEL s_menuAction = [menuItem action];
734 if ((s_menuAction == @selector(performClose:)) || (s_menuAction == @selector(performMiniaturize:)) || (s_menuAction == @selector(performZoom:)))
737 return [super validateMenuItem:menuItem];
745 - (void)setTitle:(NSString *)title
747 if (b_dark_interface)
748 [o_titlebar_view setWindowTitle: title];
749 if (b_nonembedded && [[VLCMain sharedInstance] activeVideoPlayback])
750 [o_nonembedded_window setTitle: title];
751 [super setTitle: title];
754 - (void)performClose:(id)sender
756 if (b_dark_interface)
758 [self orderOut: sender];
759 [[VLCCoreInteraction sharedInstance] stop];
762 [super performClose: sender];
765 - (void)performMiniaturize:(id)sender
767 if (b_dark_interface)
769 [self miniaturize: sender];
770 if ([[VLCMain sharedInstance] activeVideoPlayback])
771 [[VLCCoreInteraction sharedInstance] pause];
774 [super performMiniaturize: sender];
777 - (void)performZoom:(id)sender
779 if (b_dark_interface)
780 [self customZoom: sender];
782 [super performZoom: sender];
785 - (void)zoom:(id)sender
787 if (b_dark_interface)
788 [self customZoom: sender];
790 [super zoom: sender];
794 * Given a proposed frame rectangle, return a modified version
795 * which will fit inside the screen.
797 * This method is based upon NSWindow.m, part of the GNUstep GUI Library, licensed under LGPLv2+.
798 * Authors: Scott Christley <scottc@net-community.com>, Venkat Ajjanagadde <venkat@ocbi.com>,
799 * Felipe A. Rodriguez <far@ix.netcom.com>, Richard Frith-Macdonald <richard@brainstorm.co.uk>
800 * Copyright (C) 1996 Free Software Foundation, Inc.
802 - (NSRect) customConstrainFrameRect: (NSRect)frameRect toScreen: (NSScreen*)screen
804 NSRect screenRect = [screen visibleFrame];
807 /* Move top edge of the window inside the screen */
808 difference = NSMaxY (frameRect) - NSMaxY (screenRect);
811 frameRect.origin.y -= difference;
814 /* If the window is resizable, resize it (if needed) so that the
815 bottom edge is on the screen or can be on the screen when the user moves
817 difference = NSMaxY (screenRect) - NSMaxY (frameRect);
818 if (_styleMask & NSResizableWindowMask)
822 difference2 = screenRect.origin.y - frameRect.origin.y;
823 difference2 -= difference;
824 // Take in account the space between the top of window and the top of the
825 // screen which can be used to move the bottom of the window on the screen
828 frameRect.size.height -= difference2;
829 frameRect.origin.y += difference2;
832 /* Ensure that resizing doesn't makewindow smaller than minimum */
833 difference2 = [self minSize].height - frameRect.size.height;
836 frameRect.size.height += difference2;
837 frameRect.origin.y -= difference2;
847 Zooms the receiver. This method calls the delegate method
848 windowShouldZoom:toFrame: to determine if the window should
849 be allowed to zoom to full screen.
851 * This method is based upon NSWindow.m, part of the GNUstep GUI Library, licensed under LGPLv2+.
852 * Authors: Scott Christley <scottc@net-community.com>, Venkat Ajjanagadde <venkat@ocbi.com>,
853 * Felipe A. Rodriguez <far@ix.netcom.com>, Richard Frith-Macdonald <richard@brainstorm.co.uk>
854 * Copyright (C) 1996 Free Software Foundation, Inc.
856 - (void) customZoom: (id)sender
858 NSRect maxRect = [[self screen] visibleFrame];
859 NSRect currentFrame = [self frame];
861 if ([[self delegate] respondsToSelector: @selector(windowWillUseStandardFrame:defaultFrame:)])
863 maxRect = [[self delegate] windowWillUseStandardFrame: self defaultFrame: maxRect];
866 maxRect = [self customConstrainFrameRect: maxRect toScreen: [self screen]];
868 // Compare the new frame with the current one
869 if ((abs(NSMaxX(maxRect) - NSMaxX(currentFrame)) < DIST)
870 && (abs(NSMaxY(maxRect) - NSMaxY(currentFrame)) < DIST)
871 && (abs(NSMinX(maxRect) - NSMinX(currentFrame)) < DIST)
872 && (abs(NSMinY(maxRect) - NSMinY(currentFrame)) < DIST))
874 // Already in zoomed mode, reset user frame, if stored
875 if ([self frameAutosaveName] != nil)
877 [self setFrame: previousSavedFrame display: YES animate: YES];
878 [self saveFrameUsingName: [self frameAutosaveName]];
883 if ([self frameAutosaveName] != nil)
885 [self saveFrameUsingName: [self frameAutosaveName]];
886 previousSavedFrame = [self frame];
889 [self setFrame: maxRect display: YES animate: YES];
892 - (void)windowResizedOrMoved:(NSNotification *)notification
894 [self saveFrameUsingName: [self frameAutosaveName]];
897 - (void)applicationWillTerminate:(NSNotification *)notification
899 config_PutInt( VLCIntf->p_libvlc, "volume", i_lastShownVolume );
900 [self saveFrameUsingName: [self frameAutosaveName]];
903 - (void)someWindowWillClose:(NSNotification *)notification
905 if([notification object] == o_nonembedded_window || [notification object] == self)
906 [[VLCCoreInteraction sharedInstance] stop];
909 - (void)someWindowWillMiniaturize:(NSNotification *)notification
911 if([notification object] == o_nonembedded_window || [notification object] == self)
913 if([[VLCMain sharedInstance] activeVideoPlayback])
914 [[VLCCoreInteraction sharedInstance] pause];
919 #pragma mark Update interface and respond to foreign events
922 [o_right_split_view addSubview: o_dropzone_view];
923 [o_dropzone_view setFrame: [o_playlist_table frame]];
924 [[o_playlist_table animator] setHidden:YES];
929 [o_dropzone_view removeFromSuperview];
930 [[o_playlist_table animator] setHidden: NO];
933 - (void)updateTimeSlider
935 input_thread_t * p_input;
936 p_input = pl_CurrentInput( VLCIntf );
942 char psz_time[MSTRTIME_MAX_SIZE];
945 var_Get( p_input, "position", &pos );
946 f_updated = 10000. * pos.f_float;
947 [o_time_sld setFloatValue: f_updated];
949 var_Get( p_input, "time", &time );
951 mtime_t dur = input_item_GetDuration( input_GetItem( p_input ) );
952 if( [o_time_fld timeRemaining] && dur != -1 )
954 o_time = [NSString stringWithFormat: @"-%s", secstotimestr( psz_time, ((dur - time.i_time) / 1000000))];
957 o_time = [NSString stringWithUTF8String: secstotimestr( psz_time, (time.i_time / 1000000) )];
960 [o_time_sld setEnabled: NO];
961 [o_time_sld setHidden: YES];
963 [o_time_sld setEnabled: YES];
964 [o_time_sld setHidden: NO];
967 [o_time_fld setStringValue: o_time];
968 [o_time_fld setNeedsDisplay:YES];
969 [o_fspanel setStreamPos: f_updated andTime: o_time];
970 vlc_object_release( p_input );
974 [o_time_sld setFloatValue: 0.0];
975 [o_time_fld setStringValue: @"00:00"];
976 [o_time_sld setEnabled: NO];
977 [o_time_sld setHidden: YES];
980 [self performSelectorOnMainThread:@selector(drawFancyGradientEffectForTimeSlider) withObject:nil waitUntilDone:NO];
983 - (void)updateVolumeSlider
985 audio_volume_t i_volume;
986 playlist_t * p_playlist = pl_Get( VLCIntf );
988 i_volume = aout_VolumeGet( p_playlist );
990 if( i_volume != i_lastShownVolume )
992 [o_volume_sld setIntValue: i_volume];
993 [o_fspanel setVolumeLevel: i_volume];
999 NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
1000 input_thread_t * p_input;
1001 p_input = pl_CurrentInput( VLCIntf );
1005 char *format = var_InheritString( VLCIntf, "input-title-format" );
1006 char *formated = str_format_meta( p_input, format );
1008 aString = [NSString stringWithUTF8String:formated];
1011 char *uri = input_item_GetURI( input_GetItem( p_input ) );
1013 NSURL * o_url = [NSURL URLWithString: [NSString stringWithUTF8String: uri]];
1014 if ([o_url isFileURL])
1015 [self setRepresentedURL: o_url];
1017 [self setRepresentedURL: nil];
1020 if ([aString isEqualToString:@""])
1022 if ([o_url isFileURL])
1023 aString = [[NSFileManager defaultManager] displayNameAtPath: [o_url path]];
1025 aString = [o_url absoluteString];
1028 [self setTitle: aString];
1029 [o_fspanel setStreamTitle: aString];
1033 [self setTitle: _NS("VLC media player")];
1034 [self setRepresentedURL: nil];
1040 - (void)updateWindow
1042 bool b_input = false;
1043 bool b_plmul = false;
1044 bool b_control = false;
1045 bool b_seekable = false;
1046 bool b_chapters = false;
1048 playlist_t * p_playlist = pl_Get( VLCIntf );
1051 b_plmul = playlist_CurrentSize( p_playlist ) > 1;
1054 input_thread_t * p_input = playlist_CurrentInput( p_playlist );
1056 bool b_buffering = NO;
1058 if( ( b_input = ( p_input != NULL ) ) )
1060 /* seekable streams */
1061 cachedInputState = input_GetState( p_input );
1062 if ( cachedInputState == INIT_S || cachedInputState == OPENING_S )
1065 /* seekable streams */
1066 b_seekable = var_GetBool( p_input, "can-seek" );
1068 /* check whether slow/fast motion is possible */
1069 b_control = var_GetBool( p_input, "can-rate" );
1071 /* chapters & titles */
1072 //FIXME! b_chapters = p_input->stream.i_area_nb > 1;
1074 if (cachedInputState == PLAYING_S || b_buffering == YES)
1075 [self makeKeyAndOrderFront: nil];
1076 vlc_object_release( p_input );
1081 [o_progress_bar startAnimation:self];
1082 [o_progress_bar setIndeterminate:YES];
1083 [o_progress_bar setHidden:NO];
1085 [o_progress_bar stopAnimation:self];
1086 [o_progress_bar setHidden:YES];
1089 [o_stop_btn setEnabled: b_input];
1090 [o_fwd_btn setEnabled: (b_seekable || b_plmul || b_chapters)];
1091 [o_bwd_btn setEnabled: (b_seekable || b_plmul || b_chapters)];
1092 [[VLCMainMenu sharedInstance] setRateControlsEnabled: b_control];
1094 [o_time_sld setEnabled: b_seekable];
1095 [self updateTimeSlider];
1096 [o_fspanel setSeekable: b_seekable];
1099 if (p_playlist->items.i_size >= 1)
1100 [self hideDropZone];
1102 [self showDropZone];
1104 [o_sidebar_view setNeedsDisplay:YES];
1109 [o_play_btn setImage: o_pause_img];
1110 [o_play_btn setAlternateImage: o_pause_pressed_img];
1111 [o_play_btn setToolTip: _NS("Pause")];
1112 [o_fspanel setPause];
1117 [o_play_btn setImage: o_play_img];
1118 [o_play_btn setAlternateImage: o_play_pressed_img];
1119 [o_play_btn setToolTip: _NS("Play")];
1120 [o_fspanel setPlay];
1123 - (void)drawFancyGradientEffectForTimeSlider
1128 NSAutoreleasePool * o_pool = [[NSAutoreleasePool alloc] init];
1129 CGFloat f_value = [o_time_sld knobPosition];
1132 NSRect oldFrame = [o_time_sld_fancygradient_view frame];
1133 if (f_value != oldFrame.size.width)
1135 [o_time_sld_fancygradient_view setHidden: NO];
1136 [o_time_sld_fancygradient_view setFrame: NSMakeRect( oldFrame.origin.x, oldFrame.origin.y, f_value, oldFrame.size.height )];
1137 [o_time_sld_fancygradient_view setNeedsDisplay:YES];
1142 [o_time_sld_fancygradient_view setHidden: YES];
1148 #pragma mark Video Output handling
1152 vout_thread_t *p_vout = getVout();
1153 if (config_GetInt( VLCIntf, "embedded-video" ))
1155 if ([o_video_view window] != self)
1157 [o_video_view removeFromSuperviewWithoutNeedingDisplay];
1158 [o_video_view setFrame: [o_split_view frame]];
1159 [[self contentView] addSubview:o_video_view positioned:NSWindowAbove relativeTo:nil];
1165 if ([o_video_view superview] != NULL)
1166 [o_video_view removeFromSuperviewWithoutNeedingDisplay];
1167 if (o_nonembedded_window)
1168 [o_nonembedded_window release];
1170 o_nonembedded_window = [[VLCWindow alloc] initWithContentRect:[o_video_view frame] styleMask: NSTitledWindowMask|NSClosableWindowMask|NSResizableWindowMask|NSMiniaturizableWindowMask backing:NSBackingStoreBuffered defer:YES];
1171 [o_nonembedded_window setFrame:[o_video_view frame] display:NO];
1172 [o_nonembedded_window setBackgroundColor: [NSColor blackColor]];
1173 [o_nonembedded_window setMovableByWindowBackground: YES];
1174 [o_nonembedded_window setCanBecomeKeyWindow: YES];
1175 [o_nonembedded_window setHasShadow:YES];
1176 [o_nonembedded_window setContentView: o_video_view];
1177 [o_nonembedded_window setLevel:NSNormalWindowLevel];
1178 [o_nonembedded_window useOptimizedDrawing: YES];
1179 [o_nonembedded_window center];
1180 [o_nonembedded_window makeKeyAndOrderFront:self];
1181 [o_nonembedded_window orderFront:self animate:YES];
1182 [o_nonembedded_window setReleasedWhenClosed:NO];
1183 b_nonembedded = YES;
1188 if( var_GetBool( p_vout, "video-on-top" ) )
1189 [[o_video_view window] setLevel: NSStatusWindowLevel];
1191 [[o_video_view window] setLevel: NSNormalWindowLevel];
1192 vlc_object_release( p_vout );
1194 return o_video_view;
1197 - (void)setVideoplayEnabled
1199 BOOL b_videoPlayback = [[VLCMain sharedInstance] activeVideoPlayback];
1202 [o_playlist_btn setEnabled: b_videoPlayback];
1205 [o_playlist_btn setEnabled: NO];
1206 if (!b_videoPlayback)
1207 [o_nonembedded_window orderOut: nil];
1209 if( OSX_LION && b_nativeFullscreenMode )
1211 if( [NSApp presentationOptions] & NSApplicationPresentationFullScreen )
1212 [o_bottombar_view setHidden: b_videoPlayback];
1214 [o_bottombar_view setHidden: NO];
1215 if (!b_videoPlayback)
1216 [o_fspanel setNonActive: nil];
1218 if (b_videoPlayback)
1219 [self makeFirstResponder: o_video_view];
1221 [self makeFirstResponder: nil];
1223 if (!b_videoPlayback && b_fullscreen && !b_nativeFullscreenMode)
1224 [self leaveFullscreenAndFadeOut: YES];
1227 - (void)resizeWindow
1229 if ( !b_fullscreen && !(OSX_LION && [NSApp presentationOptions] == NSApplicationPresentationFullScreen && b_nativeFullscreenMode) )
1231 NSPoint topleftbase;
1232 NSPoint topleftscreen;
1235 topleftbase.y = [self frame].size.height;
1236 topleftscreen = [self convertBaseToScreen: topleftbase];
1238 /* Calculate the window's new size */
1239 new_frame.size.width = [self frame].size.width - [o_video_view frame].size.width + nativeVideoSize.width;
1240 if (b_dark_interface)
1241 new_frame.size.height = [self frame].size.height - [o_video_view frame].size.height + nativeVideoSize.height + [o_titlebar_view frame].size.height;
1243 new_frame.size.height = [self frame].size.height - [o_video_view frame].size.height + nativeVideoSize.height;
1245 new_frame.origin.x = topleftscreen.x;
1246 new_frame.origin.y = topleftscreen.y - new_frame.size.height;
1248 [[self animator] setFrame:new_frame display:YES];
1252 - (void)setNativeVideoSize:(NSSize)size
1254 if (size.width != nativeVideoSize.width || size.height != nativeVideoSize.height )
1256 nativeVideoSize = size;
1257 [self resizeWindow];
1261 // Called automatically if window's acceptsMouseMovedEvents property is true
1262 - (void)mouseMoved:(NSEvent *)theEvent
1265 [self recreateHideMouseTimer];
1267 [super mouseMoved: theEvent];
1270 - (void)recreateHideMouseTimer
1272 if (t_hide_mouse_timer != nil) {
1273 [t_hide_mouse_timer invalidate];
1274 [t_hide_mouse_timer release];
1277 t_hide_mouse_timer = [NSTimer scheduledTimerWithTimeInterval:2
1279 selector:@selector(hideMouseCursor:)
1282 [t_hide_mouse_timer retain];
1285 // NSTimer selectors require this function signature as per Apple's docs
1286 - (void)hideMouseCursor:(NSTimer *)timer
1288 [NSCursor setHiddenUntilMouseMoves: YES];
1292 #pragma mark Fullscreen support
1293 - (void)showFullscreenController
1295 if (b_fullscreen && [[VLCMain sharedInstance] activeVideoPlayback] )
1299 - (BOOL)isFullscreen
1301 return b_fullscreen;
1304 - (void)lockFullscreenAnimation
1306 [o_animation_lock lock];
1309 - (void)unlockFullscreenAnimation
1311 [o_animation_lock unlock];
1314 - (void)enterFullscreen
1316 NSMutableDictionary *dict1, *dict2;
1320 vout_thread_t *p_vout = getVout();
1321 BOOL blackout_other_displays = config_GetInt( VLCIntf, "macosx-black" );
1324 screen = [NSScreen screenWithDisplayID:(CGDirectDisplayID)config_GetInt( VLCIntf, "macosx-vdev" )];
1326 [self lockFullscreenAnimation];
1330 msg_Dbg( VLCIntf, "chosen screen isn't present, using current screen for fullscreen mode" );
1331 screen = [self screen];
1335 msg_Dbg( VLCIntf, "Using deepest screen" );
1336 screen = [NSScreen deepestScreen];
1340 vlc_object_release( p_vout );
1342 screen_rect = [screen frame];
1344 [o_fullscreen_btn setState: YES];
1346 [self recreateHideMouseTimer];
1348 if( blackout_other_displays )
1349 [screen blackoutOtherScreens];
1351 /* Make sure we don't see the window flashes in float-on-top mode */
1352 i_originalLevel = [self level];
1353 [self setLevel:NSNormalWindowLevel];
1355 /* Only create the o_fullscreen_window if we are not in the middle of the zooming animation */
1356 if (!o_fullscreen_window)
1358 /* We can't change the styleMask of an already created NSWindow, so we create another window, and do eye catching stuff */
1360 rect = [[o_video_view superview] convertRect: [o_video_view frame] toView: nil]; /* Convert to Window base coord */
1361 rect.origin.x += [self frame].origin.x;
1362 rect.origin.y += [self frame].origin.y;
1363 o_fullscreen_window = [[VLCWindow alloc] initWithContentRect:rect styleMask: NSBorderlessWindowMask backing:NSBackingStoreBuffered defer:YES];
1364 [o_fullscreen_window setBackgroundColor: [NSColor blackColor]];
1365 [o_fullscreen_window setCanBecomeKeyWindow: YES];
1367 if (![self isVisible] || [self alphaValue] == 0.0)
1369 /* We don't animate if we are not visible, instead we
1370 * simply fade the display */
1371 CGDisplayFadeReservationToken token;
1373 if( blackout_other_displays )
1375 CGAcquireDisplayFadeReservation( kCGMaxDisplayReservationInterval, &token );
1376 CGDisplayFade( token, 0.5, kCGDisplayBlendNormal, kCGDisplayBlendSolidColor, 0, 0, 0, YES );
1379 if ([screen isMainScreen])
1382 SetSystemUIMode( kUIModeAllHidden, kUIOptionAutoShowMenuBar);
1384 [NSApp setPresentationOptions:(NSApplicationPresentationAutoHideDock | NSApplicationPresentationAutoHideMenuBar)];
1387 [[o_video_view superview] replaceSubview:o_video_view with:o_temp_view];
1388 [o_temp_view setFrame:[o_video_view frame]];
1389 [o_fullscreen_window setContentView:o_video_view];
1391 [o_fullscreen_window makeKeyAndOrderFront:self];
1392 [o_fullscreen_window orderFront:self animate:YES];
1394 [o_fullscreen_window setFrame:screen_rect display:YES animate:YES];
1395 [o_fullscreen_window setLevel:NSNormalWindowLevel];
1397 if( blackout_other_displays )
1399 CGDisplayFade( token, 0.3, kCGDisplayBlendSolidColor, kCGDisplayBlendNormal, 0, 0, 0, NO );
1400 CGReleaseDisplayFadeReservation( token );
1403 /* Will release the lock */
1404 [self hasBecomeFullscreen];
1409 /* Make sure we don't see the o_video_view disappearing of the screen during this operation */
1410 NSDisableScreenUpdates();
1411 [[o_video_view superview] replaceSubview:o_video_view with:o_temp_view];
1412 [o_temp_view setFrame:[o_video_view frame]];
1413 [o_fullscreen_window setContentView:o_video_view];
1414 [o_fullscreen_window makeKeyAndOrderFront:self];
1415 NSEnableScreenUpdates();
1418 /* We are in fullscreen (and no animation is running) */
1421 /* Make sure we are hidden */
1422 [super orderOut: self];
1423 [self unlockFullscreenAnimation];
1427 if (o_fullscreen_anim1)
1429 [o_fullscreen_anim1 stopAnimation];
1430 [o_fullscreen_anim1 release];
1432 if (o_fullscreen_anim2)
1434 [o_fullscreen_anim2 stopAnimation];
1435 [o_fullscreen_anim2 release];
1438 if ([screen isMainScreen])
1441 SetSystemUIMode( kUIModeAllHidden, kUIOptionAutoShowMenuBar);
1443 [NSApp setPresentationOptions:(NSApplicationPresentationAutoHideDock | NSApplicationPresentationAutoHideMenuBar)];
1446 dict1 = [[NSMutableDictionary alloc] initWithCapacity:2];
1447 dict2 = [[NSMutableDictionary alloc] initWithCapacity:3];
1449 [dict1 setObject:self forKey:NSViewAnimationTargetKey];
1450 [dict1 setObject:NSViewAnimationFadeOutEffect forKey:NSViewAnimationEffectKey];
1452 [dict2 setObject:o_fullscreen_window forKey:NSViewAnimationTargetKey];
1453 [dict2 setObject:[NSValue valueWithRect:[o_fullscreen_window frame]] forKey:NSViewAnimationStartFrameKey];
1454 [dict2 setObject:[NSValue valueWithRect:screen_rect] forKey:NSViewAnimationEndFrameKey];
1456 /* Strategy with NSAnimation allocation:
1457 - Keep at most 2 animation at a time
1458 - leaveFullscreen/enterFullscreen are the only responsible for releasing and alloc-ing
1460 o_fullscreen_anim1 = [[NSViewAnimation alloc] initWithViewAnimations:[NSArray arrayWithObject:dict1]];
1461 o_fullscreen_anim2 = [[NSViewAnimation alloc] initWithViewAnimations:[NSArray arrayWithObject:dict2]];
1466 [o_fullscreen_anim1 setAnimationBlockingMode: NSAnimationNonblocking];
1467 [o_fullscreen_anim1 setDuration: 0.3];
1468 [o_fullscreen_anim1 setFrameRate: 30];
1469 [o_fullscreen_anim2 setAnimationBlockingMode: NSAnimationNonblocking];
1470 [o_fullscreen_anim2 setDuration: 0.2];
1471 [o_fullscreen_anim2 setFrameRate: 30];
1473 [o_fullscreen_anim2 setDelegate: self];
1474 [o_fullscreen_anim2 startWhenAnimation: o_fullscreen_anim1 reachesProgress: 1.0];
1476 [o_fullscreen_anim1 startAnimation];
1477 /* fullscreenAnimation will be unlocked when animation ends */
1480 - (void)hasBecomeFullscreen
1482 [o_fullscreen_window makeFirstResponder: o_video_view];
1484 [o_fullscreen_window makeKeyWindow];
1485 [o_fullscreen_window setAcceptsMouseMovedEvents: TRUE];
1487 /* tell the fspanel to move itself to front next time it's triggered */
1488 [o_fspanel setVoutWasUpdated: (int)[[o_fullscreen_window screen] displayID]];
1489 [o_fspanel setActive: nil];
1491 if([self isVisible])
1492 [super orderOut: self];
1494 [o_fspanel setActive: nil];
1497 [self unlockFullscreenAnimation];
1500 - (void)leaveFullscreen
1502 [self leaveFullscreenAndFadeOut: NO];
1505 - (void)leaveFullscreenAndFadeOut: (BOOL)fadeout
1507 NSMutableDictionary *dict1, *dict2;
1509 BOOL blackout_other_displays = config_GetInt( VLCIntf, "macosx-black" );
1511 [self lockFullscreenAnimation];
1514 [o_fullscreen_btn setState: NO];
1516 /* We always try to do so */
1517 if (!(OSX_LION || !b_nativeFullscreenMode))
1518 [NSScreen unblackoutScreens];
1519 vout_thread_t *p_vout = getVout();
1522 if( var_GetBool( p_vout, "video-on-top" ) )
1523 [[o_video_view window] setLevel: NSStatusWindowLevel];
1525 [[o_video_view window] setLevel: NSNormalWindowLevel];
1526 vlc_object_release( p_vout );
1528 [[o_video_view window] makeKeyAndOrderFront: nil];
1530 /* Don't do anything if o_fullscreen_window is already closed */
1531 if (!o_fullscreen_window)
1533 [self unlockFullscreenAnimation];
1539 /* We don't animate if we are not visible, instead we
1540 * simply fade the display */
1541 CGDisplayFadeReservationToken token;
1543 if( blackout_other_displays )
1545 CGAcquireDisplayFadeReservation( kCGMaxDisplayReservationInterval, &token );
1546 CGDisplayFade( token, 0.3, kCGDisplayBlendNormal, kCGDisplayBlendSolidColor, 0, 0, 0, YES );
1549 [o_fspanel setNonActive: nil];
1551 SetSystemUIMode( kUIModeNormal, kUIOptionAutoShowMenuBar);
1553 [NSApp setPresentationOptions: NSApplicationPresentationDefault];
1555 /* Will release the lock */
1556 [self hasEndedFullscreen];
1558 /* Our window is hidden, and might be faded. We need to workaround that, so note it
1560 b_window_is_invisible = YES;
1562 if( blackout_other_displays )
1564 CGDisplayFade( token, 0.5, kCGDisplayBlendSolidColor, kCGDisplayBlendNormal, 0, 0, 0, NO );
1565 CGReleaseDisplayFadeReservation( token );
1571 [self setAlphaValue: 0.0];
1572 [self orderFront: self];
1573 [[o_video_view window] orderFront: self];
1575 [o_fspanel setNonActive: nil];
1577 SetSystemUIMode( kUIModeNormal, kUIOptionAutoShowMenuBar);
1579 [NSApp setPresentationOptions:(NSApplicationPresentationDefault)];
1581 if (o_fullscreen_anim1)
1583 [o_fullscreen_anim1 stopAnimation];
1584 [o_fullscreen_anim1 release];
1586 if (o_fullscreen_anim2)
1588 [o_fullscreen_anim2 stopAnimation];
1589 [o_fullscreen_anim2 release];
1592 frame = [[o_temp_view superview] convertRect: [o_temp_view frame] toView: nil]; /* Convert to Window base coord */
1593 frame.origin.x += [self frame].origin.x;
1594 frame.origin.y += [self frame].origin.y;
1596 dict2 = [[NSMutableDictionary alloc] initWithCapacity:2];
1597 [dict2 setObject:self forKey:NSViewAnimationTargetKey];
1598 [dict2 setObject:NSViewAnimationFadeInEffect forKey:NSViewAnimationEffectKey];
1600 o_fullscreen_anim2 = [[NSViewAnimation alloc] initWithViewAnimations:[NSArray arrayWithObjects:dict2, nil]];
1603 [o_fullscreen_anim2 setAnimationBlockingMode: NSAnimationNonblocking];
1604 [o_fullscreen_anim2 setDuration: 0.3];
1605 [o_fullscreen_anim2 setFrameRate: 30];
1607 [o_fullscreen_anim2 setDelegate: self];
1609 dict1 = [[NSMutableDictionary alloc] initWithCapacity:3];
1611 [dict1 setObject:o_fullscreen_window forKey:NSViewAnimationTargetKey];
1612 [dict1 setObject:[NSValue valueWithRect:[o_fullscreen_window frame]] forKey:NSViewAnimationStartFrameKey];
1613 [dict1 setObject:[NSValue valueWithRect:frame] forKey:NSViewAnimationEndFrameKey];
1615 o_fullscreen_anim1 = [[NSViewAnimation alloc] initWithViewAnimations:[NSArray arrayWithObjects:dict1, nil]];
1618 [o_fullscreen_anim1 setAnimationBlockingMode: NSAnimationNonblocking];
1619 [o_fullscreen_anim1 setDuration: 0.2];
1620 [o_fullscreen_anim1 setFrameRate: 30];
1621 [o_fullscreen_anim2 startWhenAnimation: o_fullscreen_anim1 reachesProgress: 1.0];
1623 /* Make sure o_fullscreen_window is the frontmost window */
1624 [o_fullscreen_window orderFront: self];
1626 [o_fullscreen_anim1 startAnimation];
1627 /* fullscreenAnimation will be unlocked when animation ends */
1630 - (void)hasEndedFullscreen
1632 /* This function is private and should be only triggered at the end of the fullscreen change animation */
1633 /* Make sure we don't see the o_video_view disappearing of the screen during this operation */
1634 NSDisableScreenUpdates();
1635 [o_video_view retain];
1636 [o_video_view removeFromSuperviewWithoutNeedingDisplay];
1637 [[o_temp_view superview] replaceSubview:o_temp_view with:o_video_view];
1638 [o_video_view release];
1639 [o_video_view setFrame:[o_temp_view frame]];
1640 [self makeFirstResponder: o_video_view];
1641 if ([self isVisible])
1642 [super makeKeyAndOrderFront:self]; /* our version contains a workaround */
1643 [o_fullscreen_window orderOut: self];
1644 NSEnableScreenUpdates();
1646 [o_fullscreen_window release];
1647 o_fullscreen_window = nil;
1648 [self setLevel:i_originalLevel];
1650 [self unlockFullscreenAnimation];
1653 - (void)animationDidEnd:(NSAnimation*)animation
1655 NSArray *viewAnimations;
1656 if( o_makekey_anim == animation )
1658 [o_makekey_anim release];
1661 if ([animation currentValue] < 1.0)
1664 /* Fullscreen ended or started (we are a delegate only for leaveFullscreen's/enterFullscren's anim2) */
1665 viewAnimations = [o_fullscreen_anim2 viewAnimations];
1666 if ([viewAnimations count] >=1 &&
1667 [[[viewAnimations objectAtIndex: 0] objectForKey: NSViewAnimationEffectKey] isEqualToString:NSViewAnimationFadeInEffect])
1669 /* Fullscreen ended */
1670 [self hasEndedFullscreen];
1674 /* Fullscreen started */
1675 [self hasBecomeFullscreen];
1679 - (void)orderOut: (id)sender
1681 /* Make sure we leave fullscreen */
1682 if (!(OSX_LION || !b_nativeFullscreenMode))
1683 [self leaveFullscreenAndFadeOut: YES];
1685 [super orderOut: sender];
1688 - (void)makeKeyAndOrderFront: (id)sender
1691 * when we exit fullscreen and fade out, we may endup in
1692 * having a window that is faded. We can't have it fade in unless we
1695 if(!b_window_is_invisible)
1697 /* Make sure we don't do it too much */
1698 [super makeKeyAndOrderFront: sender];
1702 [super setAlphaValue:0.0f];
1703 [super makeKeyAndOrderFront: sender];
1705 NSMutableDictionary * dict = [[NSMutableDictionary alloc] initWithCapacity:2];
1706 [dict setObject:self forKey:NSViewAnimationTargetKey];
1707 [dict setObject:NSViewAnimationFadeInEffect forKey:NSViewAnimationEffectKey];
1709 o_makekey_anim = [[NSViewAnimation alloc] initWithViewAnimations:[NSArray arrayWithObject:dict]];
1712 [o_makekey_anim setAnimationBlockingMode: NSAnimationNonblocking];
1713 [o_makekey_anim setDuration: 0.1];
1714 [o_makekey_anim setFrameRate: 30];
1715 [o_makekey_anim setDelegate: self];
1717 [o_makekey_anim startAnimation];
1718 b_window_is_invisible = NO;
1720 /* fullscreenAnimation will be unlocked when animation ends */
1723 /* Make sure setFrame gets executed on main thread especially if we are animating.
1724 * (Thus we won't block the video output thread) */
1725 - (void)setFrame:(NSRect)frame display:(BOOL)display animate:(BOOL)animate
1727 struct { NSRect frame; BOOL display; BOOL animate;} args;
1731 args.display = display;
1732 args.animate = animate;
1734 packedargs = [NSData dataWithBytes:&args length:sizeof(args)];
1736 [self performSelectorOnMainThread:@selector(setFrameOnMainThread:)
1737 withObject: packedargs waitUntilDone: YES];
1740 - (void)setFrameOnMainThread:(NSData*)packedargs
1742 struct args { NSRect frame; BOOL display; BOOL animate; } * args = (struct args*)[packedargs bytes];
1746 /* Make sure we don't block too long and set up a non blocking animation */
1747 NSDictionary * dict = [NSDictionary dictionaryWithObjectsAndKeys:
1748 self, NSViewAnimationTargetKey,
1749 [NSValue valueWithRect:[self frame]], NSViewAnimationStartFrameKey,
1750 [NSValue valueWithRect:args->frame], NSViewAnimationEndFrameKey, nil];
1752 NSViewAnimation * anim = [[NSViewAnimation alloc] initWithViewAnimations:[NSArray arrayWithObject:dict]];
1754 [anim setAnimationBlockingMode: NSAnimationNonblocking];
1755 [anim setDuration: 0.4];
1756 [anim setFrameRate: 30];
1757 [anim startAnimation];
1762 [super setFrame:args->frame display:args->display animate:args->animate];
1767 #pragma mark Lion's native fullscreen handling
1768 - (void)windowWillEnterFullScreen:(NSNotification *)notification
1770 [o_video_view setFrame: [[self contentView] frame]];
1772 [o_fspanel setVoutWasUpdated: (int)[[self screen] displayID]];
1774 [self recreateHideMouseTimer];
1776 if (b_dark_interface)
1778 [o_titlebar_view removeFromSuperviewWithoutNeedingDisplay];
1781 CGFloat f_titleBarHeight = [o_titlebar_view frame].size.height;
1782 winrect = [self frame];
1784 winrect.size.height = winrect.size.height - f_titleBarHeight;
1785 [self setFrame: winrect display:NO animate:NO];
1786 winrect = [o_split_view frame];
1787 winrect.size.height = winrect.size.height + f_titleBarHeight;
1788 [o_split_view setFrame: winrect];
1791 if ([[VLCMain sharedInstance] activeVideoPlayback])
1792 [o_bottombar_view setHidden: YES];
1795 - (void)windowWillExitFullScreen:(NSNotification *)notification
1797 [o_video_view setFrame: [o_split_view frame]];
1798 [NSCursor setHiddenUntilMouseMoves: NO];
1799 [o_fspanel setNonActive: nil];
1802 if (b_dark_interface)
1805 CGFloat f_titleBarHeight = [o_titlebar_view frame].size.height;
1806 winrect = [self frame];
1808 [o_titlebar_view setFrame: NSMakeRect( 0, winrect.size.height - f_titleBarHeight,
1809 winrect.size.width, f_titleBarHeight )];
1810 [[self contentView] addSubview: o_titlebar_view];
1812 winrect.size.height = winrect.size.height + f_titleBarHeight;
1813 [self setFrame: winrect display:NO animate:NO];
1814 winrect = [o_split_view frame];
1815 winrect.size.height = winrect.size.height - f_titleBarHeight;
1816 [o_split_view setFrame: winrect];
1817 [o_video_view setFrame: winrect];
1820 if ([[VLCMain sharedInstance] activeVideoPlayback])
1821 [o_bottombar_view setHidden: NO];
1825 #pragma mark split view delegate
1826 - (CGFloat)splitView:(NSSplitView *)splitView constrainMinCoordinate:(CGFloat)proposedMin ofSubviewAt:(NSInteger)dividerIndex
1828 if (dividerIndex == 0)
1834 - (CGFloat)splitView:(NSSplitView *)splitView constrainMaxCoordinate:(CGFloat)proposedMax ofSubviewAt:(NSInteger)dividerIndex
1836 if (dividerIndex == 0)
1837 return ([self frame].size.width - 300.0);
1843 #pragma mark Side Bar Data handling
1844 /* taken under BSD-new from the PXSourceList sample project, adapted for VLC */
1845 - (NSUInteger)sourceList:(PXSourceList*)sourceList numberOfChildrenOfItem:(id)item
1847 //Works the same way as the NSOutlineView data source: `nil` means a parent item
1849 return [o_sidebaritems count];
1852 return [[item children] count];
1857 - (id)sourceList:(PXSourceList*)aSourceList child:(NSUInteger)index ofItem:(id)item
1859 //Works the same way as the NSOutlineView data source: `nil` means a parent item
1861 return [o_sidebaritems objectAtIndex:index];
1864 return [[item children] objectAtIndex:index];
1869 - (id)sourceList:(PXSourceList*)aSourceList objectValueForItem:(id)item
1871 return [item title];
1874 - (void)sourceList:(PXSourceList*)aSourceList setObjectValue:(id)object forItem:(id)item
1876 [item setTitle:object];
1879 - (BOOL)sourceList:(PXSourceList*)aSourceList isItemExpandable:(id)item
1881 return [item hasChildren];
1885 - (BOOL)sourceList:(PXSourceList*)aSourceList itemHasBadge:(id)item
1887 if ([[item identifier] isEqualToString: @"playlist"])
1890 return [item hasBadge];
1894 - (NSInteger)sourceList:(PXSourceList*)aSourceList badgeValueForItem:(id)item
1896 if ([[item identifier] isEqualToString: @"playlist"]) {
1897 playlist_t * p_playlist = pl_Get( VLCIntf );
1898 NSInteger i_playlist_size;
1901 i_playlist_size = p_playlist->items.i_size;
1904 return i_playlist_size;
1906 return [item badgeValue];
1910 - (BOOL)sourceList:(PXSourceList*)aSourceList itemHasIcon:(id)item
1912 return [item hasIcon];
1916 - (NSImage*)sourceList:(PXSourceList*)aSourceList iconForItem:(id)item
1921 - (NSMenu*)sourceList:(PXSourceList*)aSourceList menuForEvent:(NSEvent*)theEvent item:(id)item
1923 if ([theEvent type] == NSRightMouseDown || ([theEvent type] == NSLeftMouseDown && ([theEvent modifierFlags] & NSControlKeyMask) == NSControlKeyMask)) {
1927 if ([item sdtype] > 0)
1929 m = [[NSMenu alloc] init];
1930 playlist_t * p_playlist = pl_Get( VLCIntf );
1931 BOOL sd_loaded = playlist_IsServicesDiscoveryLoaded( p_playlist, [[item identifier] UTF8String] );
1933 [m addItemWithTitle:_NS("Enable") action:@selector(sdmenuhandler:) keyEquivalent:@""];
1935 [m addItemWithTitle:_NS("Disable") action:@selector(sdmenuhandler:) keyEquivalent:@""];
1936 [[m itemAtIndex:0] setRepresentedObject: [item identifier]];
1938 return [m autorelease];
1944 - (IBAction)sdmenuhandler:(id)sender
1946 NSString * identifier = [sender representedObject];
1947 if ([identifier length] > 0 && ![identifier isEqualToString:@"lua{sd='freebox',longname='Freebox TV'}"])
1949 playlist_t * p_playlist = pl_Get( VLCIntf );
1950 BOOL sd_loaded = playlist_IsServicesDiscoveryLoaded( p_playlist, [identifier UTF8String] );
1953 playlist_ServicesDiscoveryAdd( p_playlist, [identifier UTF8String] );
1955 playlist_ServicesDiscoveryRemove( p_playlist, [identifier UTF8String] );
1960 #pragma mark Side Bar Delegate Methods
1961 /* taken under BSD-new from the PXSourceList sample project, adapted for VLC */
1962 - (BOOL)sourceList:(PXSourceList*)aSourceList isGroupAlwaysExpanded:(id)group
1967 - (void)sourceListSelectionDidChange:(NSNotification *)notification
1969 NSIndexSet *selectedIndexes = [o_sidebar_view selectedRowIndexes];
1971 //Set the label text to represent the new selection
1972 if([selectedIndexes count]==1) {
1973 id item = [o_sidebar_view itemAtRow:[selectedIndexes firstIndex]];
1974 if ([item sdtype] > -1)
1976 playlist_t * p_playlist = pl_Get( VLCIntf );
1977 BOOL sd_loaded = playlist_IsServicesDiscoveryLoaded( p_playlist, [[item identifier] UTF8String] );
1980 playlist_ServicesDiscoveryAdd( p_playlist, [[item identifier] UTF8String] );
1984 [o_chosen_category_lbl setStringValue:[item title]];
1987 [o_chosen_category_lbl setStringValue:@"(none)"];