]> git.sesse.net Git - vlc/blob - modules/gui/macosx/intf.m
macosx gui: use vlc_path2uri.
[vlc] / modules / gui / macosx / intf.m
1 /*****************************************************************************
2  * intf.m: MacOS X interface module
3  *****************************************************************************
4  * Copyright (C) 2002-2012 VLC authors and VideoLAN
5  * $Id$
6  *
7  * Authors: Jon Lech Johansen <jon-vl@nanocrew.net>
8  *          Christophe Massiot <massiot@via.ecp.fr>
9  *          Derk-Jan Hartman <hartman at videolan.org>
10  *          Felix Paul Kühne <fkuehne at videolan dot org>
11  *
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.
16  *
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.
21  *
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  *****************************************************************************/
26
27 /*****************************************************************************
28  * Preamble
29  *****************************************************************************/
30 #ifdef HAVE_CONFIG_H
31 # include "config.h"
32 #endif
33
34 #include <stdlib.h>                                      /* malloc(), free() */
35 #include <sys/param.h>                                    /* for MAXPATHLEN */
36 #include <string.h>
37 #include <vlc_common.h>
38 #include <vlc_keys.h>
39 #include <vlc_dialog.h>
40 #include <vlc_url.h>
41 #include <vlc_modules.h>
42 #include <vlc_plugin.h>
43 #include <vlc_aout_intf.h>
44 #include <vlc_vout_window.h>
45 #include <vlc_vout_display.h>
46 #include <unistd.h> /* execl() */
47
48 #import "CompatibilityFixes.h"
49 #import "intf.h"
50 #import "MainMenu.h"
51 #import "VideoView.h"
52 #import "prefs.h"
53 #import "playlist.h"
54 #import "playlistinfo.h"
55 #import "controls.h"
56 #import "open.h"
57 #import "wizard.h"
58 #import "bookmarks.h"
59 #import "coredialogs.h"
60 #import "AppleRemote.h"
61 #import "eyetv.h"
62 #import "simple_prefs.h"
63 #import "CoreInteraction.h"
64 #import "TrackSynchronization.h"
65
66 #import <AddressBook/AddressBook.h>         /* for crashlog send mechanism */
67 #import <Sparkle/Sparkle.h>                 /* we're the update delegate */
68
69 /*****************************************************************************
70  * Local prototypes.
71  *****************************************************************************/
72 static void Run ( intf_thread_t *p_intf );
73
74 static void updateProgressPanel (void *, const char *, float);
75 static bool checkProgressPanel (void *);
76 static void destroyProgressPanel (void *);
77
78 static void MsgCallback( void *data, int type, const msg_item_t *item, const char *format, va_list ap );
79
80 static int InputEvent( vlc_object_t *, const char *,
81                       vlc_value_t, vlc_value_t, void * );
82 static int PLItemChanged( vlc_object_t *, const char *,
83                          vlc_value_t, vlc_value_t, void * );
84 static int PlaylistUpdated( vlc_object_t *, const char *,
85                            vlc_value_t, vlc_value_t, void * );
86 static int PlaybackModeUpdated( vlc_object_t *, const char *,
87                                vlc_value_t, vlc_value_t, void * );
88 static int VolumeUpdated( vlc_object_t *, const char *,
89                          vlc_value_t, vlc_value_t, void * );
90
91 #pragma mark -
92 #pragma mark VLC Interface Object Callbacks
93
94 /*****************************************************************************
95  * OpenIntf: initialize interface
96  *****************************************************************************/
97 int OpenIntf ( vlc_object_t *p_this )
98 {
99     NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
100     [VLCApplication sharedApplication];
101
102     intf_thread_t *p_intf = (intf_thread_t*) p_this;
103
104     p_intf->p_sys = malloc( sizeof( intf_sys_t ) );
105     if( p_intf->p_sys == NULL )
106         return VLC_ENOMEM;
107
108     memset( p_intf->p_sys, 0, sizeof( *p_intf->p_sys ) );
109
110     /* subscribe to LibVLCCore's messages */
111     vlc_Subscribe( &p_intf->p_sys->sub, MsgCallback, NULL );
112
113     Run( p_intf );
114
115     [o_pool release];
116     return VLC_SUCCESS;
117 }
118
119 /*****************************************************************************
120  * CloseIntf: destroy interface
121  *****************************************************************************/
122 void CloseIntf ( vlc_object_t *p_this )
123 {
124     intf_thread_t *p_intf = (intf_thread_t*) p_this;
125
126     free( p_intf->p_sys );
127 }
128
129 static int WindowControl( vout_window_t *, int i_query, va_list );
130
131 int WindowOpen( vout_window_t *p_wnd, const vout_window_cfg_t *cfg )
132 {
133     NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
134     intf_thread_t *p_intf = VLCIntf;
135     if (!p_intf) {
136         msg_Err( p_wnd, "Mac OS X interface not found" );
137         return VLC_EGENERIC;
138     }
139
140     int i_x = cfg->x;
141     int i_y = cfg->y;
142     unsigned i_width = cfg->width;
143     unsigned i_height = cfg->height;
144     p_wnd->handle.nsobject = [[VLCMain sharedInstance] getVideoViewAtPositionX: &i_x Y: &i_y withWidth: &i_width andHeight: &i_height];
145
146     if ( !p_wnd->handle.nsobject ) {
147         msg_Err( p_wnd, "got no video view from the interface" );
148         [o_pool release];
149         return VLC_EGENERIC;
150     }
151
152     [[VLCMain sharedInstance] setNativeVideoSize:NSMakeSize( cfg->width, cfg->height )];
153     [[VLCMain sharedInstance] setActiveVideoPlayback: YES];
154     p_wnd->control = WindowControl;
155     p_wnd->sys = (vout_window_sys_t *)VLCIntf;
156     [o_pool release];
157     return VLC_SUCCESS;
158 }
159
160 static int WindowControl( vout_window_t *p_wnd, int i_query, va_list args )
161 {
162     switch( i_query )
163     {
164         case VOUT_WINDOW_SET_STATE:
165         {
166             unsigned i_state = va_arg( args, unsigned );
167             [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(setWindowLevel:) withObject:[NSNumber numberWithUnsignedInt:i_state] waitUntilDone:NO];
168             return VLC_SUCCESS;
169         }
170         case VOUT_WINDOW_SET_SIZE:
171         {
172             unsigned int i_width  = va_arg( args, unsigned int );
173             unsigned int i_height = va_arg( args, unsigned int );
174             [[VLCMain sharedInstance] setNativeVideoSize:NSMakeSize( i_width, i_height )];
175             return VLC_SUCCESS;
176         }
177         case VOUT_WINDOW_SET_FULLSCREEN:
178         {
179             NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
180             int i_full = va_arg( args, int );
181             [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(checkFullscreenChange:) withObject:[NSNumber numberWithInt: i_full] waitUntilDone:NO];
182             [o_pool release];
183             return VLC_SUCCESS;
184         }
185         default:
186             msg_Warn( p_wnd, "unsupported control query" );
187             return VLC_EGENERIC;
188     }
189 }
190
191 void WindowClose( vout_window_t *p_wnd )
192 {
193     NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
194     [[VLCMain sharedInstance] setActiveVideoPlayback:NO];
195
196     [o_pool release];
197 }
198
199 /*****************************************************************************
200  * Run: main loop
201  *****************************************************************************/
202 static NSLock * o_appLock = nil;    // controls access to f_appExit
203
204 static void Run( intf_thread_t *p_intf )
205 {
206     NSAutoreleasePool * o_pool = [[NSAutoreleasePool alloc] init];
207     [VLCApplication sharedApplication];
208
209     o_appLock = [[NSLock alloc] init];
210
211     [[VLCMain sharedInstance] setIntf: p_intf];
212     [NSBundle loadNibNamed: @"MainMenu" owner: NSApp];
213
214     [NSApp run];
215     [[VLCMain sharedInstance] applicationWillTerminate:nil];
216     [o_appLock release];
217     [o_pool release];
218
219     raise(SIGTERM);
220 }
221
222 #pragma mark -
223 #pragma mark Variables Callback
224
225 /*****************************************************************************
226  * MsgCallback: Callback triggered by the core once a new debug message is
227  * ready to be displayed. We store everything in a NSArray in our Cocoa part
228  * of this file.
229  *****************************************************************************/
230 static void MsgCallback( void *data, int type, const msg_item_t *item, const char *format, va_list ap )
231 {
232     int canc = vlc_savecancel();
233     char *str;
234
235     if (vasprintf( &str, format, ap ) == -1)
236     {
237         vlc_restorecancel( canc );
238         return;
239     }
240
241     NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
242     [[VLCMain sharedInstance] processReceivedlibvlcMessage: item ofType: type withStr: str];
243     [o_pool release];
244
245     vlc_restorecancel( canc );
246     free( str );
247 }
248
249 static int InputEvent( vlc_object_t *p_this, const char *psz_var,
250                        vlc_value_t oldval, vlc_value_t new_val, void *param )
251 {
252     NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
253     switch (new_val.i_int) {
254         case INPUT_EVENT_STATE:
255             [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(playbackStatusUpdated) withObject: nil waitUntilDone:NO];
256             break;
257         case INPUT_EVENT_RATE:
258             [[[VLCMain sharedInstance] mainMenu] performSelectorOnMainThread:@selector(updatePlaybackRate) withObject: nil waitUntilDone:NO];
259             break;
260         case INPUT_EVENT_POSITION:
261             [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updatePlaybackPosition) withObject: nil waitUntilDone:NO];
262             break;
263         case INPUT_EVENT_TITLE:
264         case INPUT_EVENT_CHAPTER:
265             [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateMainMenu) withObject: nil waitUntilDone:NO];
266             break;
267         case INPUT_EVENT_CACHE:
268             [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateMainWindow) withObject: nil waitUntilDone: NO];
269             break;
270         case INPUT_EVENT_STATISTICS:
271             [[[VLCMain sharedInstance] info] performSelectorOnMainThread:@selector(updateStatistics) withObject: nil waitUntilDone: NO];
272             break;
273         case INPUT_EVENT_ES:
274             break;
275         case INPUT_EVENT_TELETEXT:
276             break;
277         case INPUT_EVENT_AOUT:
278             break;
279         case INPUT_EVENT_VOUT:
280             break;
281         case INPUT_EVENT_ITEM_META:
282         case INPUT_EVENT_ITEM_INFO:
283             [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateMainMenu) withObject: nil waitUntilDone:NO];
284             [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateName) withObject: nil waitUntilDone:NO];
285             [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateInfoandMetaPanel) withObject: nil waitUntilDone:NO];
286             break;
287         case INPUT_EVENT_BOOKMARK:
288             break;
289         case INPUT_EVENT_RECORD:
290             [[VLCMain sharedInstance] updateRecordState: var_GetBool( p_this, "record" )];
291             break;
292         case INPUT_EVENT_PROGRAM:
293             [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateMainMenu) withObject: nil waitUntilDone:NO];
294             break;
295         case INPUT_EVENT_ITEM_EPG:
296             break;
297         case INPUT_EVENT_SIGNAL:
298             break;
299
300         case INPUT_EVENT_ITEM_NAME:
301             [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateName) withObject: nil waitUntilDone:NO];
302             [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(playlistUpdated) withObject: nil waitUntilDone:NO];
303             break;
304
305         case INPUT_EVENT_AUDIO_DELAY:
306         case INPUT_EVENT_SUBTITLE_DELAY:
307             [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateDelays) withObject:nil waitUntilDone:NO];
308             break;
309
310         case INPUT_EVENT_DEAD:
311             [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateName) withObject: nil waitUntilDone:NO];
312             [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updatePlaybackPosition) withObject:nil waitUntilDone:NO];
313             break;
314
315         case INPUT_EVENT_ABORT:
316             [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateName) withObject: nil waitUntilDone:NO];
317             [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updatePlaybackPosition) withObject:nil waitUntilDone:NO];
318             break;
319
320         default:
321             //msg_Warn( p_this, "unhandled input event (%lld)", new_val.i_int );
322             break;
323     }
324
325     [o_pool release];
326     return VLC_SUCCESS;
327 }
328
329 static int PLItemChanged( vlc_object_t *p_this, const char *psz_var,
330                          vlc_value_t oldval, vlc_value_t new_val, void *param )
331 {
332     NSAutoreleasePool * o_pool = [[NSAutoreleasePool alloc] init];
333     [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(PlaylistItemChanged) withObject:nil waitUntilDone:NO];
334
335     [o_pool release];
336     return VLC_SUCCESS;
337 }
338
339 static int PlaylistUpdated( vlc_object_t *p_this, const char *psz_var,
340                          vlc_value_t oldval, vlc_value_t new_val, void *param )
341 {
342     NSAutoreleasePool * o_pool = [[NSAutoreleasePool alloc] init];
343     [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(playlistUpdated) withObject:nil waitUntilDone:NO];
344
345     [o_pool release];
346     return VLC_SUCCESS;
347 }
348
349 static int PlaybackModeUpdated( vlc_object_t *p_this, const char *psz_var,
350                          vlc_value_t oldval, vlc_value_t new_val, void *param )
351 {
352     NSAutoreleasePool * o_pool = [[NSAutoreleasePool alloc] init];
353     [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(playbackModeUpdated) withObject:nil waitUntilDone:NO];
354
355     [o_pool release];
356     return VLC_SUCCESS;
357 }
358
359 static int VolumeUpdated( vlc_object_t *p_this, const char *psz_var,
360                          vlc_value_t oldval, vlc_value_t new_val, void *param )
361 {
362     NSAutoreleasePool * o_pool = [[NSAutoreleasePool alloc] init];
363     [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateVolume) withObject:nil waitUntilDone:NO];
364
365     [o_pool release];
366     return VLC_SUCCESS;
367 }
368
369 /*****************************************************************************
370  * ShowController: Callback triggered by the show-intf playlist variable
371  * through the ShowIntf-control-intf, to let us show the controller-win;
372  * usually when in fullscreen-mode
373  *****************************************************************************/
374 static int ShowController( vlc_object_t *p_this, const char *psz_variable,
375                      vlc_value_t old_val, vlc_value_t new_val, void *param )
376 {
377     intf_thread_t * p_intf = VLCIntf;
378     if( p_intf && p_intf->p_sys )
379     {
380         playlist_t * p_playlist = pl_Get( p_intf );
381         BOOL b_fullscreen = var_GetBool( p_playlist, "fullscreen" );
382         if( strcmp(psz_variable, "intf-toggle-fscontrol") || b_fullscreen )
383         {
384             [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(showFullscreenController) withObject:nil waitUntilDone:NO];
385         }
386         else
387         {
388             [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(showMainWindow) withObject:nil waitUntilDone:NO];
389         }
390     }
391     return VLC_SUCCESS;
392 }
393
394 /*****************************************************************************
395  * FullscreenChanged: Callback triggered by the fullscreen-change playlist
396  * variable, to let the intf update the controller.
397  *****************************************************************************/
398 static int FullscreenChanged( vlc_object_t *p_this, const char *psz_variable,
399                      vlc_value_t old_val, vlc_value_t new_val, void *param )
400 {
401     intf_thread_t * p_intf = VLCIntf;
402     if (p_intf)
403     {
404         NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
405         [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(fullscreenChanged) withObject:nil waitUntilDone:NO];
406         [o_pool release];
407     }
408     return VLC_SUCCESS;
409 }
410
411 /*****************************************************************************
412  * DialogCallback: Callback triggered by the "dialog-*" variables
413  * to let the intf display error and interaction dialogs
414  *****************************************************************************/
415 static int DialogCallback( vlc_object_t *p_this, const char *type, vlc_value_t previous, vlc_value_t value, void *data )
416 {
417     NSAutoreleasePool * o_pool = [[NSAutoreleasePool alloc] init];
418     VLCMain *interface = (VLCMain *)data;
419
420     if( [[NSString stringWithUTF8String: type] isEqualToString: @"dialog-progress-bar"] )
421     {
422         /* the progress panel needs to update itself and therefore wants special treatment within this context */
423         dialog_progress_bar_t *p_dialog = (dialog_progress_bar_t *)value.p_address;
424
425         p_dialog->pf_update = updateProgressPanel;
426         p_dialog->pf_check = checkProgressPanel;
427         p_dialog->pf_destroy = destroyProgressPanel;
428         p_dialog->p_sys = VLCIntf->p_libvlc;
429     }
430
431     NSValue *o_value = [NSValue valueWithPointer:value.p_address];
432     [[VLCCoreDialogProvider sharedInstance] performEventWithObject: o_value ofType: type];
433
434     [o_pool release];
435     return VLC_SUCCESS;
436 }
437
438 void updateProgressPanel (void *priv, const char *text, float value)
439 {
440     NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
441
442     NSString *o_txt;
443     if( text != NULL )
444         o_txt = [NSString stringWithUTF8String: text];
445     else
446         o_txt = @"";
447
448     [[[VLCMain sharedInstance] coreDialogProvider] updateProgressPanelWithText: o_txt andNumber: (double)(value * 1000.)];
449
450     [o_pool release];
451 }
452
453 void destroyProgressPanel (void *priv)
454 {
455     NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
456     [[[VLCMain sharedInstance] coreDialogProvider] performSelectorOnMainThread:@selector(destroyProgressPanel) withObject:nil waitUntilDone:NO];
457     [o_pool release];
458 }
459
460 bool checkProgressPanel (void *priv)
461 {
462     NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
463     return [[[VLCMain sharedInstance] coreDialogProvider] progressCancelled];
464     [o_pool release];
465 }
466
467 #pragma mark -
468 #pragma mark Helpers
469
470 input_thread_t *getInput(void)
471 {
472     intf_thread_t *p_intf = VLCIntf;
473     if (!p_intf)
474         return NULL;
475     return pl_CurrentInput(p_intf);
476 }
477
478 vout_thread_t *getVout(void)
479 {
480     input_thread_t *p_input = getInput();
481     if (!p_input)
482         return NULL;
483     vout_thread_t *p_vout = input_GetVout(p_input);
484     vlc_object_release(p_input);
485     return p_vout;
486 }
487
488 audio_output_t *getAout(void)
489 {
490     input_thread_t *p_input = getInput();
491     if (!p_input)
492         return NULL;
493     audio_output_t *p_aout = input_GetAout(p_input);
494     vlc_object_release(p_input);
495     return p_aout;
496 }
497
498 #pragma mark -
499 #pragma mark Private
500
501 @interface VLCMain ()
502 - (void)_removeOldPreferences;
503 @end
504
505 /*****************************************************************************
506  * VLCMain implementation
507  *****************************************************************************/
508 @implementation VLCMain
509
510 #pragma mark -
511 #pragma mark Initialization
512
513 static VLCMain *_o_sharedMainInstance = nil;
514
515 + (VLCMain *)sharedInstance
516 {
517     return _o_sharedMainInstance ? _o_sharedMainInstance : [[self alloc] init];
518 }
519
520 - (id)init
521 {
522     if( _o_sharedMainInstance)
523     {
524         [self dealloc];
525         return _o_sharedMainInstance;
526     }
527     else
528         _o_sharedMainInstance = [super init];
529
530     p_intf = NULL;
531     p_current_input = NULL;
532
533     o_msg_lock = [[NSLock alloc] init];
534     o_msg_arr = [[NSMutableArray arrayWithCapacity: 600] retain];
535
536     o_open = [[VLCOpen alloc] init];
537     //o_embedded_list = [[VLCEmbeddedList alloc] init];
538     o_coredialogs = [[VLCCoreDialogProvider alloc] init];
539     o_info = [[VLCInfo alloc] init];
540     o_mainmenu = [[VLCMainMenu alloc] init];
541     o_coreinteraction = [[VLCCoreInteraction alloc] init];
542     o_eyetv = [[VLCEyeTVController alloc] init];
543     o_mainwindow = [[VLCMainWindow alloc] init];
544
545     /* announce our launch to a potential eyetv plugin */
546     [[NSDistributedNotificationCenter defaultCenter] postNotificationName: @"VLCOSXGUIInit"
547                                                                    object: @"VLCEyeTVSupport"
548                                                                  userInfo: NULL
549                                                        deliverImmediately: YES];
550
551     NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
552     NSDictionary *appDefaults = [NSDictionary dictionaryWithObject:@"NO" forKey:@"LiveUpdateTheMessagesPanel"];
553     [defaults registerDefaults:appDefaults];
554
555     return _o_sharedMainInstance;
556 }
557
558 - (void)setIntf: (intf_thread_t *)p_mainintf {
559     p_intf = p_mainintf;
560 }
561
562 - (intf_thread_t *)intf {
563     return p_intf;
564 }
565
566 - (void)awakeFromNib
567 {
568     playlist_t *p_playlist;
569     vlc_value_t val;
570     if( !p_intf ) return;
571     var_Create( p_intf, "intf-change", VLC_VAR_BOOL );
572
573     /* Check if we already did this once. Opening the other nibs calls it too,
574      because VLCMain is the owner */
575     if( nib_main_loaded ) return;
576
577     [o_msgs_panel setExcludedFromWindowsMenu: YES];
578     [o_msgs_panel setDelegate: self];
579
580     p_playlist = pl_Get( p_intf );
581
582     val.b_bool = false;
583
584     var_AddCallback(p_playlist, "fullscreen", FullscreenChanged, self);
585     var_AddCallback( p_intf->p_libvlc, "intf-toggle-fscontrol", ShowController, self);
586     var_AddCallback( p_intf->p_libvlc, "intf-show", ShowController, self);
587     //    var_AddCallback(p_playlist, "item-change", PLItemChanged, self);
588     var_AddCallback(p_playlist, "item-current", PLItemChanged, self);
589     var_AddCallback(p_playlist, "activity", PLItemChanged, self);
590     var_AddCallback(p_playlist, "leaf-to-parent", PlaylistUpdated, self);
591     var_AddCallback(p_playlist, "playlist-item-append", PlaylistUpdated, self);
592     var_AddCallback(p_playlist, "playlist-item-deleted", PlaylistUpdated, self);
593     var_AddCallback(p_playlist, "random", PlaybackModeUpdated, self);
594     var_AddCallback(p_playlist, "repeat", PlaybackModeUpdated, self);
595     var_AddCallback(p_playlist, "loop", PlaybackModeUpdated, self);
596     var_AddCallback(p_playlist, "volume", VolumeUpdated, self);
597     var_AddCallback(p_playlist, "mute", VolumeUpdated, self);
598
599     if (OSX_LION)
600     {
601         if ([NSApp currentSystemPresentationOptions] & NSApplicationPresentationFullScreen)
602             var_SetBool( p_playlist, "fullscreen", YES );
603     }
604
605     /* load our Core Dialogs nib */
606     nib_coredialogs_loaded = [NSBundle loadNibNamed:@"CoreDialogs" owner: NSApp];
607
608     /* subscribe to various interactive dialogues */
609     var_Create( p_intf, "dialog-error", VLC_VAR_ADDRESS );
610     var_AddCallback( p_intf, "dialog-error", DialogCallback, self );
611     var_Create( p_intf, "dialog-critical", VLC_VAR_ADDRESS );
612     var_AddCallback( p_intf, "dialog-critical", DialogCallback, self );
613     var_Create( p_intf, "dialog-login", VLC_VAR_ADDRESS );
614     var_AddCallback( p_intf, "dialog-login", DialogCallback, self );
615     var_Create( p_intf, "dialog-question", VLC_VAR_ADDRESS );
616     var_AddCallback( p_intf, "dialog-question", DialogCallback, self );
617     var_Create( p_intf, "dialog-progress-bar", VLC_VAR_ADDRESS );
618     var_AddCallback( p_intf, "dialog-progress-bar", DialogCallback, self );
619     dialog_Register( p_intf );
620
621     /* init Apple Remote support */
622     o_remote = [[AppleRemote alloc] init];
623     [o_remote setClickCountEnabledButtons: kRemoteButtonPlay];
624     [o_remote setDelegate: _o_sharedMainInstance];
625
626     [o_msgs_refresh_btn setImage: [NSImage imageNamed: NSImageNameRefreshTemplate]];
627
628      BOOL b_video_deco = var_InheritBool( VLCIntf, "video-deco" );
629     /* yeah, we are done */
630     b_nativeFullscreenMode = NO;
631 #ifdef MAC_OS_X_VERSION_10_7
632     if( OSX_LION && b_video_deco )
633         b_nativeFullscreenMode = var_InheritBool( p_intf, "macosx-nativefullscreenmode" );
634 #endif
635
636     /* recover stored audio device, if set
637      * in case it was unplugged in the meantime, auhal will fall back on the default */
638     int i_value = config_GetInt( p_intf, "macosx-audio-device" );
639     if (i_value > 0)
640         var_SetInteger( pl_Get( VLCIntf ), "audio-device", i_value );
641
642     if (config_GetInt( VLCIntf, "macosx-icon-change"))
643     {
644         /* After day 354 of the year, the usual VLC cone is replaced by another cone
645          * wearing a Father Xmas hat.
646          * Note: this icon doesn't represent an endorsement of The Coca-Cola Company.
647          */
648         NSCalendar *gregorian =
649         [[NSCalendar alloc] initWithCalendarIdentifier:NSGregorianCalendar];
650         NSUInteger dayOfYear = [gregorian ordinalityOfUnit:NSDayCalendarUnit inUnit:NSYearCalendarUnit forDate:[NSDate date]];
651         [gregorian release];
652
653         if (dayOfYear >= 354)
654             [[VLCApplication sharedApplication] setApplicationIconImage: [NSImage imageNamed:@"vlc-xmas"]];
655     }
656
657     nib_main_loaded = TRUE;
658 }
659
660 - (void)applicationDidFinishLaunching:(NSNotification *)aNotification
661 {
662     if( !p_intf ) return;
663
664     [self updateCurrentlyUsedHotkeys];
665
666     [o_mainwindow updateWindow];
667     [o_mainwindow updateTimeSlider];
668     [o_mainwindow updateVolumeSlider];
669     [o_mainwindow makeKeyAndOrderFront: self];
670
671     /* init media key support */
672     b_mediaKeySupport = var_InheritBool( VLCIntf, "macosx-mediakeys" );
673     if( b_mediaKeySupport )
674     {
675         o_mediaKeyController = [[SPMediaKeyTap alloc] initWithDelegate:self];
676         [o_mediaKeyController startWatchingMediaKeys];
677         [[NSUserDefaults standardUserDefaults] registerDefaults:[NSDictionary dictionaryWithObjectsAndKeys:
678                                                                  [SPMediaKeyTap defaultMediaKeyUserBundleIdentifiers], kMediaKeyUsingBundleIdentifiersDefaultsKey,
679                                                                  nil]];
680     }
681     [[NSNotificationCenter defaultCenter] addObserver: self selector: @selector(coreChangedMediaKeySupportSetting:) name: @"VLCMediaKeySupportSettingChanged" object: nil];
682
683     [self _removeOldPreferences];
684
685     /* Handle sleep notification */
686     [[[NSWorkspace sharedWorkspace] notificationCenter] addObserver:self selector:@selector(computerWillSleep:)
687            name:NSWorkspaceWillSleepNotification object:nil];
688
689     [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(lookForCrashLog) withObject:nil waitUntilDone:NO];
690
691     /* we will need this, so let's load it here so the interface appears to be more responsive */
692     nib_open_loaded = [NSBundle loadNibNamed:@"Open" owner: NSApp];
693     [self initStrings];
694 }
695
696 - (void)initStrings
697 {
698     if( !p_intf ) return;
699
700     /* messages panel */
701     [o_msgs_panel setTitle: _NS("Messages")];
702     [o_msgs_crashlog_btn setTitle: _NS("Open CrashLog...")];
703     [o_msgs_save_btn setTitle: _NS("Save this Log...")];
704
705     /* crash reporter panel */
706     [o_crashrep_send_btn setTitle: _NS("Send")];
707     [o_crashrep_dontSend_btn setTitle: _NS("Don't Send")];
708     [o_crashrep_title_txt setStringValue: _NS("VLC crashed previously")];
709     [o_crashrep_win setTitle: _NS("VLC crashed previously")];
710     [o_crashrep_desc_txt setStringValue: _NS("Do you want to send details on the crash to VLC's development team?\n\nIf you want, you can enter a few lines on what you did before VLC crashed along with other helpful information: a link to download a sample file, a URL of a network stream, ...")];
711     [o_crashrep_includeEmail_ckb setTitle: _NS("I agree to be possibly contacted about this bugreport.")];
712     [o_crashrep_includeEmail_txt setStringValue: _NS("Only your default E-Mail address will be submitted, including no further information.")];
713 }
714
715 #pragma mark -
716 #pragma mark Termination
717
718 - (void)applicationWillTerminate:(NSNotification *)notification
719 {
720     /* don't allow a double termination call. If the user has
721      * already invoked the quit then simply return this time. */
722     static bool f_appExit = false;
723     bool isTerminating;
724
725     [o_appLock lock];
726     isTerminating = f_appExit;
727     f_appExit = true;
728     [o_appLock unlock];
729
730     if (isTerminating)
731         return;
732
733     if (notification == nil)
734         [[NSNotificationCenter defaultCenter] postNotificationName: NSApplicationWillTerminateNotification object: nil];
735
736     playlist_t * p_playlist = pl_Get( p_intf );
737     int returnedValue = 0;
738
739     /* always exit fullscreen on quit, otherwise we get ugly artifacts on the next launch */
740     if (b_nativeFullscreenMode)
741     {
742         [o_mainwindow toggleFullScreen: self];
743         [NSApp setPresentationOptions:(NSApplicationPresentationDefault)];
744     }
745
746     /* Save some interface state in configuration, at module quit */
747     config_PutInt( p_intf, "random", var_GetBool( p_playlist, "random" ) );
748     config_PutInt( p_intf, "loop", var_GetBool( p_playlist, "loop" ) );
749     config_PutInt( p_intf, "repeat", var_GetBool( p_playlist, "repeat" ) );
750
751     msg_Dbg( p_intf, "Terminating" );
752
753     /* unsubscribe from the interactive dialogues */
754     dialog_Unregister( p_intf );
755     var_DelCallback( p_intf, "dialog-error", DialogCallback, self );
756     var_DelCallback( p_intf, "dialog-critical", DialogCallback, self );
757     var_DelCallback( p_intf, "dialog-login", DialogCallback, self );
758     var_DelCallback( p_intf, "dialog-question", DialogCallback, self );
759     var_DelCallback( p_intf, "dialog-progress-bar", DialogCallback, self );
760     //var_DelCallback(p_playlist, "item-change", PLItemChanged, self);
761     var_DelCallback(p_playlist, "item-current", PLItemChanged, self);
762     var_DelCallback(p_playlist, "activity", PLItemChanged, self);
763     var_DelCallback(p_playlist, "leaf-to-parent", PlaylistUpdated, self);
764     var_DelCallback(p_playlist, "playlist-item-append", PlaylistUpdated, self);
765     var_DelCallback(p_playlist, "playlist-item-deleted", PlaylistUpdated, self);
766     var_DelCallback(p_playlist, "random", PlaybackModeUpdated, self);
767     var_DelCallback(p_playlist, "repeat", PlaybackModeUpdated, self);
768     var_DelCallback(p_playlist, "loop", PlaybackModeUpdated, self);
769     var_DelCallback(p_playlist, "volume", VolumeUpdated, self);
770     var_DelCallback(p_playlist, "mute", VolumeUpdated, self);
771     var_DelCallback(p_playlist, "fullscreen", FullscreenChanged, self);
772     var_DelCallback(p_intf->p_libvlc, "intf-toggle-fscontrol", ShowController, self);
773     var_DelCallback(p_intf->p_libvlc, "intf-show", ShowController, self);
774
775     if( p_current_input )
776     {
777         var_DelCallback( p_current_input, "intf-event", InputEvent, [VLCMain sharedInstance] );
778         vlc_object_release( p_current_input );
779         p_current_input = NULL;
780     }
781
782     /* remove global observer watching for vout device changes correctly */
783     [[NSNotificationCenter defaultCenter] removeObserver: self];
784
785     /* release some other objects here, because it isn't sure whether dealloc
786      * will be called later on */
787     if( o_sprefs )
788         [o_sprefs release];
789
790     if( o_prefs )
791         [o_prefs release];
792
793     [o_open release];
794
795     if( o_info )
796         [o_info release];
797
798     if( o_wizard )
799         [o_wizard release];
800
801     [crashLogURLConnection cancel];
802     [crashLogURLConnection release];
803
804     [o_embedded_list release];
805     [o_coredialogs release];
806     [o_eyetv release];
807
808     /* unsubscribe from libvlc's debug messages */
809     vlc_Unsubscribe( &p_intf->p_sys->sub );
810
811     [o_msg_arr removeAllObjects];
812     [o_msg_arr release];
813     o_msg_arr = NULL;
814     [o_usedHotkeys release];
815     o_usedHotkeys = NULL;
816
817     [o_msg_lock release];
818
819     /* write cached user defaults to disk */
820     [[NSUserDefaults standardUserDefaults] synchronize];
821
822     /* Make sure the Menu doesn't have any references to vlc objects anymore */
823     //FIXME: this should be moved to VLCMainMenu
824     [o_mainmenu releaseRepresentedObjects:[NSApp mainMenu]];
825     [o_mainmenu release];
826
827     libvlc_Quit( p_intf->p_libvlc );
828
829     [o_mainwindow release];
830     o_mainwindow = NULL;
831
832     [self setIntf:nil];
833 }
834
835 #pragma mark -
836 #pragma mark Sparkle delegate
837 /* received directly before the update gets installed, so let's shut down a bit */
838 - (void)updater:(SUUpdater *)updater willInstallUpdate:(SUAppcastItem *)update
839 {
840     [NSApp activateIgnoringOtherApps:YES];
841     [o_remote stopListening: self];
842     [[VLCCoreInteraction sharedInstance] stop];
843 }
844
845 #pragma mark -
846 #pragma mark Media Key support
847
848 -(void)mediaKeyTap:(SPMediaKeyTap*)keyTap receivedMediaKeyEvent:(NSEvent*)event
849 {
850     if( b_mediaKeySupport )
851        {
852         assert([event type] == NSSystemDefined && [event subtype] == SPSystemDefinedEventMediaKeys);
853
854         int keyCode = (([event data1] & 0xFFFF0000) >> 16);
855         int keyFlags = ([event data1] & 0x0000FFFF);
856         int keyState = (((keyFlags & 0xFF00) >> 8)) == 0xA;
857         int keyRepeat = (keyFlags & 0x1);
858
859         if( keyCode == NX_KEYTYPE_PLAY && keyState == 0 )
860             [[VLCCoreInteraction sharedInstance] play];
861
862         if( (keyCode == NX_KEYTYPE_FAST || keyCode == NX_KEYTYPE_NEXT) && !b_mediakeyJustJumped )
863         {
864             if( keyState == 0 && keyRepeat == 0 )
865                 [[VLCCoreInteraction sharedInstance] next];
866             else if( keyRepeat == 1 )
867             {
868                 [[VLCCoreInteraction sharedInstance] forwardShort];
869                 b_mediakeyJustJumped = YES;
870                 [self performSelector:@selector(resetMediaKeyJump)
871                            withObject: NULL
872                            afterDelay:0.25];
873             }
874         }
875
876         if( (keyCode == NX_KEYTYPE_REWIND || keyCode == NX_KEYTYPE_PREVIOUS) && !b_mediakeyJustJumped )
877         {
878             if( keyState == 0 && keyRepeat == 0 )
879                 [[VLCCoreInteraction sharedInstance] previous];
880             else if( keyRepeat == 1 )
881             {
882                 [[VLCCoreInteraction sharedInstance] backwardShort];
883                 b_mediakeyJustJumped = YES;
884                 [self performSelector:@selector(resetMediaKeyJump)
885                            withObject: NULL
886                            afterDelay:0.25];
887             }
888         }
889     }
890 }
891
892 #pragma mark -
893 #pragma mark Other notification
894
895 /* Listen to the remote in exclusive mode, only when VLC is the active
896    application */
897 - (void)applicationDidBecomeActive:(NSNotification *)aNotification
898 {
899     if( !p_intf ) return;
900     if( var_InheritBool( p_intf, "macosx-appleremote" ) == YES )
901         [o_remote startListening: self];
902 }
903 - (void)applicationDidResignActive:(NSNotification *)aNotification
904 {
905     if( !p_intf ) return;
906     [o_remote stopListening: self];
907 }
908
909 /* Triggered when the computer goes to sleep */
910 - (void)computerWillSleep: (NSNotification *)notification
911 {
912     [[VLCCoreInteraction sharedInstance] pause];
913 }
914
915 #pragma mark -
916 #pragma mark File opening over dock icon
917
918 - (void)application:(NSApplication *)o_app openFiles:(NSArray *)o_names
919 {
920     BOOL b_autoplay = config_GetInt( VLCIntf, "macosx-autoplay" );
921     char *psz_uri = vlc_path2uri([[o_names objectAtIndex:0] UTF8String], "file" );
922
923     // try to add file as subtitle
924     if( [o_names count] == 1 && psz_uri )
925     {
926         input_thread_t * p_input = pl_CurrentInput( VLCIntf );
927         if( p_input )
928         {
929             BOOL b_returned = NO;
930             b_returned = input_AddSubtitle( p_input, psz_uri, true );
931             vlc_object_release( p_input );
932             if( !b_returned )
933             {
934                 free( psz_uri );
935                 return;
936             }
937         }
938     }
939     free( psz_uri );
940
941     NSArray *o_sorted_names = [o_names sortedArrayUsingSelector: @selector(caseInsensitiveCompare:)];
942     NSMutableArray *o_result = [NSMutableArray arrayWithCapacity: [o_sorted_names count]];
943     for( int i = 0; i < [o_sorted_names count]; i++ )
944     {
945         psz_uri = vlc_path2uri([[o_sorted_names objectAtIndex: i] UTF8String], "file" );
946         if( !psz_uri )
947             continue;
948
949         NSDictionary *o_dic = [NSDictionary dictionaryWithObject:[NSString stringWithCString:psz_uri encoding:NSUTF8StringEncoding] forKey:@"ITEM_URL"];
950         free( psz_uri );
951         [o_result addObject: o_dic];
952     }
953
954     if( b_autoplay )
955         [o_playlist appendArray: o_result atPos: -1 enqueue: NO];
956     else
957         [o_playlist appendArray: o_result atPos: -1 enqueue: YES];
958
959     return;
960 }
961
962 /* When user click in the Dock icon our double click in the finder */
963 - (BOOL)applicationShouldHandleReopen:(NSApplication *)theApplication hasVisibleWindows:(BOOL)hasVisibleWindows
964 {
965     if(!hasVisibleWindows)
966         [o_mainwindow makeKeyAndOrderFront:self];
967
968     return YES;
969 }
970
971 #pragma mark -
972 #pragma mark Apple Remote Control
973
974 /* Helper method for the remote control interface in order to trigger forward/backward and volume
975    increase/decrease as long as the user holds the left/right, plus/minus button */
976 - (void) executeHoldActionForRemoteButton: (NSNumber*) buttonIdentifierNumber
977 {
978     if(b_remote_button_hold)
979     {
980         switch([buttonIdentifierNumber intValue])
981         {
982             case kRemoteButtonRight_Hold:
983                 [[VLCCoreInteraction sharedInstance] forward];
984                 break;
985             case kRemoteButtonLeft_Hold:
986                 [[VLCCoreInteraction sharedInstance] backward];
987                 break;
988             case kRemoteButtonVolume_Plus_Hold:
989                 if( p_intf )
990                     var_SetInteger( p_intf->p_libvlc, "key-action", ACTIONID_VOL_UP );
991                 break;
992             case kRemoteButtonVolume_Minus_Hold:
993                 if( p_intf )
994                     var_SetInteger( p_intf->p_libvlc, "key-action", ACTIONID_VOL_DOWN );
995                 break;
996         }
997         if(b_remote_button_hold)
998         {
999             /* trigger event */
1000             [self performSelector:@selector(executeHoldActionForRemoteButton:)
1001                          withObject:buttonIdentifierNumber
1002                          afterDelay:0.25];
1003         }
1004     }
1005 }
1006
1007 /* Apple Remote callback */
1008 - (void) appleRemoteButton: (AppleRemoteEventIdentifier)buttonIdentifier
1009                pressedDown: (BOOL) pressedDown
1010                 clickCount: (unsigned int) count
1011 {
1012     switch( buttonIdentifier )
1013     {
1014         case k2009RemoteButtonFullscreen:
1015             [[VLCCoreInteraction sharedInstance] toggleFullscreen];
1016             break;
1017         case k2009RemoteButtonPlay:
1018             [[VLCCoreInteraction sharedInstance] play];
1019             break;
1020         case kRemoteButtonPlay:
1021             if(count >= 2) {
1022                 [[VLCCoreInteraction sharedInstance] toggleFullscreen];
1023             } else {
1024                 [[VLCCoreInteraction sharedInstance] play];
1025             }
1026             break;
1027         case kRemoteButtonVolume_Plus:
1028             if (config_GetInt( VLCIntf, "macosx-appleremote-sysvol"))
1029                 [NSSound increaseSystemVolume];
1030             else
1031                 if( p_intf )
1032                     var_SetInteger( p_intf->p_libvlc, "key-action", ACTIONID_VOL_UP );
1033             break;
1034         case kRemoteButtonVolume_Minus:
1035             if (config_GetInt( VLCIntf, "macosx-appleremote-sysvol"))
1036                 [NSSound decreaseSystemVolume];
1037             else
1038                 if( p_intf )
1039                     var_SetInteger( p_intf->p_libvlc, "key-action", ACTIONID_VOL_DOWN );
1040             break;
1041         case kRemoteButtonRight:
1042             [[VLCCoreInteraction sharedInstance] next];
1043             break;
1044         case kRemoteButtonLeft:
1045             [[VLCCoreInteraction sharedInstance] previous];
1046             break;
1047         case kRemoteButtonRight_Hold:
1048         case kRemoteButtonLeft_Hold:
1049         case kRemoteButtonVolume_Plus_Hold:
1050         case kRemoteButtonVolume_Minus_Hold:
1051             /* simulate an event as long as the user holds the button */
1052             b_remote_button_hold = pressedDown;
1053             if( pressedDown )
1054             {
1055                 NSNumber* buttonIdentifierNumber = [NSNumber numberWithInt: buttonIdentifier];
1056                 [self performSelector:@selector(executeHoldActionForRemoteButton:)
1057                            withObject:buttonIdentifierNumber];
1058             }
1059             break;
1060         case kRemoteButtonMenu:
1061             [o_controls showPosition: self]; //FIXME
1062             break;
1063         case kRemoteButtonPlay_Sleep:
1064         {
1065             NSAppleScript * script = [[NSAppleScript alloc] initWithSource:@"tell application \"System Events\" to sleep"];
1066             [script executeAndReturnError:nil];
1067             [script release];
1068             break;
1069         }
1070         default:
1071             /* Add here whatever you want other buttons to do */
1072             break;
1073     }
1074 }
1075
1076 #pragma mark -
1077 #pragma mark String utility
1078 // FIXME: this has nothing to do here
1079
1080 - (NSString *)localizedString:(const char *)psz
1081 {
1082     NSString * o_str = nil;
1083
1084     if( psz != NULL )
1085     {
1086         o_str = [NSString stringWithCString: _(psz) encoding:NSUTF8StringEncoding];
1087
1088         if( o_str == NULL )
1089         {
1090             msg_Err( VLCIntf, "could not translate: %s", psz );
1091             return( @"" );
1092         }
1093     }
1094     else
1095     {
1096         msg_Warn( VLCIntf, "can't translate empty strings" );
1097         return( @"" );
1098     }
1099
1100     return( o_str );
1101 }
1102
1103
1104
1105 - (char *)delocalizeString:(NSString *)id
1106 {
1107     NSData * o_data = [id dataUsingEncoding: NSUTF8StringEncoding
1108                           allowLossyConversion: NO];
1109     char * psz_string;
1110
1111     if( o_data == nil )
1112     {
1113         o_data = [id dataUsingEncoding: NSUTF8StringEncoding
1114                      allowLossyConversion: YES];
1115         psz_string = malloc( [o_data length] + 1 );
1116         [o_data getBytes: psz_string];
1117         psz_string[ [o_data length] ] = '\0';
1118         msg_Err( VLCIntf, "cannot convert to the requested encoding: %s",
1119                  psz_string );
1120     }
1121     else
1122     {
1123         psz_string = malloc( [o_data length] + 1 );
1124         [o_data getBytes: psz_string];
1125         psz_string[ [o_data length] ] = '\0';
1126     }
1127
1128     return psz_string;
1129 }
1130
1131 /* i_width is in pixels */
1132 - (NSString *)wrapString: (NSString *)o_in_string toWidth: (int) i_width
1133 {
1134     NSMutableString *o_wrapped;
1135     NSString *o_out_string;
1136     NSRange glyphRange, effectiveRange, charRange;
1137     NSRect lineFragmentRect;
1138     unsigned glyphIndex, breaksInserted = 0;
1139
1140     NSTextStorage *o_storage = [[NSTextStorage alloc] initWithString: o_in_string
1141         attributes: [NSDictionary dictionaryWithObjectsAndKeys:
1142         [NSFont labelFontOfSize: 0.0], NSFontAttributeName, nil]];
1143     NSLayoutManager *o_layout_manager = [[NSLayoutManager alloc] init];
1144     NSTextContainer *o_container = [[NSTextContainer alloc]
1145         initWithContainerSize: NSMakeSize(i_width, 2000)];
1146
1147     [o_layout_manager addTextContainer: o_container];
1148     [o_container release];
1149     [o_storage addLayoutManager: o_layout_manager];
1150     [o_layout_manager release];
1151
1152     o_wrapped = [o_in_string mutableCopy];
1153     glyphRange = [o_layout_manager glyphRangeForTextContainer: o_container];
1154
1155     for( glyphIndex = glyphRange.location ; glyphIndex < NSMaxRange(glyphRange) ;
1156             glyphIndex += effectiveRange.length) {
1157         lineFragmentRect = [o_layout_manager lineFragmentRectForGlyphAtIndex: glyphIndex
1158                                             effectiveRange: &effectiveRange];
1159         charRange = [o_layout_manager characterRangeForGlyphRange: effectiveRange
1160                                     actualGlyphRange: &effectiveRange];
1161         if([o_wrapped lineRangeForRange:
1162                 NSMakeRange(charRange.location + breaksInserted, charRange.length)].length > charRange.length) {
1163             [o_wrapped insertString: @"\n" atIndex: NSMaxRange(charRange) + breaksInserted];
1164             breaksInserted++;
1165         }
1166     }
1167     o_out_string = [NSString stringWithString: o_wrapped];
1168     [o_wrapped release];
1169     [o_storage release];
1170
1171     return o_out_string;
1172 }
1173
1174
1175 #pragma mark -
1176 #pragma mark Key Shortcuts
1177
1178 static struct
1179 {
1180     unichar i_nskey;
1181     unsigned int i_vlckey;
1182 } nskeys_to_vlckeys[] =
1183 {
1184     { NSUpArrowFunctionKey, KEY_UP },
1185     { NSDownArrowFunctionKey, KEY_DOWN },
1186     { NSLeftArrowFunctionKey, KEY_LEFT },
1187     { NSRightArrowFunctionKey, KEY_RIGHT },
1188     { NSF1FunctionKey, KEY_F1 },
1189     { NSF2FunctionKey, KEY_F2 },
1190     { NSF3FunctionKey, KEY_F3 },
1191     { NSF4FunctionKey, KEY_F4 },
1192     { NSF5FunctionKey, KEY_F5 },
1193     { NSF6FunctionKey, KEY_F6 },
1194     { NSF7FunctionKey, KEY_F7 },
1195     { NSF8FunctionKey, KEY_F8 },
1196     { NSF9FunctionKey, KEY_F9 },
1197     { NSF10FunctionKey, KEY_F10 },
1198     { NSF11FunctionKey, KEY_F11 },
1199     { NSF12FunctionKey, KEY_F12 },
1200     { NSInsertFunctionKey, KEY_INSERT },
1201     { NSHomeFunctionKey, KEY_HOME },
1202     { NSEndFunctionKey, KEY_END },
1203     { NSPageUpFunctionKey, KEY_PAGEUP },
1204     { NSPageDownFunctionKey, KEY_PAGEDOWN },
1205     { NSMenuFunctionKey, KEY_MENU },
1206     { NSTabCharacter, KEY_TAB },
1207     { NSCarriageReturnCharacter, KEY_ENTER },
1208     { NSEnterCharacter, KEY_ENTER },
1209     { NSBackspaceCharacter, KEY_BACKSPACE },
1210     { NSDeleteCharacter, KEY_DELETE },
1211     {0,0}
1212 };
1213
1214 unsigned int CocoaKeyToVLC( unichar i_key )
1215 {
1216     unsigned int i;
1217
1218     for( i = 0; nskeys_to_vlckeys[i].i_nskey != 0; i++ )
1219     {
1220         if( nskeys_to_vlckeys[i].i_nskey == i_key )
1221         {
1222             return nskeys_to_vlckeys[i].i_vlckey;
1223         }
1224     }
1225     return (unsigned int)i_key;
1226 }
1227
1228 - (unsigned int)VLCModifiersToCocoa:(NSString *)theString
1229 {
1230     unsigned int new = 0;
1231
1232     if([theString rangeOfString:@"Command"].location != NSNotFound)
1233         new |= NSCommandKeyMask;
1234     if([theString rangeOfString:@"Alt"].location != NSNotFound)
1235         new |= NSAlternateKeyMask;
1236     if([theString rangeOfString:@"Shift"].location != NSNotFound)
1237         new |= NSShiftKeyMask;
1238     if([theString rangeOfString:@"Ctrl"].location != NSNotFound)
1239         new |= NSControlKeyMask;
1240     return new;
1241 }
1242
1243 - (NSString *)VLCKeyToString:(NSString *)theString
1244 {
1245     if (![theString isEqualToString:@""]) {
1246         if ([theString characterAtIndex:([theString length] - 1)] != 0x2b)
1247             theString = [theString stringByReplacingOccurrencesOfString:@"+" withString:@""];
1248         else
1249         {
1250             theString = [theString stringByReplacingOccurrencesOfString:@"+" withString:@""];
1251             theString = [NSString stringWithFormat:@"%@+", theString];
1252         }
1253         if ([theString characterAtIndex:([theString length] - 1)] != 0x2d)
1254             theString = [theString stringByReplacingOccurrencesOfString:@"-" withString:@""];
1255         else
1256         {
1257             theString = [theString stringByReplacingOccurrencesOfString:@"-" withString:@""];
1258             theString = [NSString stringWithFormat:@"%@-", theString];
1259         }
1260         theString = [theString stringByReplacingOccurrencesOfString:@"Command" withString:@""];
1261         theString = [theString stringByReplacingOccurrencesOfString:@"Alt" withString:@""];
1262         theString = [theString stringByReplacingOccurrencesOfString:@"Shift" withString:@""];
1263         theString = [theString stringByReplacingOccurrencesOfString:@"Ctrl" withString:@""];
1264     }
1265     if ([theString length] > 1)
1266     {
1267         if([theString rangeOfString:@"Up"].location != NSNotFound)
1268             return [NSString stringWithFormat:@"%C", NSUpArrowFunctionKey];
1269         else if([theString rangeOfString:@"Down"].location != NSNotFound)
1270             return [NSString stringWithFormat:@"%C", NSDownArrowFunctionKey];
1271         else if([theString rangeOfString:@"Right"].location != NSNotFound)
1272             return [NSString stringWithFormat:@"%C", NSRightArrowFunctionKey];
1273         else if([theString rangeOfString:@"Left"].location != NSNotFound)
1274             return [NSString stringWithFormat:@"%C", NSLeftArrowFunctionKey];
1275         else if([theString rangeOfString:@"Enter"].location != NSNotFound)
1276             return [NSString stringWithFormat:@"%C", NSEnterCharacter]; // we treat NSCarriageReturnCharacter as aquivalent
1277         else if([theString rangeOfString:@"Insert"].location != NSNotFound)
1278             return [NSString stringWithFormat:@"%C", NSInsertFunctionKey];
1279         else if([theString rangeOfString:@"Home"].location != NSNotFound)
1280             return [NSString stringWithFormat:@"%C", NSHomeFunctionKey];
1281         else if([theString rangeOfString:@"End"].location != NSNotFound)
1282             return [NSString stringWithFormat:@"%C", NSEndFunctionKey];
1283         else if([theString rangeOfString:@"Pageup"].location != NSNotFound)
1284             return [NSString stringWithFormat:@"%C", NSPageUpFunctionKey];
1285         else if([theString rangeOfString:@"Pagedown"].location != NSNotFound)
1286             return [NSString stringWithFormat:@"%C", NSPageDownFunctionKey];
1287         else if([theString rangeOfString:@"Menu"].location != NSNotFound)
1288             return [NSString stringWithFormat:@"%C", NSMenuFunctionKey];
1289         else if([theString rangeOfString:@"Tab"].location != NSNotFound)
1290             return [NSString stringWithFormat:@"%C", NSTabCharacter];
1291         else if([theString rangeOfString:@"Backspace"].location != NSNotFound)
1292             return [NSString stringWithFormat:@"%C", NSBackspaceCharacter];
1293         else if([theString rangeOfString:@"Delete"].location != NSNotFound)
1294             return [NSString stringWithFormat:@"%C", NSDeleteCharacter];
1295         else if([theString rangeOfString:@"F12"].location != NSNotFound)
1296             return [NSString stringWithFormat:@"%C", NSF12FunctionKey];
1297         else if([theString rangeOfString:@"F11"].location != NSNotFound)
1298             return [NSString stringWithFormat:@"%C", NSF11FunctionKey];
1299         else if([theString rangeOfString:@"F10"].location != NSNotFound)
1300             return [NSString stringWithFormat:@"%C", NSF10FunctionKey];
1301         else if([theString rangeOfString:@"F9"].location != NSNotFound)
1302             return [NSString stringWithFormat:@"%C", NSF9FunctionKey];
1303         else if([theString rangeOfString:@"F8"].location != NSNotFound)
1304             return [NSString stringWithFormat:@"%C", NSF8FunctionKey];
1305         else if([theString rangeOfString:@"F7"].location != NSNotFound)
1306             return [NSString stringWithFormat:@"%C", NSF7FunctionKey];
1307         else if([theString rangeOfString:@"F6"].location != NSNotFound)
1308             return [NSString stringWithFormat:@"%C", NSF6FunctionKey];
1309         else if([theString rangeOfString:@"F5"].location != NSNotFound)
1310             return [NSString stringWithFormat:@"%C", NSF5FunctionKey];
1311         else if([theString rangeOfString:@"F4"].location != NSNotFound)
1312             return [NSString stringWithFormat:@"%C", NSF4FunctionKey];
1313         else if([theString rangeOfString:@"F3"].location != NSNotFound)
1314             return [NSString stringWithFormat:@"%C", NSF3FunctionKey];
1315         else if([theString rangeOfString:@"F2"].location != NSNotFound)
1316             return [NSString stringWithFormat:@"%C", NSF2FunctionKey];
1317         else if([theString rangeOfString:@"F1"].location != NSNotFound)
1318             return [NSString stringWithFormat:@"%C", NSF1FunctionKey];
1319         /* note that we don't support esc here, since it is reserved for leaving fullscreen */
1320     }
1321
1322     return theString;
1323 }
1324
1325
1326 /*****************************************************************************
1327  * hasDefinedShortcutKey: Check to see if the key press is a defined VLC
1328  * shortcut key.  If it is, pass it off to VLC for handling and return YES,
1329  * otherwise ignore it and return NO (where it will get handled by Cocoa).
1330  *****************************************************************************/
1331 - (BOOL)hasDefinedShortcutKey:(NSEvent *)o_event force:(BOOL)b_force
1332 {
1333     unichar key = 0;
1334     vlc_value_t val;
1335     unsigned int i_pressed_modifiers = 0;
1336
1337     val.i_int = 0;
1338     i_pressed_modifiers = [o_event modifierFlags];
1339
1340     if( i_pressed_modifiers & NSControlKeyMask ) {
1341         val.i_int |= KEY_MODIFIER_CTRL;
1342     }
1343     if( i_pressed_modifiers & NSAlternateKeyMask ) {
1344         val.i_int |= KEY_MODIFIER_ALT;
1345     }
1346     if( i_pressed_modifiers & NSShiftKeyMask ) {
1347         val.i_int |= KEY_MODIFIER_SHIFT;
1348     }
1349     if( i_pressed_modifiers & NSCommandKeyMask ) {
1350         val.i_int |= KEY_MODIFIER_COMMAND;
1351     }
1352
1353     NSString * characters = [o_event charactersIgnoringModifiers];
1354     if ([characters length] > 0)
1355     {
1356         key = [[characters lowercaseString] characterAtIndex: 0];
1357
1358         /* handle Lion's default key combo for fullscreen-toggle in addition to our own hotkeys */
1359         if( key == 'f' && i_pressed_modifiers & NSControlKeyMask && i_pressed_modifiers & NSCommandKeyMask )
1360         {
1361             [[VLCCoreInteraction sharedInstance] toggleFullscreen];
1362             return YES;
1363         }
1364
1365         if( !b_force )
1366         {
1367             switch( key )
1368             {
1369                 case NSDeleteCharacter:
1370                 case NSDeleteFunctionKey:
1371                 case NSDeleteCharFunctionKey:
1372                 case NSBackspaceCharacter:
1373                 case NSUpArrowFunctionKey:
1374                 case NSDownArrowFunctionKey:
1375                 case NSRightArrowFunctionKey:
1376                 case NSLeftArrowFunctionKey:
1377                 case NSEnterCharacter:
1378                 case NSCarriageReturnCharacter:
1379                     return NO;
1380             }
1381         }
1382
1383         if( key == 0x0020 ) // space key
1384         {
1385             [[VLCCoreInteraction sharedInstance] play];
1386             return YES;
1387         }
1388
1389         val.i_int |= CocoaKeyToVLC( key );
1390
1391         BOOL b_found_key = NO;
1392         for( int i = 0; i < [o_usedHotkeys count]; i++ )
1393         {
1394             NSString *str = [o_usedHotkeys objectAtIndex: i];
1395             unsigned int i_keyModifiers = [self VLCModifiersToCocoa: str];
1396
1397             if( [[characters lowercaseString] isEqualToString: [self VLCKeyToString: str]] &&
1398                (i_keyModifiers & NSShiftKeyMask)     == (i_pressed_modifiers & NSShiftKeyMask) &&
1399                (i_keyModifiers & NSControlKeyMask)   == (i_pressed_modifiers & NSControlKeyMask) &&
1400                (i_keyModifiers & NSAlternateKeyMask) == (i_pressed_modifiers & NSAlternateKeyMask) &&
1401                (i_keyModifiers & NSCommandKeyMask)   == (i_pressed_modifiers & NSCommandKeyMask) )
1402             {
1403                 b_found_key = YES;
1404                 break;
1405             }
1406         }
1407
1408         if( b_found_key )
1409         {
1410             var_SetInteger( p_intf->p_libvlc, "key-pressed", val.i_int );
1411             return YES;
1412         }
1413     }
1414
1415     return NO;
1416 }
1417
1418 - (void)updateCurrentlyUsedHotkeys
1419 {
1420     NSMutableArray *o_tempArray = [[NSMutableArray alloc] init];
1421     /* Get the main Module */
1422     module_t *p_main = module_get_main();
1423     assert( p_main );
1424     unsigned confsize;
1425     module_config_t *p_config;
1426
1427     p_config = module_config_get (p_main, &confsize);
1428
1429     for (size_t i = 0; i < confsize; i++)
1430     {
1431         module_config_t *p_item = p_config + i;
1432
1433         if( CONFIG_ITEM(p_item->i_type) && p_item->psz_name != NULL
1434            && !strncmp( p_item->psz_name , "key-", 4 )
1435            && !EMPTY_STR( p_item->psz_text ) )
1436         {
1437             if (p_item->value.psz)
1438             {
1439                 [o_tempArray addObject: [NSString stringWithUTF8String:p_item->value.psz]];
1440             }
1441         }
1442     }
1443     module_config_free (p_config);
1444
1445     if( o_usedHotkeys )
1446         [o_usedHotkeys release];
1447     o_usedHotkeys = [[NSArray alloc] initWithArray: o_tempArray copyItems: YES];
1448     [o_tempArray release];
1449 }
1450
1451 #pragma mark -
1452 #pragma mark Interface updaters
1453 - (void)fullscreenChanged
1454 {
1455     playlist_t * p_playlist = pl_Get( VLCIntf );
1456     BOOL b_fullscreen = var_GetBool( p_playlist, "fullscreen" );
1457
1458     if (b_nativeFullscreenMode)
1459     {
1460         // this is called twice in certain situations, so only toogle if we really need to
1461         if( (  b_fullscreen && !([NSApp currentSystemPresentationOptions] & NSApplicationPresentationFullScreen) ) ||
1462             ( !b_fullscreen &&  ([NSApp currentSystemPresentationOptions] & NSApplicationPresentationFullScreen) ) )
1463             [o_mainwindow toggleFullScreen: self];
1464
1465         if(b_fullscreen)
1466             [NSApp setPresentationOptions:(NSApplicationPresentationFullScreen | NSApplicationPresentationAutoHideDock | NSApplicationPresentationAutoHideMenuBar)];
1467         else
1468             [NSApp setPresentationOptions:(NSApplicationPresentationDefault)];
1469     }
1470     else
1471     {
1472         if( b_fullscreen )
1473         {
1474             input_thread_t * p_input = pl_CurrentInput( VLCIntf );
1475             if( p_input != NULL && [self activeVideoPlayback] )
1476             {
1477                 // activate app, as method can also be triggered from outside the app (prevents nasty window layout)
1478                 [NSApp activateIgnoringOtherApps:YES];
1479                 [o_mainwindow performSelectorOnMainThread:@selector(enterFullscreen) withObject:nil waitUntilDone:NO];
1480             }
1481             if (p_input)
1482                 vlc_object_release( p_input );
1483         }
1484         else
1485         {
1486             // leaving fullscreen is always allowed
1487             [o_mainwindow performSelectorOnMainThread:@selector(leaveFullscreen) withObject:nil waitUntilDone:NO];
1488         }
1489     }
1490 }
1491
1492 - (void)checkFullscreenChange:(NSNumber *)o_full
1493 {
1494     BOOL b_full = [o_full boolValue];
1495     if( p_intf && !var_GetBool( pl_Get( p_intf ), "fullscreen" ) != !b_full )
1496     {
1497         var_SetBool( pl_Get(p_intf), "fullscreen", b_full );
1498     }
1499 }
1500
1501 - (void)PlaylistItemChanged
1502 {
1503     if( p_current_input && ( p_current_input->b_dead || !vlc_object_alive( p_current_input ) ))
1504     {
1505         var_DelCallback( p_current_input, "intf-event", InputEvent, [VLCMain sharedInstance] );
1506         vlc_object_release( p_current_input );
1507         p_current_input = NULL;
1508
1509         [o_mainmenu setRateControlsEnabled: NO];
1510     }
1511     else if( !p_current_input )
1512     {
1513         // object is hold here and released then it is dead
1514         p_current_input = playlist_CurrentInput( pl_Get( VLCIntf ));
1515         if( p_current_input )
1516         {
1517             var_AddCallback( p_current_input, "intf-event", InputEvent, [VLCMain sharedInstance] );
1518             [self playbackStatusUpdated];
1519             [o_mainmenu setRateControlsEnabled: YES];
1520             if ( [self activeVideoPlayback] && [[o_mainwindow videoView] isHidden] )
1521                 [o_mainwindow performSelectorOnMainThread:@selector(togglePlaylist:) withObject: nil waitUntilDone:NO];
1522         }
1523     }
1524
1525     [o_playlist updateRowSelection];
1526     [o_mainwindow updateWindow];
1527     [self updateDelays];
1528     [self updateMainMenu];
1529 }
1530
1531 - (void)updateMainMenu
1532 {
1533     [o_mainmenu setupMenus];
1534     [o_mainmenu updatePlaybackRate];
1535 }
1536
1537 - (void)updateMainWindow
1538 {
1539     [o_mainwindow updateWindow];
1540 }
1541
1542 - (void)showMainWindow
1543 {
1544     [o_mainwindow performSelectorOnMainThread:@selector(makeKeyAndOrderFront:) withObject:nil waitUntilDone:NO];
1545 }
1546
1547 - (void)showFullscreenController
1548 {
1549     [o_mainwindow performSelectorOnMainThread:@selector(showFullscreenController) withObject:nil waitUntilDone:NO];
1550 }
1551
1552 - (void)updateDelays
1553 {
1554     [[VLCTrackSynchronization sharedInstance] performSelectorOnMainThread: @selector(updateValues) withObject: nil waitUntilDone:NO];
1555 }
1556
1557 - (void)updateName
1558 {
1559     [o_mainwindow updateName];
1560 }
1561
1562 - (void)updatePlaybackPosition
1563 {
1564     [o_mainwindow updateTimeSlider];
1565
1566     input_thread_t * p_input;
1567     p_input = pl_CurrentInput( p_intf );
1568     if( p_input )
1569     {
1570         if( var_GetInteger( p_input, "state" ) == PLAYING_S && [self activeVideoPlayback] )
1571             UpdateSystemActivity( UsrActivity );
1572         vlc_object_release( p_input );
1573     }
1574 }
1575
1576 - (void)updateVolume
1577 {
1578     [o_mainwindow updateVolumeSlider];
1579 }
1580
1581 - (void)playlistUpdated
1582 {
1583     [self playbackStatusUpdated];
1584     [o_playlist playlistUpdated];
1585     [o_mainwindow updateWindow];
1586     [o_mainwindow updateName];
1587 }
1588
1589 - (void)updateRecordState: (BOOL)b_value
1590 {
1591     [o_mainmenu updateRecordState:b_value];
1592 }
1593
1594 - (void)updateInfoandMetaPanel
1595 {
1596     [o_playlist outlineViewSelectionDidChange:nil];
1597 }
1598
1599 - (void)playbackStatusUpdated
1600 {
1601     input_thread_t * p_input;
1602
1603     p_input = pl_CurrentInput( p_intf );
1604     if( p_input )
1605     {
1606         int state = var_GetInteger( p_input, "state" );
1607         if( state == PLAYING_S )
1608         {
1609             [[self mainMenu] setPause];
1610             [o_mainwindow setPause];
1611         }
1612         else
1613         {
1614             if (state == END_S)
1615                 [o_mainmenu setSubmenusEnabled: FALSE];
1616             [[self mainMenu] setPlay];
1617             [o_mainwindow setPlay];
1618         }
1619         vlc_object_release( p_input );
1620     }
1621
1622     [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateMainWindow) withObject: nil waitUntilDone: NO];
1623     [self performSelectorOnMainThread:@selector(sendDistributedNotificationWithUpdatedPlaybackStatus) withObject: nil waitUntilDone: NO];
1624 }
1625
1626 - (void)sendDistributedNotificationWithUpdatedPlaybackStatus
1627 {
1628     [[NSDistributedNotificationCenter defaultCenter] postNotificationName:@"VLCPlayerStateDidChange"
1629                                                                    object:nil
1630                                                                  userInfo:nil
1631                                                        deliverImmediately:YES];
1632 }
1633
1634 - (void)playbackModeUpdated
1635 {
1636     vlc_value_t looping,repeating;
1637     playlist_t * p_playlist = pl_Get( VLCIntf );
1638
1639     bool loop = var_GetBool( p_playlist, "loop" );
1640     bool repeat = var_GetBool( p_playlist, "repeat" );
1641     if( repeat ) {
1642         [o_mainwindow setRepeatOne];
1643         [o_mainmenu setRepeatOne];
1644     } else if( loop ) {
1645         [o_mainwindow setRepeatAll];
1646         [o_mainmenu setRepeatAll];
1647     } else {
1648         [o_mainwindow setRepeatOff];
1649         [o_mainmenu setRepeatOff];
1650     }
1651
1652     [o_mainwindow setShuffle];
1653     [o_mainmenu setShuffle];
1654 }
1655
1656
1657 #pragma mark -
1658 #pragma mark Window updater
1659
1660 - (void)setWindowLevel:(NSNumber*)state
1661 {
1662     if( var_InheritBool( p_intf, "macosx-background" ) )
1663         return;
1664
1665     if ([state unsignedIntValue] & VOUT_WINDOW_STATE_ABOVE)
1666         [[[[VLCMainWindow sharedInstance] videoView] window] setLevel: NSStatusWindowLevel];
1667     else
1668         [[[[VLCMainWindow sharedInstance] videoView] window] setLevel: NSNormalWindowLevel];
1669 }
1670
1671 - (void)setActiveVideoPlayback:(BOOL)b_value
1672 {
1673     b_active_videoplayback = b_value;
1674     if( o_mainwindow )
1675     {
1676         [o_mainwindow performSelectorOnMainThread:@selector(setVideoplayEnabled) withObject:nil waitUntilDone:YES];
1677         [o_mainwindow performSelectorOnMainThread:@selector(togglePlaylist:) withObject:nil waitUntilDone:NO];
1678     }
1679 }
1680
1681 - (void)setNativeVideoSize:(NSSize)size
1682 {
1683     [o_mainwindow setNativeVideoSize:size];
1684 }
1685
1686 #pragma mark -
1687 #pragma mark Other objects getters
1688
1689 - (id)mainMenu
1690 {
1691     return o_mainmenu;
1692 }
1693
1694 - (id)mainWindow
1695 {
1696     return o_mainwindow;
1697 }
1698
1699 - (id)controls
1700 {
1701     if( o_controls )
1702         return o_controls;
1703
1704     return nil;
1705 }
1706
1707 - (id)bookmarks
1708 {
1709     if (!o_bookmarks )
1710         o_bookmarks = [[VLCBookmarks alloc] init];
1711
1712     if( !nib_bookmarks_loaded )
1713         nib_bookmarks_loaded = [NSBundle loadNibNamed:@"Bookmarks" owner: NSApp];
1714
1715     return o_bookmarks;
1716 }
1717
1718 - (id)open
1719 {
1720     if (!o_open)
1721         return nil;
1722
1723     if (!nib_open_loaded)
1724         nib_open_loaded = [NSBundle loadNibNamed:@"Open" owner: NSApp];
1725
1726     return o_open;
1727 }
1728
1729 - (id)simplePreferences
1730 {
1731     if (!o_sprefs)
1732         o_sprefs = [[VLCSimplePrefs alloc] init];
1733
1734     if (!nib_prefs_loaded)
1735         nib_prefs_loaded = [NSBundle loadNibNamed:@"Preferences" owner: NSApp];
1736
1737     return o_sprefs;
1738 }
1739
1740 - (id)preferences
1741 {
1742     if( !o_prefs )
1743         o_prefs = [[VLCPrefs alloc] init];
1744
1745     if( !nib_prefs_loaded )
1746         nib_prefs_loaded = [NSBundle loadNibNamed:@"Preferences" owner: NSApp];
1747
1748     return o_prefs;
1749 }
1750
1751 - (id)playlist
1752 {
1753     if( o_playlist )
1754         return o_playlist;
1755
1756     return nil;
1757 }
1758
1759 - (id)info
1760 {
1761     if(! nib_info_loaded )
1762         nib_info_loaded = [NSBundle loadNibNamed:@"MediaInfo" owner: NSApp];
1763
1764     if( o_info )
1765         return o_info;
1766
1767     return nil;
1768 }
1769
1770 - (id)wizard
1771 {
1772     if( !o_wizard )
1773         o_wizard = [[VLCWizard alloc] init];
1774
1775     if( !nib_wizard_loaded )
1776     {
1777         nib_wizard_loaded = [NSBundle loadNibNamed:@"Wizard" owner: NSApp];
1778         [o_wizard initStrings];
1779     }
1780     return o_wizard;
1781 }
1782
1783 - (id)getVideoViewAtPositionX: (int *)pi_x Y: (int *)pi_y withWidth: (unsigned int*)pi_width andHeight: (unsigned int*)pi_height
1784 {
1785     id videoView = [o_mainwindow setupVideoView];
1786     NSRect videoRect = [videoView frame];
1787     int i_x = (int)videoRect.origin.x;
1788     int i_y = (int)videoRect.origin.y;
1789     unsigned int i_width = (int)videoRect.size.width;
1790     unsigned int i_height = (int)videoRect.size.height;
1791     pi_x = &i_x;
1792     pi_y = &i_y;
1793     pi_width = &i_width;
1794     pi_height = &i_height;
1795     msg_Dbg( VLCIntf, "returning videoview with x=%i, y=%i, width=%i, height=%i", i_x, i_y, i_width, i_height );
1796     return videoView;
1797 }
1798
1799 - (id)embeddedList
1800 {
1801     if( o_embedded_list )
1802         return o_embedded_list;
1803
1804     return nil;
1805 }
1806
1807 - (id)coreDialogProvider
1808 {
1809     if( o_coredialogs )
1810         return o_coredialogs;
1811
1812     return nil;
1813 }
1814
1815 - (id)eyeTVController
1816 {
1817     if( o_eyetv )
1818         return o_eyetv;
1819
1820     return nil;
1821 }
1822
1823 - (id)appleRemoteController
1824 {
1825     return o_remote;
1826 }
1827
1828 - (BOOL)activeVideoPlayback
1829 {
1830     return b_active_videoplayback;
1831 }
1832
1833 #pragma mark -
1834 #pragma mark Crash Log
1835 - (void)sendCrashLog:(NSString *)crashLog withUserComment:(NSString *)userComment
1836 {
1837     NSString *urlStr = @"http://jones.videolan.org/crashlog/sendcrashreport.php";
1838     NSURL *url = [NSURL URLWithString:urlStr];
1839
1840     NSMutableURLRequest *req = [NSMutableURLRequest requestWithURL:url];
1841     [req setHTTPMethod:@"POST"];
1842
1843     NSString * email;
1844     if( [o_crashrep_includeEmail_ckb state] == NSOnState )
1845     {
1846         ABPerson * contact = [[ABAddressBook sharedAddressBook] me];
1847         ABMultiValue *emails = [contact valueForProperty:kABEmailProperty];
1848         email = [emails valueAtIndex:[emails indexForIdentifier:
1849                     [emails primaryIdentifier]]];
1850     }
1851     else
1852         email = [NSString string];
1853
1854     NSString *postBody;
1855     postBody = [NSString stringWithFormat:@"CrashLog=%@&Comment=%@&Email=%@\r\n",
1856             [crashLog stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding],
1857             [userComment stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding],
1858             [email stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding]];
1859
1860     [req setHTTPBody:[postBody dataUsingEncoding:NSUTF8StringEncoding]];
1861
1862     /* Released from delegate */
1863     crashLogURLConnection = [[NSURLConnection alloc] initWithRequest:req delegate:self];
1864 }
1865
1866 - (void)connectionDidFinishLoading:(NSURLConnection *)connection
1867 {
1868     [crashLogURLConnection release];
1869     crashLogURLConnection = nil;
1870 }
1871
1872 - (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error
1873 {
1874     NSRunCriticalAlertPanel(_NS("Error when sending the Crash Report"), [error localizedDescription], @"OK", nil, nil);
1875     [crashLogURLConnection release];
1876     crashLogURLConnection = nil;
1877 }
1878
1879 - (NSString *)latestCrashLogPathPreviouslySeen:(BOOL)previouslySeen
1880 {
1881     NSString * crashReporter = [@"~/Library/Logs/CrashReporter" stringByExpandingTildeInPath];
1882     NSDirectoryEnumerator *direnum = [[NSFileManager defaultManager] enumeratorAtPath:crashReporter];
1883     NSString *fname;
1884     NSString * latestLog = nil;
1885     int year  = !previouslySeen ? [[NSUserDefaults standardUserDefaults] integerForKey:@"LatestCrashReportYear"] : 0;
1886     int month = !previouslySeen ? [[NSUserDefaults standardUserDefaults] integerForKey:@"LatestCrashReportMonth"]: 0;
1887     int day   = !previouslySeen ? [[NSUserDefaults standardUserDefaults] integerForKey:@"LatestCrashReportDay"]  : 0;
1888     int hours = !previouslySeen ? [[NSUserDefaults standardUserDefaults] integerForKey:@"LatestCrashReportHours"]: 0;
1889
1890     while (fname = [direnum nextObject])
1891     {
1892         [direnum skipDescendents];
1893         if([fname hasPrefix:@"VLC"] && [fname hasSuffix:@"crash"])
1894         {
1895             NSArray * compo = [fname componentsSeparatedByString:@"_"];
1896             if( [compo count] < 3 ) continue;
1897             compo = [[compo objectAtIndex:1] componentsSeparatedByString:@"-"];
1898             if( [compo count] < 4 ) continue;
1899
1900             // Dooh. ugly.
1901             if( year < [[compo objectAtIndex:0] intValue] ||
1902                 (year ==[[compo objectAtIndex:0] intValue] &&
1903                  (month < [[compo objectAtIndex:1] intValue] ||
1904                   (month ==[[compo objectAtIndex:1] intValue] &&
1905                    (day   < [[compo objectAtIndex:2] intValue] ||
1906                     (day   ==[[compo objectAtIndex:2] intValue] &&
1907                       hours < [[compo objectAtIndex:3] intValue] ))))))
1908             {
1909                 year  = [[compo objectAtIndex:0] intValue];
1910                 month = [[compo objectAtIndex:1] intValue];
1911                 day   = [[compo objectAtIndex:2] intValue];
1912                 hours = [[compo objectAtIndex:3] intValue];
1913                 latestLog = [crashReporter stringByAppendingPathComponent:fname];
1914             }
1915         }
1916     }
1917
1918     if(!(latestLog && [[NSFileManager defaultManager] fileExistsAtPath:latestLog]))
1919         return nil;
1920
1921     if( !previouslySeen )
1922     {
1923         [[NSUserDefaults standardUserDefaults] setInteger:year  forKey:@"LatestCrashReportYear"];
1924         [[NSUserDefaults standardUserDefaults] setInteger:month forKey:@"LatestCrashReportMonth"];
1925         [[NSUserDefaults standardUserDefaults] setInteger:day   forKey:@"LatestCrashReportDay"];
1926         [[NSUserDefaults standardUserDefaults] setInteger:hours forKey:@"LatestCrashReportHours"];
1927     }
1928     return latestLog;
1929 }
1930
1931 - (NSString *)latestCrashLogPath
1932 {
1933     return [self latestCrashLogPathPreviouslySeen:YES];
1934 }
1935
1936 - (void)lookForCrashLog
1937 {
1938     NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
1939     // This pref key doesn't exists? this VLC is an upgrade, and this crash log come from previous version
1940     BOOL areCrashLogsTooOld = ![[NSUserDefaults standardUserDefaults] integerForKey:@"LatestCrashReportYear"];
1941     NSString * latestLog = [self latestCrashLogPathPreviouslySeen:NO];
1942     if( latestLog && !areCrashLogsTooOld )
1943         [NSApp runModalForWindow: o_crashrep_win];
1944     [o_pool release];
1945 }
1946
1947 - (IBAction)crashReporterAction:(id)sender
1948 {
1949     if( sender == o_crashrep_send_btn )
1950         [self sendCrashLog:[NSString stringWithContentsOfFile: [self latestCrashLogPath] encoding: NSUTF8StringEncoding error: NULL] withUserComment: [o_crashrep_fld string]];
1951
1952     [NSApp stopModal];
1953     [o_crashrep_win orderOut: sender];
1954 }
1955
1956 - (IBAction)openCrashLog:(id)sender
1957 {
1958     NSString * latestLog = [self latestCrashLogPath];
1959     if( latestLog )
1960     {
1961         [[NSWorkspace sharedWorkspace] openFile: latestLog withApplication: @"Console"];
1962     }
1963     else
1964     {
1965         NSBeginInformationalAlertSheet(_NS("No CrashLog found"), _NS("Continue"), nil, nil, o_msgs_panel, self, NULL, NULL, nil, _NS("Couldn't find any trace of a previous crash.") );
1966     }
1967 }
1968
1969 #pragma mark -
1970 #pragma mark Remove old prefs
1971
1972 - (void)_removeOldPreferences
1973 {
1974     static NSString * kVLCPreferencesVersion = @"VLCPreferencesVersion";
1975     static const int kCurrentPreferencesVersion = 2;
1976     int version = [[NSUserDefaults standardUserDefaults] integerForKey:kVLCPreferencesVersion];
1977     if( version >= kCurrentPreferencesVersion ) return;
1978
1979     if( version == 1 )
1980     {
1981         [[NSUserDefaults standardUserDefaults] setInteger:kCurrentPreferencesVersion forKey:kVLCPreferencesVersion];
1982         [[NSUserDefaults standardUserDefaults] synchronize];
1983
1984         if (![[VLCCoreInteraction sharedInstance] fixPreferences])
1985             return;
1986         else
1987             config_SaveConfigFile( VLCIntf ); // we need to do manually, since we won't quit libvlc cleanly
1988     }
1989     else
1990     {
1991         NSArray *libraries = NSSearchPathForDirectoriesInDomains(NSLibraryDirectory,
1992             NSUserDomainMask, YES);
1993         if( !libraries || [libraries count] == 0) return;
1994         NSString * preferences = [[libraries objectAtIndex:0] stringByAppendingPathComponent:@"Preferences"];
1995
1996         /* File not found, don't attempt anything */
1997         if(![[NSFileManager defaultManager] fileExistsAtPath:[preferences stringByAppendingPathComponent:@"org.videolan.vlc"]] &&
1998            ![[NSFileManager defaultManager] fileExistsAtPath:[preferences stringByAppendingPathComponent:@"org.videolan.vlc.plist"]] )
1999         {
2000             [[NSUserDefaults standardUserDefaults] setInteger:kCurrentPreferencesVersion forKey:kVLCPreferencesVersion];
2001             return;
2002         }
2003
2004         int res = NSRunInformationalAlertPanel(_NS("Remove old preferences?"),
2005                     _NS("We just found an older version of VLC's preferences files."),
2006                     _NS("Move To Trash and Relaunch VLC"), _NS("Ignore"), nil, nil);
2007         if( res != NSOKButton )
2008         {
2009             [[NSUserDefaults standardUserDefaults] setInteger:kCurrentPreferencesVersion forKey:kVLCPreferencesVersion];
2010             return;
2011         }
2012
2013         NSArray * ourPreferences = [NSArray arrayWithObjects:@"org.videolan.vlc.plist", @"VLC", @"org.videolan.vlc", nil];
2014
2015         /* Move the file to trash so that user can find them later */
2016         [[NSWorkspace sharedWorkspace] performFileOperation:NSWorkspaceRecycleOperation source:preferences destination:nil files:ourPreferences tag:0];
2017
2018         /* really reset the defaults from now on */
2019         [NSUserDefaults resetStandardUserDefaults];
2020
2021         [[NSUserDefaults standardUserDefaults] setInteger:kCurrentPreferencesVersion forKey:kVLCPreferencesVersion];
2022         [[NSUserDefaults standardUserDefaults] synchronize];
2023     }
2024
2025     /* Relaunch now */
2026     const char * path = [[[NSBundle mainBundle] executablePath] UTF8String];
2027
2028     /* For some reason we need to fork(), not just execl(), which reports a ENOTSUP then. */
2029     if(fork() != 0)
2030     {
2031         exit(0);
2032         return;
2033     }
2034     execl(path, path, NULL);
2035 }
2036
2037 #pragma mark -
2038 #pragma mark Errors, warnings and messages
2039 - (IBAction)updateMessagesPanel:(id)sender
2040 {
2041     [self windowDidBecomeKey:nil];
2042 }
2043
2044 - (IBAction)showMessagesPanel:(id)sender
2045 {
2046     [o_msgs_panel makeKeyAndOrderFront: sender];
2047 }
2048
2049 - (void)windowDidBecomeKey:(NSNotification *)o_notification
2050 {
2051     [o_msgs_table reloadData];
2052     [o_msgs_table scrollRowToVisible: [o_msg_arr count] - 1];
2053 }
2054
2055 - (NSInteger)numberOfRowsInTableView:(NSTableView *)aTableView
2056 {
2057     if (aTableView == o_msgs_table)
2058         return [o_msg_arr count];
2059     return 0;
2060 }
2061
2062 - (id)tableView:(NSTableView *)aTableView objectValueForTableColumn:(NSTableColumn *)aTableColumn row:(NSInteger)rowIndex
2063 {
2064     NSMutableAttributedString *result = NULL;
2065
2066     [o_msg_lock lock];
2067     if( rowIndex < [o_msg_arr count] )
2068         result = [o_msg_arr objectAtIndex: rowIndex];
2069     [o_msg_lock unlock];
2070
2071     if( result != NULL )
2072         return result;
2073     else
2074         return @"";
2075 }
2076
2077 - (void)processReceivedlibvlcMessage:(const msg_item_t *) item ofType: (int)i_type withStr: (char *)str
2078 {
2079     if (o_msg_arr)
2080     {
2081         NSColor *o_white = [NSColor whiteColor];
2082         NSColor *o_red = [NSColor redColor];
2083         NSColor *o_yellow = [NSColor yellowColor];
2084         NSColor *o_gray = [NSColor grayColor];
2085         NSString * firstString, * secondString;
2086
2087         NSColor * pp_color[4] = { o_white, o_red, o_yellow, o_gray };
2088         static const char * ppsz_type[4] = { ": ", " error: ", " warning: ", " debug: " };
2089
2090         NSDictionary *o_attr;
2091         NSMutableAttributedString *o_msg_color;
2092
2093         [o_msg_lock lock];
2094
2095         if( [o_msg_arr count] + 2 > 600 )
2096         {
2097             [o_msg_arr removeObjectAtIndex: 0];
2098             [o_msg_arr removeObjectAtIndex: 1];
2099         }
2100         firstString = [NSString stringWithFormat:@"%s%s", item->psz_module, ppsz_type[i_type]];
2101         secondString = [NSString stringWithFormat:@"%@%s\n", firstString, str];
2102
2103         o_attr = [NSDictionary dictionaryWithObject: pp_color[i_type]  forKey: NSForegroundColorAttributeName];
2104         o_msg_color = [[NSMutableAttributedString alloc] initWithString: secondString attributes: o_attr];
2105         o_attr = [NSDictionary dictionaryWithObject: pp_color[3] forKey: NSForegroundColorAttributeName];
2106         [o_msg_color setAttributes: o_attr range: NSMakeRange( 0, [firstString length] )];
2107         [o_msg_arr addObject: [o_msg_color autorelease]];
2108
2109         b_msg_arr_changed = YES;
2110         [o_msg_lock unlock];
2111     }
2112 }
2113
2114 - (IBAction)saveDebugLog:(id)sender
2115 {
2116     NSSavePanel * saveFolderPanel = [[NSSavePanel alloc] init];
2117
2118     [saveFolderPanel setCanSelectHiddenExtension: NO];
2119     [saveFolderPanel setCanCreateDirectories: YES];
2120     [saveFolderPanel setAllowedFileTypes: [NSArray arrayWithObject:@"rtf"]];
2121     [saveFolderPanel beginSheetForDirectory:nil file: [NSString stringWithFormat: _NS("VLC Debug Log (%s).rtf"), VERSION_MESSAGE] modalForWindow: o_msgs_panel modalDelegate:self didEndSelector:@selector(saveDebugLogAsRTF:returnCode:contextInfo:) contextInfo:nil];
2122 }
2123
2124 - (void)saveDebugLogAsRTF: (NSSavePanel *)sheet returnCode: (int)returnCode contextInfo: (void *)contextInfo
2125 {
2126     if( returnCode == NSOKButton )
2127     {
2128         NSUInteger count = [o_msg_arr count];
2129         NSMutableAttributedString * string = [[NSMutableAttributedString alloc] init];
2130         for (NSUInteger i = 0; i < count; i++)
2131         {
2132             [string appendAttributedString: [o_msg_arr objectAtIndex: i]];
2133         }
2134
2135         NSData *data = [string RTFFromRange:NSMakeRange( 0, [string length] )
2136                          documentAttributes:[NSDictionary dictionaryWithObject: NSRTFTextDocumentType forKey: NSDocumentTypeDocumentAttribute]];        
2137
2138         if( [data writeToFile: [[sheet URL] path] atomically: YES] == NO )
2139             msg_Warn( p_intf, "Error while saving the debug log" );
2140
2141         [string release];
2142     }
2143 }
2144
2145 #pragma mark -
2146 #pragma mark Playlist toggling
2147
2148 - (void)updateTogglePlaylistState
2149 {
2150     [[self playlist] outlineViewSelectionDidChange: NULL];
2151 }
2152
2153 #pragma mark -
2154
2155 @end
2156
2157 @implementation VLCMain (Internal)
2158
2159 - (void)handlePortMessage:(NSPortMessage *)o_msg
2160 {
2161     id ** val;
2162     NSData * o_data;
2163     NSValue * o_value;
2164     NSInvocation * o_inv;
2165     NSConditionLock * o_lock;
2166
2167     o_data = [[o_msg components] lastObject];
2168     o_inv = *((NSInvocation **)[o_data bytes]);
2169     [o_inv getArgument: &o_value atIndex: 2];
2170     val = (id **)[o_value pointerValue];
2171     [o_inv setArgument: val[1] atIndex: 2];
2172     o_lock = *(val[0]);
2173
2174     [o_lock lock];
2175     [o_inv invoke];
2176     [o_lock unlockWithCondition: 1];
2177 }
2178 - (void)resetMediaKeyJump
2179 {
2180     b_mediakeyJustJumped = NO;
2181 }
2182 - (void)coreChangedMediaKeySupportSetting: (NSNotification *)o_notification
2183 {
2184     b_mediaKeySupport = var_InheritBool( VLCIntf, "macosx-mediakeys" );
2185     if (b_mediaKeySupport)
2186     {
2187         if (!o_mediaKeyController)
2188             o_mediaKeyController = [[SPMediaKeyTap alloc] initWithDelegate:self];
2189         [o_mediaKeyController startWatchingMediaKeys];
2190     }
2191     else if (!b_mediaKeySupport && o_mediaKeyController)
2192         [o_mediaKeyController stopWatchingMediaKeys];
2193 }
2194
2195 @end
2196
2197 /*****************************************************************************
2198  * VLCApplication interface
2199  *****************************************************************************/
2200
2201 @implementation VLCApplication
2202 // when user selects the quit menu from dock it sends a terminate:
2203 // but we need to send a stop: to properly exits libvlc.
2204 // However, we are not able to change the action-method sent by this standard menu item.
2205 // thus we override terminate: to send a stop:
2206 // see [af97f24d528acab89969d6541d83f17ce1ecd580] that introduced the removal of setjmp() and longjmp()
2207 - (void)terminate:(id)sender
2208 {
2209     [self activateIgnoringOtherApps:YES];
2210     [self stop:sender];
2211 }
2212
2213 @end