]> git.sesse.net Git - vlc/blob - modules/gui/qt4/dialogs_provider.cpp
Qt: bugfix: playlist's add directory function was failing on input_Read (assert failed)
[vlc] / modules / gui / qt4 / dialogs_provider.cpp
1 /*****************************************************************************
2  * dialogs_provider.cpp : Dialog Provider
3  *****************************************************************************
4  * Copyright (C) 2006-2008 the VideoLAN team
5  * $Id$
6  *
7  * Authors: ClĂ©ment Stenac <zorglub@videolan.org>
8  *          Jean-Baptiste Kempf <jb@videolan.org>
9  *
10  * This program is free software; you can redistribute it and/or modify
11  * it under the terms of the GNU General Public License as published by
12  * the Free Software Foundation; either version 2 of the License, or
13  * (at your option) any later version.
14  *
15  * This program is distributed in the hope that it will be useful,
16  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18  * GNU General Public License for more details.
19  *
20  * You should have received a copy of the GNU General Public License
21  * along with this program; if not, write to the Free Software
22  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
23  *****************************************************************************/
24 #ifdef HAVE_CONFIG_H
25 # include "config.h"
26 #endif
27
28 #include <QEvent>
29 #include <QApplication>
30 #include <QSignalMapper>
31 #include <QFileDialog>
32
33 #include <vlc_common.h>
34 #include "qt4.hpp"
35 #include "dialogs_provider.hpp"
36 #include "main_interface.hpp"
37 #include "menus.hpp"
38 #include <vlc_intf_strings.h>
39 #include "input_manager.hpp"
40
41 /* The dialogs */
42 #include "dialogs/playlist.hpp"
43 #include "dialogs/bookmarks.hpp"
44 #include "dialogs/preferences.hpp"
45 #include "dialogs/mediainfo.hpp"
46 #include "dialogs/messages.hpp"
47 #include "dialogs/extended.hpp"
48 #include "dialogs/vlm.hpp"
49 #include "dialogs/sout.hpp"
50 #include "dialogs/open.hpp"
51 #include "dialogs/help.hpp"
52 #include "dialogs/gototime.hpp"
53 #include "dialogs/podcast_configuration.hpp"
54
55 DialogsProvider* DialogsProvider::instance = NULL;
56
57 DialogsProvider::DialogsProvider( intf_thread_t *_p_intf ) :
58                                   QObject( NULL ), p_intf( _p_intf )
59 {
60     b_isDying = false;
61     fixed_timer = new QTimer( this );
62     fixed_timer->start( 150 /* milliseconds */ );
63
64     menusMapper = new QSignalMapper();
65     CONNECT( menusMapper, mapped(QObject *), this, menuAction( QObject *) );
66
67     menusUpdateMapper = new QSignalMapper();
68     CONNECT( menusUpdateMapper, mapped(QObject *),
69              this, menuUpdateAction( QObject *) );
70
71     SDMapper = new QSignalMapper();
72     CONNECT( SDMapper, mapped (QString), this, SDMenuAction( QString ) );
73 }
74
75 DialogsProvider::~DialogsProvider()
76 {
77     msg_Dbg( p_intf, "Destroying the Dialog Provider" );
78     PlaylistDialog::killInstance();
79     MediaInfoDialog::killInstance();
80     MessagesDialog::killInstance();
81     ExtendedDialog::killInstance();
82     BookmarksDialog::killInstance();
83     HelpDialog::killInstance();
84 #ifdef UPDATE_CHECK
85     UpdateDialog::killInstance();
86 #endif
87
88     fixed_timer->stop();
89     delete menusMapper;
90     delete menusUpdateMapper;
91     delete SDMapper;
92 }
93
94 void DialogsProvider::quit()
95 {
96     /* Stop the playlist */
97     playlist_Stop( THEPL );
98     b_isDying = true;
99     vlc_object_kill( p_intf->p_libvlc );
100     QApplication::closeAllWindows();
101     QApplication::quit();
102 }
103
104 void DialogsProvider::customEvent( QEvent *event )
105 {
106     if( event->type() == DialogEvent_Type )
107     {
108         DialogEvent *de = static_cast<DialogEvent*>(event);
109         switch( de->i_dialog )
110         {
111         case INTF_DIALOG_FILE_SIMPLE:
112         case INTF_DIALOG_FILE:
113             openDialog(); break;
114         case INTF_DIALOG_FILE_GENERIC:
115             openFileGenericDialog( de->p_arg ); break;
116         case INTF_DIALOG_DISC:
117             openDiscDialog(); break;
118         case INTF_DIALOG_NET:
119             openNetDialog(); break;
120         case INTF_DIALOG_SAT:
121         case INTF_DIALOG_CAPTURE:
122             openCaptureDialog(); break;
123         case INTF_DIALOG_DIRECTORY:
124             PLAppendDir(); break;
125         case INTF_DIALOG_PLAYLIST:
126             playlistDialog(); break;
127         case INTF_DIALOG_MESSAGES:
128             messagesDialog(); break;
129         case INTF_DIALOG_FILEINFO:
130            mediaInfoDialog(); break;
131         case INTF_DIALOG_PREFS:
132            prefsDialog(); break;
133         case INTF_DIALOG_BOOKMARKS:
134            bookmarksDialog(); break;
135         case INTF_DIALOG_EXTENDED:
136            extendedDialog(); break;
137 #ifdef ENABLE_VLM
138         case INTF_DIALOG_VLM:
139            vlmDialog(); break;
140 #endif
141         case INTF_DIALOG_INTERACTION:
142            doInteraction( de->p_arg ); break;
143         case INTF_DIALOG_POPUPMENU:
144            QVLCMenu::PopupMenu( p_intf, (de->i_arg != 0) ); break;
145         case INTF_DIALOG_AUDIOPOPUPMENU:
146            QVLCMenu::AudioPopupMenu( p_intf ); break;
147         case INTF_DIALOG_VIDEOPOPUPMENU:
148            QVLCMenu::VideoPopupMenu( p_intf ); break;
149         case INTF_DIALOG_MISCPOPUPMENU:
150            QVLCMenu::MiscPopupMenu( p_intf ); break;
151         case INTF_DIALOG_WIZARD:
152         case INTF_DIALOG_STREAMWIZARD:
153             openThenStreamingDialogs(); break;
154 #ifdef UPDATE_CHECK
155         case INTF_DIALOG_UPDATEVLC:
156             updateDialog(); break;
157 #endif
158         case INTF_DIALOG_EXIT:
159             quit(); break;
160         default:
161            msg_Warn( p_intf, "unimplemented dialog" );
162         }
163     }
164 }
165
166 /****************************************************************************
167  * Individual simple dialogs
168  ****************************************************************************/
169 void DialogsProvider::playlistDialog()
170 {
171     PlaylistDialog::getInstance( p_intf )->toggleVisible();
172 }
173
174 void DialogsProvider::prefsDialog()
175 {
176     PrefsDialog::getInstance( p_intf )->toggleVisible();
177 }
178
179 void DialogsProvider::extendedDialog()
180 {
181     ExtendedDialog::getInstance( p_intf )->toggleVisible();
182 }
183
184 void DialogsProvider::messagesDialog()
185 {
186     MessagesDialog::getInstance( p_intf )->toggleVisible();
187 }
188
189 void DialogsProvider::gotoTimeDialog()
190 {
191     GotoTimeDialog::getInstance( p_intf )->toggleVisible();
192 }
193
194 #ifdef ENABLE_VLM
195 void DialogsProvider::vlmDialog()
196 {
197     VLMDialog::getInstance( p_intf )->toggleVisible();
198 }
199 #endif
200
201 void DialogsProvider::helpDialog()
202 {
203     HelpDialog::getInstance( p_intf )->toggleVisible();
204 }
205
206 #ifdef UPDATE_CHECK
207 void DialogsProvider::updateDialog()
208 {
209     UpdateDialog::getInstance( p_intf )->toggleVisible();
210 }
211 #endif
212
213 void DialogsProvider::aboutDialog()
214 {
215     AboutDialog::getInstance( p_intf )->toggleVisible();
216 }
217
218 void DialogsProvider::mediaInfoDialog()
219 {
220     MediaInfoDialog::getInstance( p_intf )->toggleVisible();
221 }
222
223 void DialogsProvider::mediaCodecDialog()
224 {
225     MediaInfoDialog::getInstance( p_intf )->showTab( 2 );
226 }
227
228 void DialogsProvider::bookmarksDialog()
229 {
230     BookmarksDialog::getInstance( p_intf )->toggleVisible();
231 }
232
233 void DialogsProvider::podcastConfigureDialog()
234 {
235     PodcastConfigDialog::getInstance( p_intf )->toggleVisible();
236 }
237
238
239 /****************************************************************************
240  * All the open/add stuff
241  * Open Dialog first - Simple Open then
242  ****************************************************************************/
243
244 void DialogsProvider::openDialog( int i_tab )
245 {
246     OpenDialog::getInstance( p_intf->p_sys->p_mi , p_intf )->showTab( i_tab );
247 }
248 void DialogsProvider::openDialog()
249 {
250     openDialog( OPEN_FILE_TAB );
251 }
252 void DialogsProvider::openFileGenericDialog( intf_dialog_args_t *p_arg )
253 {
254     if( p_arg == NULL )
255     {
256         msg_Warn( p_intf, "openFileGenericDialog() called with NULL arg" );
257         return;
258     }
259
260     /* Replace the extensions to a Qt format */
261     int i = 0;
262     QString extensions = qfu( p_arg->psz_extensions );
263     while ( ( i = extensions.indexOf( "|", i ) ) != -1 )
264     {
265         if( ( extensions.count( "|" ) % 2 ) == 0 )
266             extensions.replace( i, 1, ");;" );
267         else
268             extensions.replace( i, 1, "(" );
269     }
270     extensions.replace(QString(";*"), QString(" *"));
271     extensions.append( ")" );
272
273     /* Save */
274     if( p_arg->b_save )
275     {
276         QString file = QFileDialog::getSaveFileName( NULL, p_arg->psz_title,
277                             qfu( p_intf->p_sys->psz_filepath ), extensions );
278         if( !file.isEmpty() )
279         {
280             p_arg->i_results = 1;
281             p_arg->psz_results = (char **)malloc( p_arg->i_results * sizeof( char * ) );
282             p_arg->psz_results[0] = strdup( qtu( file ) );
283         }
284         else
285             p_arg->i_results = 0;
286     }
287     else /* non-save mode */
288     {
289         QStringList files = QFileDialog::getOpenFileNames( NULL,
290                 p_arg->psz_title, qfu( p_intf->p_sys->psz_filepath ),
291                 extensions );
292         p_arg->i_results = files.count();
293         p_arg->psz_results = (char **)malloc( p_arg->i_results * sizeof( char * ) );
294         i = 0;
295         foreach( QString file, files )
296             p_arg->psz_results[i++] = strdup( qtu( file ) );
297     }
298
299     /* Callback */
300     if( p_arg->pf_callback )
301         p_arg->pf_callback( p_arg );
302
303     /* Clean afterwards */
304     if( p_arg->psz_results )
305     {
306         for( i = 0; i < p_arg->i_results; i++ )
307             free( p_arg->psz_results[i] );
308         free( p_arg->psz_results );
309     }
310     free( p_arg->psz_title );
311     free( p_arg->psz_extensions );
312     free( p_arg );
313 }
314
315 void DialogsProvider::openFileDialog()
316 {
317     openDialog( OPEN_FILE_TAB );
318 }
319 void DialogsProvider::openDiscDialog()
320 {
321     openDialog( OPEN_DISC_TAB );
322 }
323 void DialogsProvider::openNetDialog()
324 {
325     openDialog( OPEN_NETWORK_TAB );
326 }
327 void DialogsProvider::openCaptureDialog()
328 {
329     openDialog( OPEN_CAPTURE_TAB );
330 }
331
332 /* Same as the open one, but force the enqueue */
333 void DialogsProvider::PLAppendDialog()
334 {
335     OpenDialog::getInstance( p_intf->p_sys->p_mi, p_intf, false, OPEN_AND_ENQUEUE)
336                             ->showTab( OPEN_FILE_TAB );
337 }
338
339 void DialogsProvider::MLAppendDialog()
340 {
341     OpenDialog::getInstance( p_intf->p_sys->p_mi, p_intf, false,
342                             OPEN_AND_ENQUEUE, false, false )
343                                     ->showTab( OPEN_FILE_TAB );
344 }
345
346 /**
347  * Simple open
348  ***/
349 QStringList DialogsProvider::showSimpleOpen( QString help,
350                                              int filters,
351                                              QString path )
352 {
353     QString fileTypes = "";
354     if( filters & EXT_FILTER_MEDIA ) {
355         ADD_FILTER_MEDIA( fileTypes );
356     }
357     if( filters & EXT_FILTER_VIDEO ) {
358         ADD_FILTER_VIDEO( fileTypes );
359     }
360     if( filters & EXT_FILTER_AUDIO ) {
361         ADD_FILTER_AUDIO( fileTypes );
362     }
363     if( filters & EXT_FILTER_PLAYLIST ) {
364         ADD_FILTER_PLAYLIST( fileTypes );
365     }
366     if( filters & EXT_FILTER_SUBTITLE ) {
367         ADD_FILTER_SUBTITLE( fileTypes );
368     }
369     ADD_FILTER_ALL( fileTypes );
370     fileTypes.replace(QString(";*"), QString(" *"));
371     return QFileDialog::getOpenFileNames( NULL,
372         help.isNull() ? qfu(I_OP_SEL_FILES ) : help,
373         path.isNull() ? qfu( p_intf->p_sys->psz_filepath ) : path,
374         fileTypes );
375 }
376
377 /**
378  * Open a file,
379  * pl helps you to choose from playlist or media library,
380  * go to start or enqueue
381  **/
382 void DialogsProvider::addFromSimple( bool pl, bool go)
383 {
384     QStringList files = DialogsProvider::showSimpleOpen();
385     int i = 0;
386     foreach( QString file, files )
387     {
388         const char * psz_utf8 = qtu( file );
389         playlist_Add( THEPL, psz_utf8, NULL,
390                       go ? ( PLAYLIST_APPEND | ( i ? 0 : PLAYLIST_GO ) |
391                                                ( i ? PLAYLIST_PREPARSE : 0 ) )
392                          : ( PLAYLIST_APPEND | PLAYLIST_PREPARSE ),
393                       PLAYLIST_END,
394                       pl ? true : false, false );
395         i++;
396     }
397 }
398
399 void DialogsProvider::simpleOpenDialog()
400 {
401     addFromSimple( true, true ); /* Playlist and Go */
402 }
403
404 void DialogsProvider::simplePLAppendDialog()
405 {
406     addFromSimple( true, false );
407 }
408
409 void DialogsProvider::simpleMLAppendDialog()
410 {
411     addFromSimple( false, false );
412 }
413
414 /* Directory */
415 /**
416  * Open a directory,
417  * pl helps you to choose from playlist or media library,
418  * go to start or enqueue
419  **/
420 static void openDirectory( intf_thread_t *p_intf, bool pl, bool go )
421 {
422     QString dir = QFileDialog::getExistingDirectory( 0, qtr("Open Directory") );
423     if (!dir.isEmpty()) {
424         input_item_t *p_input = input_item_NewExt( THEPL,
425                                         qtu( "directory://" + dir ), NULL,
426                                         0, NULL, -1 );
427
428         /* FIXME: playlist_AddInput() can fail */
429         playlist_AddInput( THEPL, p_input,
430                        go ? ( PLAYLIST_APPEND | PLAYLIST_GO ) : PLAYLIST_APPEND,
431                        PLAYLIST_END, pl, pl_Unlocked );
432         if( !go )
433             input_Read( THEPL, p_input, true );
434         vlc_gc_decref( p_input );
435     }
436 }
437
438 void DialogsProvider::PLOpenDir()
439 {
440     openDirectory( p_intf, true, true );
441 }
442
443 void DialogsProvider::PLAppendDir()
444 {
445     openDirectory( p_intf, true, false );
446 }
447
448 void DialogsProvider::MLAppendDir()
449 {
450     openDirectory( p_intf, false , false );
451 }
452
453 /****************
454  * Playlist     *
455  ****************/
456 void DialogsProvider::openAPlaylist()
457 {
458     QStringList files = showSimpleOpen( qtr( "Open playlist file" ),
459                                         EXT_FILTER_PLAYLIST );
460     foreach( QString file, files )
461     {
462         playlist_Import( THEPL, qtu(file) );
463     }
464 }
465
466 void DialogsProvider::saveAPlaylist()
467 {
468     QFileDialog *qfd = new QFileDialog( NULL,
469                                    qtr( "Choose a filename to save playlist" ),
470                                    qfu( p_intf->p_sys->psz_filepath ),
471                                    qtr( "XSPF playlist (*.xspf);; " ) +
472                                    qtr( "M3U playlist (*.m3u);; Any (*.*) " ) );
473     qfd->setFileMode( QFileDialog::AnyFile );
474     qfd->setAcceptMode( QFileDialog::AcceptSave );
475     qfd->setConfirmOverwrite( true );
476
477     if( qfd->exec() == QDialog::Accepted )
478     {
479         if( qfd->selectedFiles().count() > 0 )
480         {
481             static const char psz_xspf[] = "export-xspf",
482                               psz_m3u[] = "export-m3u";
483             const char *psz_module;
484
485             QString file = qfd->selectedFiles().first();
486             QString filter = qfd->selectedFilter();
487
488             if( file.contains( ".xsp" ) ||
489                 ( filter.contains( ".xspf" ) && !file.contains( ".m3u" ) ) )
490             {
491                 psz_module = psz_xspf;
492                 if( !file.contains( ".xsp" ) )
493                     file.append( ".xspf" );
494             }
495             else
496             {
497                 psz_module = psz_m3u;
498                 if( !file.contains( ".m3u" ) )
499                     file.append( ".m3u" );
500             }
501
502             playlist_Export( THEPL, qtu( file ), THEPL->p_local_category,
503                              psz_module);
504         }
505     }
506     delete qfd;
507 }
508
509
510 /****************************************************************************
511  * Sout emulation
512  ****************************************************************************/
513
514 void DialogsProvider::streamingDialog( QWidget *parent, QString mrl,
515                                        bool b_transcode_only )
516 {
517     SoutDialog *s = SoutDialog::getInstance( parent, p_intf, b_transcode_only );
518
519     if( s->exec() == QDialog::Accepted )
520     {
521         msg_Dbg( p_intf, "Sout mrl %s", qta( s->getMrl() ) );
522         /* Just do it */
523         int i_len = strlen( qtu( s->getMrl() ) ) + 10;
524         char *psz_option = (char*)malloc( i_len );
525         snprintf( psz_option, i_len - 1, "%s", qtu( s->getMrl() ) );
526
527         playlist_AddExt( THEPL, qtu( mrl ), "Streaming",
528                          PLAYLIST_APPEND | PLAYLIST_GO, PLAYLIST_END,
529                         -1, &psz_option, 1, true, pl_Unlocked );
530     }
531 }
532
533 void DialogsProvider::openThenStreamingDialogs()
534 {
535     OpenDialog::getInstance( p_intf->p_sys->p_mi, p_intf, false, OPEN_AND_STREAM )
536                                 ->showTab( OPEN_FILE_TAB );
537 }
538
539 void DialogsProvider::openThenTranscodingDialogs()
540 {
541     OpenDialog::getInstance( p_intf->p_sys->p_mi , p_intf, false, OPEN_AND_SAVE )
542                                 ->showTab( OPEN_FILE_TAB );
543 }
544
545 /****************************************************************************
546  * Menus / Interaction
547  ****************************************************************************/
548
549 void DialogsProvider::menuAction( QObject *data )
550 {
551     QVLCMenu::DoAction( p_intf, data );
552 }
553
554 void DialogsProvider::menuUpdateAction( QObject *data )
555 {
556     MenuFunc * f = qobject_cast<MenuFunc *>(data);
557     f->doFunc( p_intf );
558 }
559
560 void DialogsProvider::SDMenuAction( QString data )
561 {
562     char *psz_sd = strdup( qtu( data ) );
563     if( !playlist_IsServicesDiscoveryLoaded( THEPL, psz_sd ) )
564         playlist_ServicesDiscoveryAdd( THEPL, psz_sd );
565     else
566         playlist_ServicesDiscoveryRemove( THEPL, psz_sd );
567     free( psz_sd );
568 }
569
570 void DialogsProvider::doInteraction( intf_dialog_args_t *p_arg )
571 {
572     InteractionDialog *qdialog;
573     interaction_dialog_t *p_dialog = p_arg->p_dialog;
574     switch( p_dialog->i_action )
575     {
576     case INTERACT_NEW:
577         qdialog = new InteractionDialog( p_intf, p_dialog );
578         p_dialog->p_private = (void*)qdialog;
579         if( !(p_dialog->i_status == ANSWERED_DIALOG) )
580             qdialog->show();
581         break;
582     case INTERACT_UPDATE:
583         qdialog = (InteractionDialog*)(p_dialog->p_private);
584         if( qdialog )
585             qdialog->update();
586         else
587         {
588             /* The INTERACT_NEW message was forgotten
589                so we must create the dialog and update it*/
590             qdialog = new InteractionDialog( p_intf, p_dialog );
591             p_dialog->p_private = (void*)qdialog;
592             if( !(p_dialog->i_status == ANSWERED_DIALOG) )
593                 qdialog->show();
594             if( qdialog )
595                 qdialog->update();
596         }
597         break;
598     case INTERACT_HIDE:
599         msg_Dbg( p_intf, "Hide the Interaction Dialog" );
600         qdialog = (InteractionDialog*)(p_dialog->p_private);
601         if( qdialog )
602             qdialog->hide();
603         p_dialog->i_status = HIDDEN_DIALOG;
604         break;
605     case INTERACT_DESTROY:
606         msg_Dbg( p_intf, "Destroy the Interaction Dialog" );
607         qdialog = (InteractionDialog*)(p_dialog->p_private);
608         if( !p_dialog->i_flags & DIALOG_NONBLOCKING_ERROR )
609             delete qdialog;
610         p_dialog->i_status = DESTROYED_DIALOG;
611         break;
612     }
613 }
614
615 void DialogsProvider::loadSubtitlesFile()
616 {
617     input_thread_t *p_input = THEMIM->getInput();
618     if( !p_input )
619         return;
620     input_item_t *p_item = input_GetItem( p_input );
621     if( !p_item )
622         return;
623     char *path = input_item_GetURI( p_item );
624     if( !path )
625         path = strdup( "" );
626     char *sep = strrchr( path, DIR_SEP_CHAR );
627     if( sep )
628         *sep = '\0';
629     QStringList qsl = showSimpleOpen( qtr( "Open subtitles file" ),
630                                       EXT_FILTER_SUBTITLE,
631                                       path );
632     free( path );
633     QString qsFile;
634     foreach( qsFile, qsl )
635     {
636         if( !input_AddSubtitles( p_input, qtu( qsFile ), true ) )
637             msg_Warn( p_intf, "unable to load subtitles from '%s'",
638                       qtu( qsFile ) );
639     }
640 }