intf.m 71.1 KB
Newer Older
1
/*****************************************************************************
2
 * intf.m: MacOS X interface module
3
 *****************************************************************************
4
 * Copyright (C) 2002-2012 VLC authors and VideoLAN
5
 * $Id$
6
7
8
 *
 * Authors: Jon Lech Johansen <jon-vl@nanocrew.net>
 *          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
13
14
15
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
16
 *
17
18
19
20
21
22
23
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
dionoea's avatar
dionoea committed
24
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
25
26
27
28
29
 *****************************************************************************/

/*****************************************************************************
 * Preamble
 *****************************************************************************/
30
31
32
33
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif

34
35
36
#include <stdlib.h>                                      /* malloc(), free() */
#include <sys/param.h>                                    /* for MAXPATHLEN */
#include <string.h>
hartman's avatar
hartman committed
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_display.h>
45
#include <unistd.h> /* execl() */
46

47
#import "CompatibilityFixes.h"
48
#import "intf.h"
49
#import "StringUtility.h"
50
#import "MainMenu.h"
51
#import "VideoView.h"
52
53
#import "prefs.h"
#import "playlist.h"
54
#import "playlistinfo.h"
55
56
57
58
#import "controls.h"
#import "open.h"
#import "wizard.h"
#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
#import "VLCVoutWindowController.h"
66

67
#import <AddressBook/AddressBook.h>         /* for crashlog send mechanism */
Rafaël Carré's avatar
Rafaël Carré committed
68
#import <Sparkle/Sparkle.h>                 /* we're the update delegate */
69

70
71
72
/*****************************************************************************
 * Local prototypes.
 *****************************************************************************/
73
static void Run (intf_thread_t *p_intf);
74

75
76
77
78
static void updateProgressPanel (void *, const char *, float);
static bool checkProgressPanel (void *);
static void destroyProgressPanel (void *);

79
static void MsgCallback(void *data, int type, const msg_item_t *item, const char *format, va_list ap);
80

81
82
83
84
85
86
87
88
89
90
static int InputEvent(vlc_object_t *, const char *,
                      vlc_value_t, vlc_value_t, void *);
static int PLItemChanged(vlc_object_t *, const char *,
                         vlc_value_t, vlc_value_t, void *);
static int PlaylistUpdated(vlc_object_t *, const char *,
                           vlc_value_t, vlc_value_t, void *);
static int PlaybackModeUpdated(vlc_object_t *, const char *,
                               vlc_value_t, vlc_value_t, void *);
static int VolumeUpdated(vlc_object_t *, const char *,
                         vlc_value_t, vlc_value_t, void *);
91

92
#pragma mark -
93
#pragma mark VLC Interface Object Callbacks
94

95
96
97
/*****************************************************************************
 * OpenIntf: initialize interface
 *****************************************************************************/
98
int OpenIntf (vlc_object_t *p_this)
Jérome Decoodt's avatar
Jérome Decoodt committed
99
{
100
101
    NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
    [VLCApplication sharedApplication];
102

103
104
    intf_thread_t *p_intf = (intf_thread_t*) p_this;

105
106
    p_intf->p_sys = malloc(sizeof(intf_sys_t));
    if (p_intf->p_sys == NULL)
Jean-Paul Saman's avatar
Jean-Paul Saman committed
107
        return VLC_ENOMEM;
108

109
    memset(p_intf->p_sys, 0, sizeof(*p_intf->p_sys));
Jérome Decoodt's avatar
Jérome Decoodt committed
110

111
    /* subscribe to LibVLCCore's messages */
112
    vlc_Subscribe(&p_intf->p_sys->sub, MsgCallback, NULL);
113

114
    Run(p_intf);
115

116
    [o_pool release];
Jean-Paul Saman's avatar
Jean-Paul Saman committed
117
    return VLC_SUCCESS;
118
119
120
121
122
}

/*****************************************************************************
 * CloseIntf: destroy interface
 *****************************************************************************/
123
void CloseIntf (vlc_object_t *p_this)
124
125
{
    intf_thread_t *p_intf = (intf_thread_t*) p_this;
Jérome Decoodt's avatar
Jérome Decoodt committed
126

127
    free(p_intf->p_sys);
128
129
}

130
static int WindowControl(vout_window_t *, int i_query, va_list);
131

132
int WindowOpen(vout_window_t *p_wnd, const vout_window_cfg_t *cfg)
133
{
134
    NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
135
136
    intf_thread_t *p_intf = VLCIntf;
    if (!p_intf) {
137
        msg_Err(p_wnd, "Mac OS X interface not found");
138
139
140
141
142
143
144
        return VLC_EGENERIC;
    }

    int i_x = cfg->x;
    int i_y = cfg->y;
    unsigned i_width = cfg->width;
    unsigned i_height = cfg->height;
145
    p_wnd->handle.nsobject = [[VLCMain sharedInstance] getVideoViewAtPositionX: &i_x Y: &i_y withWidth: &i_width andHeight: &i_height forWindow: p_wnd];
146

147
148
    if (!p_wnd->handle.nsobject) {
        msg_Err(p_wnd, "got no video view from the interface");
149
        [o_pool release];
150
151
152
        return VLC_EGENERIC;
    }

153
154
155
156
157
158
159
160
161
162
163
164
    // TODO: this seems to be strange. Why not just allocating in the right size?
    // This could avoid strange resize-animations...
    NSSize newSize = NSMakeSize(cfg->width, cfg->height);
    SEL sel = @selector(setNativeVideoSize:forWindow:);
    NSInvocation *inv = [NSInvocation invocationWithMethodSignature:[[[VLCMain sharedInstance] voutController] methodSignatureForSelector:sel]];
    [inv setTarget:[[VLCMain sharedInstance] voutController]];
    [inv setSelector:sel];
    [inv setArgument:&newSize atIndex:2]; // starting at 2!
    [inv setArgument:&p_wnd atIndex:3];
    [inv performSelectorOnMainThread:@selector(invoke) withObject:nil
                       waitUntilDone:NO];

165
166
167
168
169
170
171
172
173
174
175
176
177
178
    // TODO: find a cleaner way for "start in fullscreen"
    if (var_GetBool(pl_Get(VLCIntf), "fullscreen")) {
        int i_full = 1;

        SEL sel = @selector(setFullscreen:forWindow:);
        NSInvocation *inv = [NSInvocation invocationWithMethodSignature:[[VLCMain sharedInstance] methodSignatureForSelector:sel]];
        [inv setTarget:[VLCMain sharedInstance]];
        [inv setSelector:sel];
        [inv setArgument:&i_full atIndex:2];
        [inv setArgument:&p_wnd atIndex:3];
        [inv performSelectorOnMainThread:@selector(invoke) withObject:nil
                           waitUntilDone:NO];
    }

179
    [[VLCMain sharedInstance] setActiveVideoPlayback: YES];
180
181
    p_wnd->control = WindowControl;
    p_wnd->sys = (vout_window_sys_t *)VLCIntf;
182
    [o_pool release];
183
184
185
    return VLC_SUCCESS;
}

186
static int WindowControl(vout_window_t *p_wnd, int i_query, va_list args)
187
{
188
    switch(i_query) {
189
190
        case VOUT_WINDOW_SET_STATE:
        {
191
            unsigned i_state = va_arg(args, unsigned);
192
193
194
195
196
            [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(setWindowLevel:) withObject:[NSNumber numberWithUnsignedInt:i_state] waitUntilDone:NO];
            return VLC_SUCCESS;
        }
        case VOUT_WINDOW_SET_SIZE:
        {
197
198
            NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];

199
200
            unsigned int i_width  = va_arg(args, unsigned int);
            unsigned int i_height = va_arg(args, unsigned int);
201
202
203
204
205
206
207
208
209
210
211
212

            NSSize newSize = NSMakeSize(i_width, i_height);            
            SEL sel = @selector(setNativeVideoSize:forWindow:);
            NSInvocation *inv = [NSInvocation invocationWithMethodSignature:[[[VLCMain sharedInstance] voutController] methodSignatureForSelector:sel]];
            [inv setTarget:[[VLCMain sharedInstance] voutController]];
            [inv setSelector:sel];
            [inv setArgument:&newSize atIndex:2]; // starting at 2!
            [inv setArgument:&p_wnd atIndex:3];
            [inv performSelectorOnMainThread:@selector(invoke) withObject:nil
                               waitUntilDone:NO];

            [o_pool release];
213
214
215
216
217
            return VLC_SUCCESS;
        }
        case VOUT_WINDOW_SET_FULLSCREEN:
        {
            NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
218
            int i_full = va_arg(args, int);
219
220
221
222
223
224
225
226
227
228
229
230

            SEL sel = @selector(setFullscreen:forWindow:);
            NSInvocation *inv = [NSInvocation invocationWithMethodSignature:[[VLCMain sharedInstance] methodSignatureForSelector:sel]];
            [inv setTarget:[VLCMain sharedInstance]];
            [inv setSelector:sel];
            [inv setArgument:&i_full atIndex:2]; // starting at 2!
            [inv setArgument:&p_wnd atIndex:3];
            [inv performSelectorOnMainThread:@selector(invoke) withObject:nil
                               waitUntilDone:NO];


            //[[VLCMain sharedInstance] performSelectorOnMainThread:@selector(fullscreenChanged:) withObject:[NSValue valueWithPointer:p_wnd] waitUntilDone:NO];
231
232
233
234
            [o_pool release];
            return VLC_SUCCESS;
        }
        default:
235
            msg_Warn(p_wnd, "unsupported control query");
236
            return VLC_EGENERIC;
237
    }
238
239
}

240
void WindowClose(vout_window_t *p_wnd)
241
{
242
    NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
243

244
245
    [[[VLCMain sharedInstance] voutController] performSelectorOnMainThread:@selector(removeVoutforDisplay:) withObject:[NSValue valueWithPointer:p_wnd] waitUntilDone:NO];

246
    [o_pool release];
247
248
}

249
250
251
/*****************************************************************************
 * Run: main loop
 *****************************************************************************/
252
static NSLock * o_appLock = nil;    // controls access to f_appExit
253

254
static void Run(intf_thread_t *p_intf)
255
{
256
    NSAutoreleasePool * o_pool = [[NSAutoreleasePool alloc] init];
257
    [VLCApplication sharedApplication];
258

259
260
    o_appLock = [[NSLock alloc] init];

hartman's avatar
-    
hartman committed
261
    [[VLCMain sharedInstance] setIntf: p_intf];
hartman's avatar
hartman committed
262
    [NSBundle loadNibNamed: @"MainMenu" owner: NSApp];
263

264
265
    [NSApp run];
    [[VLCMain sharedInstance] applicationWillTerminate:nil];
266
    [o_appLock release];
267
    [o_pool release];
268
269

    raise(SIGTERM);
270
271
}

272
273
#pragma mark -
#pragma mark Variables Callback
274

275
276
277
/*****************************************************************************
 * MsgCallback: Callback triggered by the core once a new debug message is
 * ready to be displayed. We store everything in a NSArray in our Cocoa part
278
 * of this file.
279
 *****************************************************************************/
280
static void MsgCallback(void *data, int type, const msg_item_t *item, const char *format, va_list ap)
281
282
{
    int canc = vlc_savecancel();
283
    char *str;
284

285
286
    if (vasprintf(&str, format, ap) == -1) {
        vlc_restorecancel(canc);
287
        return;
288
    }
289

290
    NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
291
    [[VLCMain sharedInstance] processReceivedlibvlcMessage: item ofType: type withStr: str];
292
    [o_pool release];
293

294
295
    vlc_restorecancel(canc);
    free(str);
296
297
}

298
299
static int InputEvent(vlc_object_t *p_this, const char *psz_var,
                       vlc_value_t oldval, vlc_value_t new_val, void *param)
300
{
301
    NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
302
303
    switch (new_val.i_int) {
        case INPUT_EVENT_STATE:
304
            [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(playbackStatusUpdated) withObject: nil waitUntilDone:NO];
305
306
            break;
        case INPUT_EVENT_RATE:
307
            [[[VLCMain sharedInstance] mainMenu] performSelectorOnMainThread:@selector(updatePlaybackRate) withObject: nil waitUntilDone:NO];
308
309
            break;
        case INPUT_EVENT_POSITION:
310
            [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updatePlaybackPosition) withObject: nil waitUntilDone:NO];
311
312
313
            break;
        case INPUT_EVENT_TITLE:
        case INPUT_EVENT_CHAPTER:
314
            [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateMainMenu) withObject: nil waitUntilDone:NO];
315
316
            break;
        case INPUT_EVENT_CACHE:
317
            [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateMainWindow) withObject: nil waitUntilDone: NO];
318
319
320
321
322
323
324
325
326
327
328
329
330
331
            break;
        case INPUT_EVENT_STATISTICS:
            [[[VLCMain sharedInstance] info] performSelectorOnMainThread:@selector(updateStatistics) withObject: nil waitUntilDone: NO];
            break;
        case INPUT_EVENT_ES:
            break;
        case INPUT_EVENT_TELETEXT:
            break;
        case INPUT_EVENT_AOUT:
            break;
        case INPUT_EVENT_VOUT:
            break;
        case INPUT_EVENT_ITEM_META:
        case INPUT_EVENT_ITEM_INFO:
332
            [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateMainMenu) withObject: nil waitUntilDone:NO];
333
            [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateName) withObject: nil waitUntilDone:NO];
334
            [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateInfoandMetaPanel) withObject: nil waitUntilDone:NO];
335
336
337
338
            break;
        case INPUT_EVENT_BOOKMARK:
            break;
        case INPUT_EVENT_RECORD:
339
            [[VLCMain sharedInstance] updateRecordState: var_GetBool(p_this, "record")];
340
341
            break;
        case INPUT_EVENT_PROGRAM:
342
            [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateMainMenu) withObject: nil waitUntilDone:NO];
343
344
345
346
347
348
349
            break;
        case INPUT_EVENT_ITEM_EPG:
            break;
        case INPUT_EVENT_SIGNAL:
            break;

        case INPUT_EVENT_ITEM_NAME:
350
            [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateName) withObject: nil waitUntilDone:NO];
351
            [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(playlistUpdated) withObject: nil waitUntilDone:NO];
352
353
354
355
            break;

        case INPUT_EVENT_AUDIO_DELAY:
        case INPUT_EVENT_SUBTITLE_DELAY:
356
            [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateDelays) withObject:nil waitUntilDone:NO];
357
358
359
            break;

        case INPUT_EVENT_DEAD:
360
            [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateName) withObject: nil waitUntilDone:NO];
361
            [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updatePlaybackPosition) withObject:nil waitUntilDone:NO];
362
363
364
            break;

        case INPUT_EVENT_ABORT:
365
            [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateName) withObject: nil waitUntilDone:NO];
366
            [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updatePlaybackPosition) withObject:nil waitUntilDone:NO];
367
368
369
            break;

        default:
370
            //msg_Warn(p_this, "unhandled input event (%lld)", new_val.i_int);
371
            break;
Jean-Paul Saman's avatar
Jean-Paul Saman committed
372
    }
373

374
    [o_pool release];
375
376
377
    return VLC_SUCCESS;
}

378
379
static int PLItemChanged(vlc_object_t *p_this, const char *psz_var,
                         vlc_value_t oldval, vlc_value_t new_val, void *param)
380
{
381
    NSAutoreleasePool * o_pool = [[NSAutoreleasePool alloc] init];
382
383
    [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(PlaylistItemChanged) withObject:nil waitUntilDone:NO];

384
    [o_pool release];
385
386
387
    return VLC_SUCCESS;
}

388
389
static int PlaylistUpdated(vlc_object_t *p_this, const char *psz_var,
                         vlc_value_t oldval, vlc_value_t new_val, void *param)
390
{
391
    NSAutoreleasePool * o_pool = [[NSAutoreleasePool alloc] init];
392
393
    [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(playlistUpdated) withObject:nil waitUntilDone:NO];

394
    [o_pool release];
395
396
397
    return VLC_SUCCESS;
}

398
399
static int PlaybackModeUpdated(vlc_object_t *p_this, const char *psz_var,
                         vlc_value_t oldval, vlc_value_t new_val, void *param)
400
{
401
    NSAutoreleasePool * o_pool = [[NSAutoreleasePool alloc] init];
402
    [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(playbackModeUpdated) withObject:nil waitUntilDone:NO];
403

404
    [o_pool release];
405
406
407
    return VLC_SUCCESS;
}

408
409
static int VolumeUpdated(vlc_object_t *p_this, const char *psz_var,
                         vlc_value_t oldval, vlc_value_t new_val, void *param)
410
{
411
    NSAutoreleasePool * o_pool = [[NSAutoreleasePool alloc] init];
412
    [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(updateVolume) withObject:nil waitUntilDone:NO];
413

414
    [o_pool release];
415
416
417
418
419
420
421
422
    return VLC_SUCCESS;
}

/*****************************************************************************
 * ShowController: Callback triggered by the show-intf playlist variable
 * through the ShowIntf-control-intf, to let us show the controller-win;
 * usually when in fullscreen-mode
 *****************************************************************************/
423
424
static int ShowController(vlc_object_t *p_this, const char *psz_variable,
                     vlc_value_t old_val, vlc_value_t new_val, void *param)
425
426
{
    intf_thread_t * p_intf = VLCIntf;
427
428
429
430
    if (p_intf && p_intf->p_sys) {
        playlist_t * p_playlist = pl_Get(p_intf);
        BOOL b_fullscreen = var_GetBool(p_playlist, "fullscreen");
        if (strcmp(psz_variable, "intf-toggle-fscontrol") || b_fullscreen)
431
432
433
            [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(showFullscreenController) withObject:nil waitUntilDone:NO];
        else
            [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(showMainWindow) withObject:nil waitUntilDone:NO];
434
    }
435
436
437
    return VLC_SUCCESS;
}

438
/*****************************************************************************
439
 * DialogCallback: Callback triggered by the "dialog-*" variables
440
 * to let the intf display error and interaction dialogs
441
 *****************************************************************************/
442
static int DialogCallback(vlc_object_t *p_this, const char *type, vlc_value_t previous, vlc_value_t value, void *data)
443
444
{
    NSAutoreleasePool * o_pool = [[NSAutoreleasePool alloc] init];
445
446
    VLCMain *interface = (VLCMain *)data;

447
    if ([[NSString stringWithUTF8String: type] isEqualToString: @"dialog-progress-bar"]) {
448
449
450
451
452
453
454
455
        /* the progress panel needs to update itself and therefore wants special treatment within this context */
        dialog_progress_bar_t *p_dialog = (dialog_progress_bar_t *)value.p_address;

        p_dialog->pf_update = updateProgressPanel;
        p_dialog->pf_check = checkProgressPanel;
        p_dialog->pf_destroy = destroyProgressPanel;
        p_dialog->p_sys = VLCIntf->p_libvlc;
    }
456

457
    NSValue *o_value = [NSValue valueWithPointer:value.p_address];
458
    [[VLCCoreDialogProvider sharedInstance] performEventWithObject: o_value ofType: type];
459

460
461
462
    [o_pool release];
    return VLC_SUCCESS;
}
bigben's avatar
bigben committed
463

464
465
466
467
468
void updateProgressPanel (void *priv, const char *text, float value)
{
    NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];

    NSString *o_txt;
469
    if (text != NULL)
470
471
472
473
474
475
476
477
478
479
480
481
        o_txt = [NSString stringWithUTF8String: text];
    else
        o_txt = @"";

    [[[VLCMain sharedInstance] coreDialogProvider] updateProgressPanelWithText: o_txt andNumber: (double)(value * 1000.)];

    [o_pool release];
}

void destroyProgressPanel (void *priv)
{
    NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
482
    [[[VLCMain sharedInstance] coreDialogProvider] performSelectorOnMainThread:@selector(destroyProgressPanel) withObject:nil waitUntilDone:YES];
483
484
485
486
487
488
489
490
491
492
    [o_pool release];
}

bool checkProgressPanel (void *priv)
{
    NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];
    return [[[VLCMain sharedInstance] coreDialogProvider] progressCancelled];
    [o_pool release];
}

493
494
495
496
497
498
#pragma mark -
#pragma mark Helpers

input_thread_t *getInput(void)
{
    intf_thread_t *p_intf = VLCIntf;
499
500
    if (!p_intf)
        return NULL;
501
    return pl_CurrentInput(p_intf);
502
503
504
505
506
}

vout_thread_t *getVout(void)
{
    input_thread_t *p_input = getInput();
507
508
    if (!p_input)
        return NULL;
509
510
511
512
513
    vout_thread_t *p_vout = input_GetVout(p_input);
    vlc_object_release(p_input);
    return p_vout;
}

514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
vout_thread_t *getVoutForActiveWindow(void)
{
    vout_thread_t *p_vout = nil;

    id currentWindow = [NSApp keyWindow];
    if ([currentWindow respondsToSelector:@selector(videoView)]) {
        VLCVoutView *videoView = [currentWindow videoView];
        if (videoView) {
            p_vout = [videoView voutThread];
        }
    }

    if (!p_vout)
        p_vout = getVout();

    return p_vout;
}

532
audio_output_t *getAout(void)
533
534
{
    input_thread_t *p_input = getInput();
535
536
    if (!p_input)
        return NULL;
537
    audio_output_t *p_aout = input_GetAout(p_input);
538
539
540
541
    vlc_object_release(p_input);
    return p_aout;
}

542
#pragma mark -
543
544
545
546
547
#pragma mark Private

@interface VLCMain ()
- (void)_removeOldPreferences;
@end
548

549
/*****************************************************************************
Jérome Decoodt's avatar
Jérome Decoodt committed
550
 * VLCMain implementation
551
552
553
 *****************************************************************************/
@implementation VLCMain

554
@synthesize voutController=o_vout_controller;
555
@synthesize nativeFullscreenMode=b_nativeFullscreenMode;
556

557
558
559
#pragma mark -
#pragma mark Initialization

hartman's avatar
hartman committed
560
561
562
563
564
565
566
static VLCMain *_o_sharedMainInstance = nil;

+ (VLCMain *)sharedInstance
{
    return _o_sharedMainInstance ? _o_sharedMainInstance : [[self alloc] init];
}

Jérome Decoodt's avatar
Jérome Decoodt committed
567
- (id)init
hartman's avatar
hartman committed
568
{
569
    if (_o_sharedMainInstance) {
hartman's avatar
hartman committed
570
        [self dealloc];
571
        return _o_sharedMainInstance;
572
    } else
hartman's avatar
hartman committed
573
        _o_sharedMainInstance = [super init];
574

575
    p_intf = NULL;
576
    p_current_input = NULL;
577

578
    o_msg_lock = [[NSLock alloc] init];
579
    o_msg_arr = [[NSMutableArray arrayWithCapacity: 600] retain];
580

581
    o_open = [[VLCOpen alloc] init];
582
    o_coredialogs = [[VLCCoreDialogProvider alloc] init];
583
    o_info = [[VLCInfo alloc] init];
584
    o_mainmenu = [[VLCMainMenu alloc] init];
585
    o_coreinteraction = [[VLCCoreInteraction alloc] init];
586
    o_eyetv = [[VLCEyeTVController alloc] init];
587
    o_mainwindow = [[VLCMainWindow alloc] init];
588
589
590
591
592
593

    /* announce our launch to a potential eyetv plugin */
    [[NSDistributedNotificationCenter defaultCenter] postNotificationName: @"VLCOSXGUIInit"
                                                                   object: @"VLCEyeTVSupport"
                                                                 userInfo: NULL
                                                       deliverImmediately: YES];
594
595
596
597
598

    NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
    NSDictionary *appDefaults = [NSDictionary dictionaryWithObject:@"NO" forKey:@"LiveUpdateTheMessagesPanel"];
    [defaults registerDefaults:appDefaults];

599
600
    o_vout_controller = [[VLCVoutWindowController alloc] init];

hartman's avatar
hartman committed
601
602
603
    return _o_sharedMainInstance;
}

604
605
- (void)setIntf: (intf_thread_t *)p_mainintf
{
hartman's avatar
hartman committed
606
607
608
    p_intf = p_mainintf;
}

609
610
- (intf_thread_t *)intf
{
hartman's avatar
hartman committed
611
612
613
    return p_intf;
}

614
615
- (void)awakeFromNib
{
616
    playlist_t *p_playlist;
617
    vlc_value_t val;
618
619
    if (!p_intf) return;
    var_Create(p_intf, "intf-change", VLC_VAR_BOOL);
620

621
622
    /* Check if we already did this once. Opening the other nibs calls it too,
     because VLCMain is the owner */
623
624
    if (nib_main_loaded)
        return;
625

626
627
    [o_msgs_panel setExcludedFromWindowsMenu: YES];
    [o_msgs_panel setDelegate: self];
628

629
    p_playlist = pl_Get(p_intf);
hartman's avatar
hartman committed
630

631
    val.b_bool = false;
bigben's avatar
bigben committed
632

633
634
    var_AddCallback(p_intf->p_libvlc, "intf-toggle-fscontrol", ShowController, self);
    var_AddCallback(p_intf->p_libvlc, "intf-show", ShowController, self);
635
    //    var_AddCallback(p_playlist, "item-change", PLItemChanged, self);
636
637
638
639
640
641
642
643
644
645
    var_AddCallback(p_playlist, "item-current", PLItemChanged, self);
    var_AddCallback(p_playlist, "activity", PLItemChanged, self);
    var_AddCallback(p_playlist, "leaf-to-parent", PlaylistUpdated, self);
    var_AddCallback(p_playlist, "playlist-item-append", PlaylistUpdated, self);
    var_AddCallback(p_playlist, "playlist-item-deleted", PlaylistUpdated, self);
    var_AddCallback(p_playlist, "random", PlaybackModeUpdated, self);
    var_AddCallback(p_playlist, "repeat", PlaybackModeUpdated, self);
    var_AddCallback(p_playlist, "loop", PlaybackModeUpdated, self);
    var_AddCallback(p_playlist, "volume", VolumeUpdated, self);
    var_AddCallback(p_playlist, "mute", VolumeUpdated, self);
bigben's avatar
bigben committed
646

647
    if (!OSX_SNOW_LEOPARD) {
648
        if ([NSApp currentSystemPresentationOptions] & NSApplicationPresentationFullScreen)
649
            var_SetBool(p_playlist, "fullscreen", YES);
650
    }
651

652
    /* load our Core and Shared Dialogs nibs */
653
    nib_coredialogs_loaded = [NSBundle loadNibNamed:@"CoreDialogs" owner: NSApp];
654
    [NSBundle loadNibNamed:@"SharedDialogs" owner: NSApp];
655

656
    /* subscribe to various interactive dialogues */
657
658
659
660
661
662
663
664
665
666
667
    var_Create(p_intf, "dialog-error", VLC_VAR_ADDRESS);
    var_AddCallback(p_intf, "dialog-error", DialogCallback, self);
    var_Create(p_intf, "dialog-critical", VLC_VAR_ADDRESS);
    var_AddCallback(p_intf, "dialog-critical", DialogCallback, self);
    var_Create(p_intf, "dialog-login", VLC_VAR_ADDRESS);
    var_AddCallback(p_intf, "dialog-login", DialogCallback, self);
    var_Create(p_intf, "dialog-question", VLC_VAR_ADDRESS);
    var_AddCallback(p_intf, "dialog-question", DialogCallback, self);
    var_Create(p_intf, "dialog-progress-bar", VLC_VAR_ADDRESS);
    var_AddCallback(p_intf, "dialog-progress-bar", DialogCallback, self);
    dialog_Register(p_intf);
668

669
670
671
672
673
    /* init Apple Remote support */
    o_remote = [[AppleRemote alloc] init];
    [o_remote setClickCountEnabledButtons: kRemoteButtonPlay];
    [o_remote setDelegate: _o_sharedMainInstance];

674
    [o_msgs_refresh_btn setImage: [NSImage imageNamed: NSImageNameRefreshTemplate]];
675

676
    /* yeah, we are done */
677
678
    b_nativeFullscreenMode = NO;
#ifdef MAC_OS_X_VERSION_10_7
679
    if (!OSX_SNOW_LEOPARD)
680
        b_nativeFullscreenMode = var_InheritBool(p_intf, "macosx-nativefullscreenmode");
681
#endif
682
683
684

    /* recover stored audio device, if set
     * in case it was unplugged in the meantime, auhal will fall back on the default */
685
    int i_value = config_GetInt(p_intf, "macosx-audio-device");
686
    if (i_value > 0)
687
        var_SetInteger(pl_Get(VLCIntf), "audio-device", i_value);
688

689
    if (config_GetInt(VLCIntf, "macosx-icon-change")) {
690
691
692
693
694
695
696
697
698
699
700
701
702
        /* After day 354 of the year, the usual VLC cone is replaced by another cone
         * wearing a Father Xmas hat.
         * Note: this icon doesn't represent an endorsement of The Coca-Cola Company.
         */
        NSCalendar *gregorian =
        [[NSCalendar alloc] initWithCalendarIdentifier:NSGregorianCalendar];
        NSUInteger dayOfYear = [gregorian ordinalityOfUnit:NSDayCalendarUnit inUnit:NSYearCalendarUnit forDate:[NSDate date]];
        [gregorian release];

        if (dayOfYear >= 354)
            [[VLCApplication sharedApplication] setApplicationIconImage: [NSImage imageNamed:@"vlc-xmas"]];
    }

703
704
    [self initStrings];

705
    nib_main_loaded = TRUE;
706
707
}

708
- (void)applicationDidFinishLaunching:(NSNotification *)aNotification
709
{
710
711
    if (!p_intf)
        return;
712

713
714
    [self updateCurrentlyUsedHotkeys];

715
    /* init media key support */
716
717
    b_mediaKeySupport = var_InheritBool(VLCIntf, "macosx-mediakeys");
    if (b_mediaKeySupport) {
718
719
720
721
722
723
        o_mediaKeyController = [[SPMediaKeyTap alloc] initWithDelegate:self];
        [o_mediaKeyController startWatchingMediaKeys];
        [[NSUserDefaults standardUserDefaults] registerDefaults:[NSDictionary dictionaryWithObjectsAndKeys:
                                                                 [SPMediaKeyTap defaultMediaKeyUserBundleIdentifiers], kMediaKeyUsingBundleIdentifiersDefaultsKey,
                                                                 nil]];
    }
724
725
    [[NSNotificationCenter defaultCenter] addObserver: self selector: @selector(coreChangedMediaKeySupportSetting:) name: @"VLCMediaKeySupportSettingChanged" object: nil];

726
727
    [self _removeOldPreferences];

728
729
730
731
    /* Handle sleep notification */
    [[[NSWorkspace sharedWorkspace] notificationCenter] addObserver:self selector:@selector(computerWillSleep:)
           name:NSWorkspaceWillSleepNotification object:nil];

732
    [[VLCMain sharedInstance] performSelectorOnMainThread:@selector(lookForCrashLog) withObject:nil waitUntilDone:NO];
733

734
735
    /* we will need this, so let's load it here so the interface appears to be more responsive */
    nib_open_loaded = [NSBundle loadNibNamed:@"Open" owner: NSApp];
736
737
738
739
740

    /* update the main window */
    [o_mainwindow updateWindow];
    [o_mainwindow updateTimeSlider];
    [o_mainwindow updateVolumeSlider];
741
742
}

hartman's avatar
hartman committed
743
- (void)initStrings
744
{
745
746
    if (!p_intf)
        return;
747

Jérome Decoodt's avatar
Jérome Decoodt committed
748
    /* messages panel */
749
    [o_msgs_panel setTitle: _NS("Messages")];
750
751
    [o_msgs_crashlog_btn setTitle: _NS("Open CrashLog...")];
    [o_msgs_save_btn setTitle: _NS("Save this Log...")];
752

753
754
755
756
757
758
    /* crash reporter panel */
    [o_crashrep_send_btn setTitle: _NS("Send")];
    [o_crashrep_dontSend_btn setTitle: _NS("Don't Send")];
    [o_crashrep_title_txt setStringValue: _NS("VLC crashed previously")];
    [o_crashrep_win setTitle: _NS("VLC crashed previously")];
    [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, ...")];
759
760
    [o_crashrep_includeEmail_ckb setTitle: _NS("I agree to be possibly contacted about this bugreport.")];
    [o_crashrep_includeEmail_txt setStringValue: _NS("Only your default E-Mail address will be submitted, including no further information.")];
761
    [o_crashrep_dontaskagain_ckb setTitle: _NS("Don't ask again")];
762
763
}

764
765
766
767
#pragma mark -
#pragma mark Termination

- (void)applicationWillTerminate:(NSNotification *)notification
768
{
769
770
    /* don't allow a double termination call. If the user has
     * already invoked the quit then simply return this time. */
771
772
    static bool f_appExit = false;
    bool isTerminating;
773
774

    [o_appLock lock];
775
776
    isTerminating = f_appExit;
    f_appExit = true;
777
778
779
780
    [o_appLock unlock];

    if (isTerminating)
        return;
781

782
783
    if (notification == nil)
        [[NSNotificationCenter defaultCenter] postNotificationName: NSApplicationWillTerminateNotification object: nil];
784

785
    playlist_t * p_playlist = pl_Get(p_intf);
786
787
    int returnedValue = 0;

788
    /* always exit fullscreen on quit, otherwise we get ugly artifacts on the next launch */
789
    if (b_nativeFullscreenMode) {
790
791
792
793
        [o_mainwindow toggleFullScreen: self];
        [NSApp setPresentationOptions:(NSApplicationPresentationDefault)];
    }

794
    /* Save some interface state in configuration, at module quit */
795
796
797
    config_PutInt(p_intf, "random", var_GetBool(p_playlist, "random"));
    config_PutInt(p_intf, "loop", var_GetBool(p_playlist, "loop"));
    config_PutInt(p_intf, "repeat", var_GetBool(p_playlist, "repeat"));
798

799
    msg_Dbg(p_intf, "Terminating");
800

801
    /* unsubscribe from the interactive dialogues */
802
803
804
805
806
807
    dialog_Unregister(p_intf);
    var_DelCallback(p_intf, "dialog-error", DialogCallback, self);
    var_DelCallback(p_intf, "dialog-critical", DialogCallback, self);
    var_DelCallback(p_intf, "dialog-login", DialogCallback, self);
    var_DelCallback(p_intf, "dialog-question", DialogCallback, self);
    var_DelCallback(p_intf, "dialog-progress-bar", DialogCallback, self);
808
809
810
811
812
813
814
815
816
817
818
    //var_DelCallback(p_playlist, "item-change", PLItemChanged, self);
    var_DelCallback(p_playlist, "item-current", PLItemChanged, self);
    var_DelCallback(p_playlist, "activity", PLItemChanged, self);
    var_DelCallback(p_playlist, "leaf-to-parent", PlaylistUpdated, self);
    var_DelCallback(p_playlist, "playlist-item-append", PlaylistUpdated, self);
    var_DelCallback(p_playlist, "playlist-item-deleted", PlaylistUpdated, self);
    var_DelCallback(p_playlist, "random", PlaybackModeUpdated, self);
    var_DelCallback(p_playlist, "repeat", PlaybackModeUpdated, self);
    var_DelCallback(p_playlist, "loop", PlaybackModeUpdated, self);
    var_DelCallback(p_playlist, "volume", VolumeUpdated, self);
    var_DelCallback(p_playlist, "mute", VolumeUpdated, self);
819
    var_DelCallback(p_intf->p_libvlc, "intf-toggle-fscontrol", ShowController, self);
820
    var_DelCallback(p_intf->p_libvlc, "intf-show", ShowController, self);
821

822
823
824
    if (p_current_input) {
        var_DelCallback(p_current_input, "intf-event", InputEvent, [VLCMain sharedInstance]);
        vlc_object_release(p_current_input);
825
        p_current_input = NULL;
826
827
    }

828
829
    /* remove global observer watching for vout device changes correctly */
    [[NSNotificationCenter defaultCenter] removeObserver: self];
Jérome Decoodt's avatar
Jérome Decoodt committed
830

831
832
    /* release some other objects here, because it isn't sure whether dealloc
     * will be called later on */
833
    if (o_sprefs)
834
835
        [o_sprefs release];

836
    if (o_prefs)
837
        [o_prefs release];
838

839
    [o_open release];
840

841
    if (o_info)
842
        [o_info release];
hartman's avatar
hartman committed
843

844
    if (o_wizard)
845
846
847
848
        [o_wizard release];

    [crashLogURLConnection cancel];
    [crashLogURLConnection release];
849

850
    [o_coredialogs release];
851
852
    [o_eyetv release];

853
    /* unsubscribe from libvlc's debug messages */
854
    vlc_Unsubscribe(&p_intf->p_sys->sub);
855

856
857
    [o_msg_arr removeAllObjects];
    [o_msg_arr release];
858
    o_msg_arr = NULL;
859
860
    [o_usedHotkeys release];
    o_usedHotkeys = NULL;
861
862
863
864
865
866

    [o_msg_lock release];

    /* write cached user defaults to disk */
    [[NSUserDefaults standardUserDefaults] synchronize];

867
    [o_mainmenu release];
hartman's avatar
hartman committed
868

869
870
871
    [o_vout_controller release];
    o_vout_controller = nil;

872
    libvlc_Quit(p_intf->p_libvlc);
873