1 /*****************************************************************************
2 * dialogProvider.m: Minimal Dialog Provider for Mac OS X
3 *****************************************************************************
4 * Copyright (C) 2009-2010 the VideoLAN team
7 * Authors: Felix Paul Kühne <fkuehne at videolan dot org>
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2 of the License, or
12 * (at your option) any later version.
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
22 *****************************************************************************/
24 /*****************************************************************************
26 *****************************************************************************/
27 #import <stdlib.h> /* malloc(), free() */
34 #import <vlc_common.h>
35 #import <vlc_plugin.h>
36 #import <vlc_dialog.h>
37 #import <vlc_interface.h>
39 #import <Cocoa/Cocoa.h>
40 #import "VLCLoginPanel.h"
41 #import "VLCProgressPanel.h"
43 /*****************************************************************************
45 *****************************************************************************/
46 static int OpenIntf(vlc_object_t *);
47 static void CloseIntf(vlc_object_t *);
48 static void Run(intf_thread_t * );
50 static int DisplayError(vlc_object_t *,const char *,vlc_value_t,vlc_value_t,void * );
51 static int DisplayCritical(vlc_object_t *,const char *,vlc_value_t,vlc_value_t,void * );
52 static int DisplayQuestion(vlc_object_t *,const char *,vlc_value_t,vlc_value_t,void * );
53 static int DisplayLogin(vlc_object_t *,const char *,vlc_value_t,vlc_value_t,void * );
54 static int DisplayProgressPanelAction(vlc_object_t *,const char *,vlc_value_t,vlc_value_t,void * );
56 static void updateProgressPanel (void *, const char *, float);
57 static bool checkProgressPanel (void *);
58 static void destroyProgressPanel (void *);
60 @interface VLCDialogDisplayer : NSObject
62 VLCProgressPanel *_currentProgressBarPanel;
65 + (NSDictionary *)dictionaryForDialog:(const char *)title :(const char *)message :(const char *)yes :(const char *)no :(const char *)cancel;
67 - (void)displayError:(NSDictionary *)dialog;
68 - (void)displayCritical:(NSDictionary *)dialog;
69 - (NSNumber *)displayQuestion:(NSDictionary *)dialog;
70 - (NSDictionary *)displayLogin:(NSDictionary *)dialog;
72 - (void)displayProgressBar:(NSDictionary *)dict;
73 - (void)updateProgressPanel:(NSDictionary *)dict;
74 - (void)destroyProgressPanel;
75 - (NSNumber *)checkProgressPanel;
77 - (id)resultFromSelectorOnMainThread:(SEL)sel withObject:(id)object;
80 static inline NSDictionary *DictFromDialogFatal(dialog_fatal_t *dialog) {
81 return [VLCDialogDisplayer dictionaryForDialog:dialog->title :dialog->message :NULL :NULL :NULL];
83 static inline NSDictionary *DictFromDialogLogin(dialog_login_t *dialog) {
84 return [VLCDialogDisplayer dictionaryForDialog:dialog->title :dialog->message :NULL :NULL :NULL];
86 static inline NSDictionary *DictFromDialogQuestion(dialog_question_t *dialog) {
87 return [VLCDialogDisplayer dictionaryForDialog:dialog->title :dialog->message :dialog->yes :dialog->no :dialog->cancel];
89 static inline NSDictionary *DictFromDialogProgressBar(dialog_progress_bar_t *dialog) {
90 return [VLCDialogDisplayer dictionaryForDialog:dialog->title :dialog->message :NULL :NULL :dialog->cancel];
95 VLCDialogDisplayer *displayer;
102 /*****************************************************************************
104 *****************************************************************************/
107 /* Minimal interface. see intf.m */
108 set_shortname("Mac OS X Dialogs")
109 add_shortcut("macosx_dialog_provider")
110 add_shortcut("miosx")
111 set_description("Minimal Mac OS X Dialog Provider")
112 set_capability("interface", 50)
113 set_callbacks(OpenIntf, CloseIntf)
114 set_category(CAT_INTERFACE)
115 set_subcategory(SUBCAT_INTERFACE_MAIN)
118 /*****************************************************************************
119 * OpenIntf: initialize interface
120 *****************************************************************************/
121 int OpenIntf(vlc_object_t *p_this)
123 intf_thread_t *p_intf = (intf_thread_t*) p_this;
125 p_intf->p_sys = malloc(sizeof(intf_sys_t));
129 memset(p_intf->p_sys,0,sizeof(*p_intf->p_sys));
131 p_intf->p_sys->displayer = [[VLCDialogDisplayer alloc] init];
133 /* subscribe to various interactive dialogues */
134 var_Create(p_intf,"dialog-error",VLC_VAR_ADDRESS);
135 var_AddCallback(p_intf,"dialog-error",DisplayError,p_intf);
136 var_Create(p_intf,"dialog-critical",VLC_VAR_ADDRESS);
137 var_AddCallback(p_intf,"dialog-critical",DisplayCritical,p_intf);
138 var_Create(p_intf,"dialog-login",VLC_VAR_ADDRESS);
139 var_AddCallback(p_intf,"dialog-login",DisplayLogin,p_intf);
140 var_Create(p_intf,"dialog-question",VLC_VAR_ADDRESS);
141 var_AddCallback(p_intf,"dialog-question",DisplayQuestion,p_intf);
142 var_Create(p_intf,"dialog-progress-bar",VLC_VAR_ADDRESS);
143 var_AddCallback(p_intf,"dialog-progress-bar",DisplayProgressPanelAction,p_intf);
144 dialog_Register(p_intf);
146 msg_Dbg(p_intf,"Mac OS X dialog provider initialised");
151 /*****************************************************************************
152 * CloseIntf: destroy interface
153 *****************************************************************************/
154 void CloseIntf(vlc_object_t *p_this)
156 intf_thread_t *p_intf = (intf_thread_t*) p_this;
158 /* unsubscribe from the interactive dialogues */
159 dialog_Unregister(p_intf);
160 var_DelCallback(p_intf,"dialog-error",DisplayError,p_intf);
161 var_DelCallback(p_intf,"dialog-critical",DisplayCritical,p_intf);
162 var_DelCallback(p_intf,"dialog-login",DisplayLogin,p_intf);
163 var_DelCallback(p_intf,"dialog-question",DisplayQuestion,p_intf);
164 var_DelCallback(p_intf,"dialog-progress-bar",DisplayProgressPanelAction,p_intf);
166 [p_intf->p_sys->displayer release];
168 msg_Dbg(p_intf,"Mac OS X dialog provider closed");
173 /*****************************************************************************
174 * Callbacks triggered by the "dialog-*" variables
175 *****************************************************************************/
176 static int DisplayError(vlc_object_t *p_this, const char *type, vlc_value_t previous, vlc_value_t value, void *data)
178 NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
179 dialog_fatal_t *dialog = value.p_address;
180 intf_thread_t *p_intf = (intf_thread_t*) p_this;
181 intf_sys_t *sys = p_intf->p_sys;
182 [sys->displayer performSelectorOnMainThread:@selector(displayError:) withObject:DictFromDialogFatal(dialog) waitUntilDone:NO];
187 static int DisplayCritical(vlc_object_t *p_this, const char *type, vlc_value_t previous, vlc_value_t value, void *data)
189 NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
190 dialog_fatal_t *dialog = value.p_address;
191 intf_thread_t *p_intf = (intf_thread_t*) p_this;
192 intf_sys_t *sys = p_intf->p_sys;
193 [sys->displayer performSelectorOnMainThread:@selector(displayCritical:) withObject:DictFromDialogFatal(dialog) waitUntilDone:NO];
198 static int DisplayQuestion(vlc_object_t *p_this, const char *type, vlc_value_t previous, vlc_value_t value, void *data)
200 NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
201 dialog_question_t *dialog = value.p_address;
202 intf_thread_t *p_intf = (intf_thread_t*) p_this;
203 intf_sys_t *sys = p_intf->p_sys;
204 dialog->answer = [[sys->displayer resultFromSelectorOnMainThread:@selector(displayQuestion:) withObject:DictFromDialogQuestion(dialog)] intValue];
209 static int DisplayLogin(vlc_object_t *p_this, const char *type, vlc_value_t previous, vlc_value_t value, void *data)
211 NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
212 dialog_login_t *dialog = value.p_address;
213 intf_thread_t *p_intf = (intf_thread_t*) p_this;
214 intf_sys_t *sys = p_intf->p_sys;
215 NSDictionary *dict = [sys->displayer resultFromSelectorOnMainThread:@selector(displayCritical:) withObject:DictFromDialogLogin(dialog)];
217 *dialog->username = strdup([[dict objectForKey:@"username"] UTF8String]);
218 *dialog->password = strdup([[dict objectForKey:@"password"] UTF8String]);
224 static int DisplayProgressPanelAction(vlc_object_t *p_this, const char *type, vlc_value_t previous, vlc_value_t value, void *data)
226 NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
227 dialog_progress_bar_t *dialog = value.p_address;
228 intf_thread_t *p_intf = (intf_thread_t*) p_this;
229 intf_sys_t *sys = p_intf->p_sys;
231 [sys->displayer performSelectorOnMainThread:@selector(displayProgressBar) withObject:DictFromDialogProgressBar(dialog) waitUntilDone:YES];
233 dialog->pf_update = updateProgressPanel;
234 dialog->pf_check = checkProgressPanel;
235 dialog->pf_destroy = destroyProgressPanel;
236 dialog->p_sys = p_intf->p_sys;
242 void updateProgressPanel (void *priv, const char *text, float value)
244 NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
245 intf_sys_t *sys = (intf_sys_t *)priv;
247 NSDictionary *dict = [NSDictionary dictionaryWithObjectsAndKeys:
248 [NSNumber numberWithFloat:value], @"value",
249 text ? [NSString stringWithUTF8String:text] : nil, @"text",
252 [sys->displayer performSelectorOnMainThread:@selector(updateProgressPanel) withObject:dict waitUntilDone:YES];
257 void destroyProgressPanel (void *priv)
259 NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
260 intf_sys_t *sys = (intf_sys_t *)priv;
262 [sys->displayer performSelectorOnMainThread:@selector(destroyProgressPanel) withObject:nil waitUntilDone:YES];
267 bool checkProgressPanel (void *priv)
269 NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
270 intf_sys_t *sys = (intf_sys_t *)priv;
273 ret = [[sys->displayer resultFromSelectorOnMainThread:@selector(checkProgressPanel) withObject:nil] boolValue];
280 @implementation VLCDialogDisplayer
283 assert(!_currentProgressBarPanel); // This has to be closed on main thread.
287 + (NSDictionary *)dictionaryForDialog:(const char *)title :(const char *)message :(const char *)yes :(const char *)no :(const char *)cancel
289 NSMutableDictionary *dict = [NSMutableDictionary dictionary];
291 [dict setObject:[NSString stringWithUTF8String:title] forKey:@"title"];
293 [dict setObject:[NSString stringWithUTF8String:message] forKey:@"message"];
295 [dict setObject:[NSString stringWithUTF8String:yes] forKey:@"yes"];
297 [dict setObject:[NSString stringWithUTF8String:no] forKey:@"no"];
299 [dict setObject:[NSString stringWithUTF8String:cancel] forKey:@"cancel"];
303 #define VLCAssertIsMainThread() assert([NSThread isMainThread])
305 - (void)displayError:(NSDictionary *)dialog
307 VLCAssertIsMainThread();
309 NSRunInformationalAlertPanel([dialog objectForKey:@"title"],
310 [dialog objectForKey:@"message"],
314 - (void)displayCritical:(NSDictionary *)dialog
316 VLCAssertIsMainThread();
318 NSRunCriticalAlertPanel([dialog objectForKey:@"title"],
319 [dialog objectForKey:@"message"],
323 - (NSNumber *)displayQuestion:(NSDictionary *)dialog
325 VLCAssertIsMainThread();
327 NSInteger alertRet = 0;
329 NSAlert *alert = [NSAlert alertWithMessageText:[dialog objectForKey:@"title"]
330 defaultButton:[dialog objectForKey:@"yes"]
331 alternateButton:[dialog objectForKey:@"no"]
332 otherButton:[dialog objectForKey:@"cancel"]
333 informativeTextWithFormat:[dialog objectForKey:@"message"]];
334 [alert setAlertStyle:NSInformationalAlertStyle];
335 alertRet = [alert runModal];
339 case NSAlertDefaultReturn:
342 case NSAlertAlternateReturn:
345 case NSAlertOtherReturn:
354 return [NSNumber numberWithInt:ret];
357 - (NSDictionary *)displayLogin:(NSDictionary *)dialog
359 VLCAssertIsMainThread();
361 VLCLoginPanel *panel = [[VLCLoginPanel alloc] init];
362 [panel createContentView];
363 [panel setDialogTitle:[dialog objectForKey:@"title"]];
364 [panel setDialogMessage:[dialog objectForKey:@"message"]];
366 NSInteger ret = [NSApp runModalForWindow:panel];
372 return [NSDictionary dictionaryWithObjectsAndKeys:
373 [panel userName], @"username",
374 [panel password], @"password",
378 - (void)displayProgressBar:(NSDictionary *)dialog
380 VLCAssertIsMainThread();
382 if(_currentProgressBarPanel)
383 [self destroyProgressPanel];
385 assert(!_currentProgressBarPanel);
386 _currentProgressBarPanel = [[VLCProgressPanel alloc] init];
387 [_currentProgressBarPanel createContentView];
388 [_currentProgressBarPanel setDialogTitle:[dialog objectForKey:@"title"]];
389 [_currentProgressBarPanel setDialogMessage:[dialog objectForKey:@"message"]];
390 [_currentProgressBarPanel setCancelButtonLabel:[dialog objectForKey:@"cancel"]];
392 [_currentProgressBarPanel center];
393 [_currentProgressBarPanel makeKeyAndOrderFront:nil];
396 - (void)updateProgressPanel:(NSDictionary *)dict
398 VLCAssertIsMainThread();
400 assert(_currentProgressBarPanel);
401 [_currentProgressBarPanel setDialogMessage:[dict objectForKey:@"text"] ?: @""];
402 [_currentProgressBarPanel setProgressAsDouble:[[dict objectForKey:@"value"] doubleValue] * 1000.];
405 - (void)destroyProgressPanel
407 VLCAssertIsMainThread();
409 [_currentProgressBarPanel close];
410 [_currentProgressBarPanel release];
411 _currentProgressBarPanel = nil;
414 - (NSNumber *)checkProgressPanel
416 VLCAssertIsMainThread();
418 return [NSNumber numberWithBool:[_currentProgressBarPanel isCancelled]];
423 * Helper to execute a function on main thread and get its return value.
425 - (void)execute:(NSDictionary *)dict
427 SEL sel = [[dict objectForKey:@"sel"] pointerValue];
428 id * result = [[dict objectForKey:@"result"] pointerValue];
429 id object = [dict objectForKey:@"object"];
430 *result = [self performSelector:sel withObject:object];
431 [*result retain]; // Balanced in -resultFromSelectorOnMainThread
434 - (id)resultFromSelectorOnMainThread:(SEL)sel withObject:(id)object
437 [NSDictionary dictionaryWithObjectsAndKeys:
438 [NSValue valueWithPointer:sel], @"sel",
439 [NSValue valueWithPointer:&result], @"result",
440 object, @"object", nil];
441 [self performSelectorOnMainThread:@selector(execute:) withObject:object waitUntilDone:YES];
442 return [result autorelease];