]> git.sesse.net Git - vlc/blob - modules/gui/qt4/dialogs/open.cpp
Fix potential segfault.
[vlc] / modules / gui / qt4 / dialogs / open.cpp
1 /*****************************************************************************
2  * open.cpp : Advanced open dialog
3  *****************************************************************************
4  * Copyright © 2006-2007 the VideoLAN team
5  * $Id$
6  *
7  * Authors: Jean-Baptiste Kempf <jb@videolan.org>
8  *
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.
13  *
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.
18  *
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  *****************************************************************************/
23 #ifdef HAVE_CONFIG_H
24 # include "config.h"
25 #endif
26
27 #include "input_manager.hpp"
28
29 #include "dialogs/open.hpp"
30
31 #include <QTabWidget>
32 #include <QGridLayout>
33 #include <QFileDialog>
34 #include <QRegExp>
35 #include <QMenu>
36
37 OpenDialog *OpenDialog::instance = NULL;
38
39 OpenDialog* OpenDialog::getInstance( QWidget *parent, intf_thread_t *p_intf,
40         bool b_rawInstance, int _action_flag, bool b_selectMode )
41 {
42     /* Creation */
43     if( !instance )
44         instance = new OpenDialog( parent, p_intf, b_selectMode, _action_flag );
45     else if( !b_rawInstance )
46     {
47         /* Request the instance but change small details:
48            - Button menu
49            - Modality on top of the parent dialog */
50         if( b_selectMode )
51         {
52             instance->setWindowModality( Qt::WindowModal );
53             _action_flag = SELECT; /* This should be useless, but we never know
54                                       if the call is correct */
55         }
56         instance->i_action_flag = _action_flag;
57         instance->setMenuAction();
58     }
59     return instance;
60 }
61
62 OpenDialog::OpenDialog( QWidget *parent,
63                         intf_thread_t *_p_intf,
64                         bool b_selectMode,
65                         int _action_flag )  :  QVLCDialog( parent, _p_intf )
66 {
67     i_action_flag = _action_flag;
68
69     if( b_selectMode ) /* Select mode */
70     {
71         setWindowModality( Qt::WindowModal );
72         i_action_flag = SELECT;
73     }
74
75     /* Basic Creation of the Window */
76     ui.setupUi( this );
77     setWindowTitle( qtr( "Open" ) );
78     /* resize( 410, 600 ); */
79     setMinimumSize( 520, 460 );
80
81     /* Tab definition and creation */
82     fileOpenPanel    = new FileOpenPanel( ui.Tab, p_intf );
83     discOpenPanel    = new DiscOpenPanel( ui.Tab, p_intf );
84     netOpenPanel     = new NetOpenPanel( ui.Tab, p_intf );
85     captureOpenPanel = new CaptureOpenPanel( ui.Tab, p_intf );
86
87     /* Insert the tabs */
88     ui.Tab->insertTab( OPEN_FILE_TAB, fileOpenPanel, qtr( "&File" ) );
89     ui.Tab->insertTab( OPEN_DISC_TAB, discOpenPanel, qtr( "&Disc" ) );
90     ui.Tab->insertTab( OPEN_NETWORK_TAB, netOpenPanel, qtr( "&Network" ) );
91     ui.Tab->insertTab( OPEN_CAPTURE_TAB, captureOpenPanel,
92                        qtr( "Capture &Device" ) );
93
94     /* Hide the Slave input widgets */
95     ui.slaveLabel->hide();
96     ui.slaveText->hide();
97     ui.slaveBrowseButton->hide();
98
99     /* Buttons Creation */
100     /* Play Button */
101     playButton = ui.playButton;
102
103     /* Cancel Button */
104     cancelButton = new QPushButton( qtr( "&Cancel" ) );
105
106     /* Select Button */
107     selectButton = new QPushButton( qtr( "&Select" ) );
108
109     /* Menu for the Play button */
110     QMenu * openButtonMenu = new QMenu( "Open" );
111     openButtonMenu->addAction( qtr( "&Enqueue" ), this, SLOT( enqueue() ),
112                                     QKeySequence( "Alt+E" ) );
113     openButtonMenu->addAction( qtr( "&Play" ), this, SLOT( play() ),
114                                     QKeySequence( "Alt+P" ) );
115     openButtonMenu->addAction( qtr( "&Stream" ), this, SLOT( stream() ) ,
116                                     QKeySequence( "Alt+S" ) );
117     openButtonMenu->addAction( qtr( "&Convert" ), this, SLOT( transcode() ) ,
118                                     QKeySequence( "Alt+C" ) );
119
120     ui.menuButton->setMenu( openButtonMenu );
121
122     /* Add the three Buttons */
123     ui.buttonsBox->addButton( selectButton, QDialogButtonBox::AcceptRole );
124     ui.buttonsBox->addButton( cancelButton, QDialogButtonBox::RejectRole );
125
126     /* At creation time, modify the default buttons */
127     setMenuAction();
128
129     /* Force MRL update on tab change */
130     CONNECT( ui.Tab, currentChanged( int ), this, signalCurrent() );
131
132     CONNECT( fileOpenPanel, mrlUpdated( QString ), this, updateMRL( QString ) );
133     CONNECT( netOpenPanel, mrlUpdated( QString ), this, updateMRL( QString ) );
134     CONNECT( discOpenPanel, mrlUpdated( QString ), this, updateMRL( QString ) );
135     CONNECT( captureOpenPanel, mrlUpdated( QString ), this, updateMRL( QString ) );
136
137     CONNECT( fileOpenPanel, methodChanged( QString ),
138              this, newCachingMethod( QString ) );
139     CONNECT( netOpenPanel, methodChanged( QString ),
140              this, newCachingMethod( QString ) );
141     CONNECT( discOpenPanel, methodChanged( QString ),
142              this, newCachingMethod( QString ) );
143     CONNECT( captureOpenPanel, methodChanged( QString ),
144              this, newCachingMethod( QString ) );
145
146     /* Advanced frame Connects */
147     CONNECT( ui.slaveText, textChanged( QString ), this, updateMRL() );
148     CONNECT( ui.cacheSpinBox, valueChanged( int ), this, updateMRL() );
149     CONNECT( ui.startTimeSpinBox, valueChanged( int ), this, updateMRL() );
150     BUTTONACT( ui.advancedCheckBox, toggleAdvancedPanel() );
151     BUTTONACT( ui.slaveBrowseButton, browseInputSlave() );
152
153     /* Buttons action */
154     BUTTONACT( playButton, selectSlots() );
155     BUTTONACT( selectButton, close() );
156     BUTTONACT( cancelButton, cancel() );
157
158     /* Hide the advancedPanel */
159     if( !config_GetInt( p_intf, "qt-adv-options" ) )
160         ui.advancedFrame->hide();
161     else
162         ui.advancedCheckBox->setChecked( true );
163
164     /* Initialize caching */
165     storedMethod = "";
166     newCachingMethod( "file-caching" );
167 }
168
169 OpenDialog::~OpenDialog()
170 {}
171
172 /* Finish the dialog and decide if you open another one after */
173 void OpenDialog::setMenuAction()
174 {
175     if( i_action_flag == SELECT )
176     {
177         playButton->hide();
178         selectButton->show();
179         selectButton->setDefault( true );
180     }
181     else
182     {
183         switch ( i_action_flag )
184         {
185         case OPEN_AND_STREAM:
186             playButton->setText( qtr( "&Stream" ) );
187             break;
188         case OPEN_AND_SAVE:
189             playButton->setText( qtr( "&Convert / Save" ) );
190             break;
191         case OPEN_AND_ENQUEUE:
192             playButton->setText( qtr( "&Enqueue" ) );
193             break;
194         case OPEN_AND_PLAY:
195         default:
196             playButton->setText( qtr( "&Play" ) );
197         }
198         playButton->show();
199         selectButton->hide();
200         playButton->setDefault( true );
201     }
202 }
203
204 void OpenDialog::showTab( int i_tab )
205 {
206     ui.Tab->setCurrentIndex( i_tab );
207     show();
208 }
209
210 /* Function called on signal currentChanged triggered */
211 void OpenDialog::signalCurrent()
212 {
213     if( ui.Tab->currentWidget() != NULL )
214         ( dynamic_cast<OpenPanel *>( ui.Tab->currentWidget() ) )->updateMRL();
215 }
216
217 void OpenDialog::toggleAdvancedPanel()
218 {
219     if( ui.advancedFrame->isVisible() )
220     {
221         ui.advancedFrame->hide();
222         //setMinimumSize( 520, 460 );
223         if( size().isValid() )
224             resize( size().width(), size().height()
225                     - ui.advancedFrame->height() );
226     }
227     else
228     {
229         ui.advancedFrame->show();
230         //setMinimumSize( 520, 460 + ui.advancedFrame->height() );
231         if( size().isValid() )
232             resize( size().width(), size().height()
233                     + ui.advancedFrame->height() );
234     }
235 }
236
237 /***********
238  * Actions *
239  ***********/
240 /* If Cancel is pressed or escaped */
241 void OpenDialog::cancel()
242 {
243     /* Clear the panels */
244     for( int i = 0; i < OPEN_TAB_MAX; i++ )
245         dynamic_cast<OpenPanel*>( ui.Tab->widget( i ) )->clear();
246
247     /* Clear the variables */
248     mrl.clear();
249     mainMRL.clear();
250
251     /* If in Select Mode, reject instead of hiding */
252     if( i_action_flag == SELECT ) reject();
253     else hide();
254 }
255
256 /* If EnterKey is pressed */
257 void OpenDialog::close()
258 {
259     /* If in Select Mode, accept instead of selecting a Slot */
260     if( i_action_flag == SELECT )
261         accept();
262     else
263         selectSlots();
264 }
265
266 /* Play button */
267 void OpenDialog::selectSlots()
268 {
269     switch ( i_action_flag )
270     {
271     case OPEN_AND_STREAM:
272         stream();
273         break;
274     case OPEN_AND_SAVE:
275         transcode();
276         break;
277     case OPEN_AND_ENQUEUE:
278         enqueue();
279         break;
280     case OPEN_AND_PLAY:
281     default:
282         play();
283     }
284 }
285
286 void OpenDialog::play()
287 {
288     finish( false );
289 }
290
291 void OpenDialog::enqueue()
292 {
293     finish( true );
294 }
295
296
297 void OpenDialog::finish( bool b_enqueue = false )
298 {
299     toggleVisible();
300     mrl = ui.advancedLineInput->text();
301
302     if( i_action_flag != SELECT )
303     {
304         QStringList tempMRL = SeparateEntries( mrl );
305         for( size_t i = 0; i < tempMRL.size(); i++ )
306         {
307             bool b_start = !i && !b_enqueue;
308             input_item_t *p_input;
309
310             p_input = input_ItemNew( p_intf, qtu( tempMRL[i] ), NULL );
311
312             /* Insert options */
313             while( i + 1 < tempMRL.size() && tempMRL[i + 1].startsWith( ":" ) )
314             {
315                 i++;
316                 input_ItemAddOption( p_input, qtu( tempMRL[i] ) );
317             }
318
319             /* Switch between enqueuing and starting the item */
320             /* FIXME: playlist_AddInput() can fail */
321             playlist_AddInput( THEPL, p_input,
322                 PLAYLIST_APPEND | ( b_start ? PLAYLIST_GO : PLAYLIST_PREPARSE ),
323                 PLAYLIST_END, true, pl_Unlocked );
324             vlc_gc_decref( p_input );
325         }
326     }
327     else
328         accept();
329 }
330
331 void OpenDialog::transcode()
332 {
333     stream( true );
334 }
335
336 void OpenDialog::stream( bool b_transcode_only )
337 {
338     mrl = ui.advancedLineInput->text();
339     toggleVisible();
340     QStringList listMRL = SeparateEntries( mrl );
341     if( listMRL.size() > 0 )
342     THEDP->streamingDialog( this, SeparateEntries( mrl )[0], b_transcode_only );
343 }
344
345 /* Update the MRL */
346 void OpenDialog::updateMRL( QString tempMRL )
347 {
348     mainMRL = tempMRL;
349     updateMRL();
350 }
351
352 void OpenDialog::updateMRL() {
353     mrl = mainMRL;
354     if( ui.slaveCheckbox->isChecked() ) {
355         mrl += " :input-slave=" + ui.slaveText->text();
356     }
357     int i_cache = config_GetInt( p_intf, qta( storedMethod ) );
358     if( i_cache != ui.cacheSpinBox->value() ) {
359         mrl += QString( " :%1=%2" ).arg( storedMethod ).
360                                   arg( ui.cacheSpinBox->value() );
361     }
362     if( ui.startTimeSpinBox->value() ) {
363         mrl += " :start-time=" + QString( "%1" ).
364             arg( ui.startTimeSpinBox->value() );
365     }
366     ui.advancedLineInput->setText( mrl );
367 }
368
369 void OpenDialog::newCachingMethod( QString method )
370 {
371     if( method != storedMethod ) {
372         storedMethod = method;
373         int i_value = config_GetInt( p_intf, qta( storedMethod ) );
374         ui.cacheSpinBox->setValue( i_value );
375     }
376 }
377
378 QStringList OpenDialog::SeparateEntries( QString entries )
379 {
380     bool b_quotes_mode = false;
381
382     QStringList entries_array;
383     QString entry;
384
385     int index = 0;
386     while( index < entries.size() )
387     {
388         int delim_pos = entries.indexOf( QRegExp( "\\s+|\"" ), index );
389         if( delim_pos < 0 ) delim_pos = entries.size() - 1;
390         entry += entries.mid( index, delim_pos - index + 1 );
391         index = delim_pos + 1;
392
393         if( entry.isEmpty() ) continue;
394
395         if( !b_quotes_mode && entry.endsWith( "\"" ) )
396         {
397             /* Enters quotes mode */
398             entry.truncate( entry.size() - 1 );
399             b_quotes_mode = true;
400         }
401         else if( b_quotes_mode && entry.endsWith( "\"" ) )
402         {
403             /* Finished the quotes mode */
404             entry.truncate( entry.size() - 1 );
405             b_quotes_mode = false;
406         }
407         else if( !b_quotes_mode && !entry.endsWith( "\"" ) )
408         {
409             /* we found a non-quoted standalone string */
410             if( index < entries.size() ||
411                 entry.endsWith( " " ) || entry.endsWith( "\t" ) ||
412                 entry.endsWith( "\r" ) || entry.endsWith( "\n" ) )
413                 entry.truncate( entry.size() - 1 );
414             if( !entry.isEmpty() ) entries_array.append( entry );
415             entry.clear();
416         }
417         else
418         {;}
419     }
420
421     if( !entry.isEmpty() ) entries_array.append( entry );
422
423     return entries_array;
424 }
425
426 void OpenDialog::browseInputSlave()
427 {
428     OpenDialog *od = new OpenDialog( this, p_intf, true, SELECT );
429     od->exec();
430     ui.slaveText->setText( od->getMRL() );
431     delete od;
432 }