/*****************************************************************************
* open.cpp : Advanced open dialog
*****************************************************************************
- * Copyright (C) 2006-2007 the VideoLAN team
- * $Id$
+ * Copyright © 2006-2011 the VideoLAN team
*
* Authors: Jean-Baptiste Kempf <jb@videolan.org>
*
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
*****************************************************************************/
-#include "input_manager.hpp"
+#ifdef HAVE_CONFIG_H
+# include "config.h"
+#endif
#include "dialogs/open.hpp"
+#include "dialogs_provider.hpp"
+#include "recents.hpp"
+#include "util/qt_dirs.hpp"
#include <QTabWidget>
-#include <QGridLayout>
-#include <QFileDialog>
#include <QRegExp>
#include <QMenu>
+#ifndef NDEBUG
+# define DEBUG_QT 1
+#endif
+
OpenDialog *OpenDialog::instance = NULL;
-OpenDialog::OpenDialog( QWidget *parent, intf_thread_t *_p_intf, bool modal,
- int _action_flag ) : QVLCDialog( parent, _p_intf )
+OpenDialog* OpenDialog::getInstance( QWidget *parent, intf_thread_t *p_intf,
+ bool b_rawInstance, int _action_flag, bool b_selectMode, bool _b_pl )
+{
+ /* Creation */
+ if( !instance )
+ instance = new OpenDialog( parent, p_intf, b_selectMode,
+ _action_flag, _b_pl );
+ else if( !b_rawInstance )
+ {
+ /* Request the instance but change small details:
+ - Button menu */
+ if( b_selectMode )
+ _action_flag = SELECT; /* This should be useless, but we never know
+ if the call is correct */
+ instance->setWindowModality( Qt::WindowModal );
+ instance->i_action_flag = _action_flag;
+ instance->b_pl = _b_pl;
+ instance->setMenuAction();
+ }
+ return instance;
+}
+
+OpenDialog::OpenDialog( QWidget *parent,
+ intf_thread_t *_p_intf,
+ bool b_selectMode,
+ int _action_flag,
+ bool _b_pl) : QVLCDialog( parent, _p_intf )
{
- setModal( modal );
i_action_flag = _action_flag;
+ b_pl =_b_pl;
+
+ if( b_selectMode ) /* Select mode */
+ i_action_flag = SELECT;
/* Basic Creation of the Window */
ui.setupUi( this );
- setWindowTitle( qtr("Open" ) );
- resize( 410, 300);
+ setWindowTitle( qtr( "Open Media" ) );
+ setWindowRole( "vlc-open-media" );
+ setWindowModality( Qt::WindowModal );
/* Tab definition and creation */
- fileOpenPanel = new FileOpenPanel( ui.Tab, p_intf );
- discOpenPanel = new DiscOpenPanel( ui.Tab, p_intf );
- netOpenPanel = new NetOpenPanel( ui.Tab, p_intf );
- captureOpenPanel = new CaptureOpenPanel( ui.Tab, p_intf );
+ fileOpenPanel = new FileOpenPanel( this, p_intf );
+ discOpenPanel = new DiscOpenPanel( this, p_intf );
+ netOpenPanel = new NetOpenPanel( this, p_intf );
+ captureOpenPanel = new CaptureOpenPanel( this, p_intf );
/* Insert the tabs */
- ui.Tab->insertTab( OPEN_FILE_TAB, fileOpenPanel, qtr( "&File" ) );
- ui.Tab->insertTab( OPEN_DISC_TAB, discOpenPanel, qtr( "&Disc" ) );
- ui.Tab->insertTab( OPEN_NETWORK_TAB, netOpenPanel, qtr( "&Network" ) );
+ ui.Tab->insertTab( OPEN_FILE_TAB, fileOpenPanel, QIcon( ":/type/folder-grey" ),
+ qtr( "&File" ) );
+ ui.Tab->insertTab( OPEN_DISC_TAB, discOpenPanel, QIcon( ":/type/disc" ),
+ qtr( "&Disc" ) );
+ ui.Tab->insertTab( OPEN_NETWORK_TAB, netOpenPanel, QIcon( ":/type/network" ),
+ qtr( "&Network" ) );
ui.Tab->insertTab( OPEN_CAPTURE_TAB, captureOpenPanel,
- qtr( "Capture &Device" ) );
+ QIcon( ":/type/capture-card" ), qtr( "Capture &Device" ) );
/* Hide the Slave input widgets */
ui.slaveLabel->hide();
ui.slaveText->hide();
ui.slaveBrowseButton->hide();
- /* Hide the advancedPanel */
- if(! config_GetInt( p_intf, "qt-adv-options") )
- ui.advancedFrame->hide();
- else
- ui.advancedCheckBox->setCheckState( Qt::Checked );
-
/* Buttons Creation */
- QSizePolicy buttonSizePolicy( QSizePolicy::Expanding, QSizePolicy::Minimum );
- buttonSizePolicy.setHorizontalStretch( 0 );
- buttonSizePolicy.setVerticalStretch( 0 );
-
/* Play Button */
- playButton = new QToolButton( this );
- playButton->setText( qtr( "&Play" ) );
- playButton->setSizePolicy( buttonSizePolicy );
- playButton->setMinimumSize( QSize(90, 0) );
- playButton->setPopupMode( QToolButton::MenuButtonPopup );
- playButton->setToolButtonStyle( Qt::ToolButtonTextOnly );
+ playButton = ui.playButton;
/* Cancel Button */
- cancelButton = new QPushButton();
- cancelButton->setText( qtr( "&Cancel" ) );
- cancelButton->setSizePolicy( buttonSizePolicy );
+ cancelButton = new QPushButton( qtr( "&Cancel" ) );
+
+ /* Select Button */
+ selectButton = new QPushButton( qtr( "&Select" ) );
/* Menu for the Play button */
QMenu * openButtonMenu = new QMenu( "Open" );
openButtonMenu->addAction( qtr( "&Enqueue" ), this, SLOT( enqueue() ),
- QKeySequence( "Alt+E") );
+ QKeySequence( "Alt+E" ) );
openButtonMenu->addAction( qtr( "&Play" ), this, SLOT( play() ),
QKeySequence( "Alt+P" ) );
openButtonMenu->addAction( qtr( "&Stream" ), this, SLOT( stream() ) ,
QKeySequence( "Alt+S" ) );
- openButtonMenu->addAction( qtr( "&Convert" ), this, SLOT( transcode() ) ,
- QKeySequence( "Alt+C" ) );
+ openButtonMenu->addAction( qtr( "C&onvert" ), this, SLOT( transcode() ) ,
+ QKeySequence( "Alt+O" ) );
playButton->setMenu( openButtonMenu );
- ui.buttonsBox->addButton( playButton, QDialogButtonBox::AcceptRole );
+ /* Add the three Buttons */
+ ui.buttonsBox->addButton( selectButton, QDialogButtonBox::AcceptRole );
ui.buttonsBox->addButton( cancelButton, QDialogButtonBox::RejectRole );
+ /* At creation time, modify the default buttons */
+ setMenuAction();
+
/* Force MRL update on tab change */
- CONNECT( ui.Tab, currentChanged(int), this, signalCurrent() );
-
- CONNECT( fileOpenPanel, mrlUpdated( QString ), this, updateMRL(QString) );
- CONNECT( netOpenPanel, mrlUpdated( QString ), this, updateMRL(QString) );
- CONNECT( discOpenPanel, mrlUpdated( QString ), this, updateMRL(QString) );
- CONNECT( captureOpenPanel, mrlUpdated( QString ), this, updateMRL(QString) );
-
- CONNECT( fileOpenPanel, methodChanged( QString ),
- this, newCachingMethod(QString) );
- CONNECT( netOpenPanel, methodChanged( QString ),
- this, newCachingMethod(QString) );
- CONNECT( discOpenPanel, methodChanged( QString ),
- this, newCachingMethod(QString) );
- CONNECT( captureOpenPanel, methodChanged( QString ),
- this, newCachingMethod(QString) );
+ CONNECT( ui.Tab, currentChanged( int ), this, signalCurrent( int ) );
+
+ CONNECT( fileOpenPanel, mrlUpdated( const QStringList&, const QString& ),
+ this, updateMRL( const QStringList&, const QString& ) );
+ CONNECT( netOpenPanel, mrlUpdated( const QStringList&, const QString& ),
+ this, updateMRL( const QStringList&, const QString& ) );
+ CONNECT( discOpenPanel, mrlUpdated( const QStringList&, const QString& ),
+ this, updateMRL( const QStringList&, const QString& ) );
+ CONNECT( captureOpenPanel, mrlUpdated( const QStringList&, const QString& ),
+ this, updateMRL( const QStringList&, const QString& ) );
+
+ CONNECT( fileOpenPanel, methodChanged( const QString& ),
+ this, newCachingMethod( const QString& ) );
+ CONNECT( netOpenPanel, methodChanged( const QString& ),
+ this, newCachingMethod( const QString& ) );
+ CONNECT( discOpenPanel, methodChanged( const QString& ),
+ this, newCachingMethod( const QString& ) );
+ CONNECT( captureOpenPanel, methodChanged( const QString& ),
+ this, newCachingMethod( const QString& ) );
/* Advanced frame Connects */
- CONNECT( ui.slaveText, textChanged(QString), this, updateMRL() );
- CONNECT( ui.cacheSpinBox, valueChanged(int), this, updateMRL() );
- CONNECT( ui.startTimeSpinBox, valueChanged(int), this, updateMRL() );
- BUTTONACT( ui.advancedCheckBox , toggleAdvancedPanel() );
+ CONNECT( ui.slaveCheckbox, toggled( bool ), this, updateMRL() );
+ CONNECT( ui.slaveText, textChanged( const QString& ), this, updateMRL() );
+ CONNECT( ui.cacheSpinBox, valueChanged( int ), this, updateMRL() );
+ CONNECT( ui.startTimeTimeEdit, timeChanged ( const QTime& ), this, updateMRL() );
+ BUTTONACT( ui.advancedCheckBox, toggleAdvancedPanel() );
+ BUTTONACT( ui.slaveBrowseButton, browseInputSlave() );
/* Buttons action */
BUTTONACT( playButton, selectSlots() );
+ BUTTONACT( selectButton, close() );
BUTTONACT( cancelButton, cancel() );
- /* At creation time, modify the default buttons */
- if ( i_action_flag ) setMenuAction();
+ /* Hide the advancedPanel */
+ if( !getSettings()->value( "OpenDialog/advanced", false ).toBool())
+ {
+ ui.advancedFrame->hide();
+ ui.advancedFrame->setEnabled( false );
+ }
+ else
+ ui.advancedCheckBox->setChecked( true );
/* Initialize caching */
storedMethod = "";
newCachingMethod( "file-caching" );
- mainHeight = advHeight = 0;
-}
+ /* enforce section due to .ui bug */
+ ui.startTimeTimeEdit->setCurrentSection( QDateTimeEdit::SecondSection );
-OpenDialog::~OpenDialog()
-{
+ setMinimumSize( sizeHint() );
+ setMaximumWidth( 900 );
+ resize( getSettings()->value( "OpenDialog/size", QSize( 500, 400 ) ).toSize() );
}
/* Finish the dialog and decide if you open another one after */
void OpenDialog::setMenuAction()
{
- switch ( i_action_flag )
+ if( i_action_flag == SELECT )
{
- case OPEN_AND_STREAM:
- playButton->setText( qtr( "&Stream" ) );
- break;
- case OPEN_AND_SAVE:
- playButton->setText( qtr( "&Convert / Save" ) );
- break;
- case OPEN_AND_ENQUEUE:
- playButton->setText( qtr( "&Enqueue" ) );
- break;
- case OPEN_AND_PLAY:
- default:
- playButton->setText( qtr( "&Play" ) );
- }
+ playButton->hide();
+ selectButton->show();
+ selectButton->setDefault( true );
+ }
+ else
+ {
+ switch ( i_action_flag )
+ {
+ case OPEN_AND_STREAM:
+ playButton->setText( qtr( "&Stream" ) );
+ break;
+ case OPEN_AND_SAVE:
+ playButton->setText( qtr( "C&onvert / Save" ) );
+ break;
+ case OPEN_AND_ENQUEUE:
+ playButton->setText( qtr( "&Enqueue" ) );
+ break;
+ case OPEN_AND_PLAY:
+ default:
+ playButton->setText( qtr( "&Play" ) );
+ }
+ playButton->show();
+ selectButton->hide();
+ }
}
-void OpenDialog::showTab( int i_tab=0 )
+OpenDialog::~OpenDialog()
{
- this->show();
- ui.Tab->setCurrentIndex( i_tab );
+ getSettings()->setValue( "OpenDialog/size", size() -
+ ( ui.advancedFrame->isEnabled() ?
+ QSize(0, ui.advancedFrame->height()) : QSize(0, 0) ) );
+ getSettings()->setValue( "OpenDialog/advanced", ui.advancedFrame->isVisible() );
}
-void OpenDialog::signalCurrent() {
- if (ui.Tab->currentWidget() != NULL)
- (dynamic_cast<OpenPanel *>( ui.Tab->currentWidget() ))->updateMRL();
+/* Used by VLM dialog and inputSlave selection */
+QString OpenDialog::getMRL( bool b_all )
+{
+ if( itemsMRL.count() == 0 ) return "";
+ return b_all ? itemsMRL[0] + getOptions()
+ : itemsMRL[0];
+}
+
+QString OpenDialog::getOptions()
+{
+ return ui.advancedLineInput->text();
+}
+
+void OpenDialog::showTab( int i_tab )
+{
+ if( i_tab == OPEN_CAPTURE_TAB ) captureOpenPanel->initialize();
+ ui.Tab->setCurrentIndex( i_tab );
+ show();
+ if( ui.Tab->currentWidget() != NULL )
+ {
+ OpenPanel *panel = qobject_cast<OpenPanel *>( ui.Tab->currentWidget() );
+ assert( panel );
+ panel->onFocus();
+ }
}
void OpenDialog::toggleAdvancedPanel()
{
- //FIXME does not work under Windows
- if( ui.advancedFrame->isVisible() ) {
+ if( ui.advancedFrame->isVisible() )
+ {
ui.advancedFrame->hide();
-#ifndef WIN32
- setMinimumHeight( 1 );
- resize( width(), mainHeight );
-#endif
- } else {
-#ifndef WIN32
- if( mainHeight == 0 )
- mainHeight = height();
-#endif
-
+ ui.advancedFrame->setEnabled( false );
+ if( size().isValid() )
+ resize( size().width(), size().height()
+ - ui.advancedFrame->height() );
+ }
+ else
+ {
ui.advancedFrame->show();
-#ifndef WIN32
- if( advHeight == 0 ) {
- advHeight = height() - mainHeight;
- }
- resize( width(), mainHeight + advHeight );
-#endif
+ ui.advancedFrame->setEnabled( true );
+ if( size().isValid() )
+ resize( size().width(), size().height()
+ + ui.advancedFrame->height() );
+ }
+}
+
+void OpenDialog::browseInputSlave()
+{
+ OpenDialog *od = new OpenDialog( this, p_intf, true, SELECT );
+ od->exec();
+ ui.slaveText->setText( od->getMRL( false ) );
+ delete od;
+}
+
+/* Function called on signal currentChanged triggered */
+void OpenDialog::signalCurrent( int i_tab )
+{
+ if( i_tab == OPEN_CAPTURE_TAB ) captureOpenPanel->initialize();
+ if( ui.Tab->currentWidget() != NULL )
+ {
+ OpenPanel *panel = qobject_cast<OpenPanel *>( ui.Tab->currentWidget() );
+ assert( panel );
+ panel->onFocus();
+ panel->updateMRL();
}
}
/* If Cancel is pressed or escaped */
void OpenDialog::cancel()
{
+ /* Clear the panels */
for( int i = 0; i < OPEN_TAB_MAX; i++ )
- dynamic_cast<OpenPanel*>(ui.Tab->widget( i ))->clear();
- toggleVisible();
- if( isModal() ) reject();
+ qobject_cast<OpenPanel*>( ui.Tab->widget( i ) )->clear();
+
+ /* Clear the variables */
+ itemsMRL.clear();
+ optionsMRL.clear();
+
+ /* If in Select Mode, reject instead of hiding */
+ if( i_action_flag == SELECT ) reject();
+ else hide();
}
/* If EnterKey is pressed */
void OpenDialog::close()
{
- selectSlots();
+ /* If in Select Mode, accept instead of selecting a Slot */
+ if( i_action_flag == SELECT )
+ accept();
+ else
+ selectSlots();
}
/* Play button */
}
}
+/* Play Action, called from selectSlots or play Menu */
void OpenDialog::play()
{
- finish( false );
+ enqueue( false );
}
-void OpenDialog::enqueue()
+/* Enqueue Action, called from selectSlots or enqueue Menu */
+void OpenDialog::enqueue( bool b_enqueue )
{
- finish( true );
-}
+ toggleVisible();
-void OpenDialog::transcode()
-{
- stream( true );
-}
+ if( i_action_flag == SELECT )
+ {
+ accept();
+ return;
+ }
-void OpenDialog::stream( bool b_transcode_only )
-{
- /* not finished FIXME */
- /* Should go through the finish function */
- THEDP->streamingDialog( mrl, b_transcode_only );
-}
+ for( int i = 0; i < OPEN_TAB_MAX; i++ )
+ qobject_cast<OpenPanel*>( ui.Tab->widget( i ) )->onAccept();
-void OpenDialog::finish( bool b_enqueue = false )
-{
- toggleVisible();
- mrl = ui.advancedLineInput->text();
+ /* Sort alphabetically */
+ itemsMRL.sort();
- if( !isModal() )
+ /* Go through the item list */
+ for( int i = 0; i < itemsMRL.count(); i++ )
{
- QStringList tempMRL = SeparateEntries( mrl );
- for( size_t i = 0; i < tempMRL.size(); i++ )
- {
- bool b_start = !i && !b_enqueue;
- input_item_t *p_input;
+ bool b_start = !i && !b_enqueue;
- p_input = input_ItemNew( p_intf, qtu( tempMRL[i] ), NULL );
+ input_item_t *p_input_item;
+ p_input_item = input_item_New( qtu( itemsMRL[i] ), NULL );
- /* Insert options */
- while( i + 1 < tempMRL.size() && tempMRL[i + 1].startsWith( ":" ) )
- {
- i++;
- input_ItemAddOption( p_input, qtu( tempMRL[i] ) );
- }
+ /* Take options from the UI, not from what we stored */
+ QStringList optionsList = getOptions().split( " :" );
- /* Switch between enqueuing and starting the item */
- if( b_start )
- {
- playlist_AddInput( THEPL, p_input,
- PLAYLIST_APPEND | PLAYLIST_GO,
- PLAYLIST_END, VLC_TRUE, VLC_FALSE );
- }
- else
+ /* Insert options */
+ for( int j = 0; j < optionsList.count(); j++ )
+ {
+ QString qs = colon_unescape( optionsList[j] );
+ if( !qs.isEmpty() )
{
- playlist_AddInput( THEPL, p_input,
- PLAYLIST_APPEND | PLAYLIST_PREPARSE,
- PLAYLIST_END, VLC_TRUE, VLC_FALSE );
+ input_item_AddOption( p_input_item, qtu( qs ),
+ VLC_INPUT_OPTION_TRUSTED );
+#ifdef DEBUG_QT
+ msg_Warn( p_intf, "Input option: %s", qtu( qs ) );
+#endif
}
}
+
+ /* Switch between enqueuing and starting the item */
+ /* FIXME: playlist_AddInput() can fail */
+ playlist_AddInput( THEPL, p_input_item,
+ PLAYLIST_APPEND | ( b_start ? PLAYLIST_GO : PLAYLIST_PREPARSE ),
+ PLAYLIST_END, b_pl ? true : false, pl_Unlocked );
+ vlc_gc_decref( p_input_item );
+
+ /* Do not add the current MRL if playlist_AddInput fail */
+ RecentsMRL::getInstance( p_intf )->addRecent( itemsMRL[i] );
}
- else
- accept();
}
+void OpenDialog::transcode()
+{
+ stream( true );
+}
+
+void OpenDialog::stream( bool b_transcode_only )
+{
+ QString soutMRL = getMRL( false );
+ if( soutMRL.isEmpty() ) return;
+ toggleVisible();
-/* Update the MRL */
-void OpenDialog::updateMRL( QString tempMRL )
+ /* Dbg and send :D */
+ msg_Dbg( p_intf, "MRL passed to the Sout: %s", qtu( soutMRL ) );
+ THEDP->streamingDialog( this, soutMRL, b_transcode_only,
+ getOptions().split( " :" ) );
+}
+
+/* Update the MRL items from the panels */
+void OpenDialog::updateMRL( const QStringList& item, const QString& tempMRL )
{
- mainMRL = tempMRL;
+ optionsMRL = tempMRL;
+ itemsMRL = item;
updateMRL();
}
+/* Update the complete MRL */
void OpenDialog::updateMRL() {
- mrl = mainMRL;
+ QString mrl = optionsMRL;
if( ui.slaveCheckbox->isChecked() ) {
mrl += " :input-slave=" + ui.slaveText->text();
}
- int i_cache = config_GetInt( p_intf, qta( storedMethod ) );
- if( i_cache != ui.cacheSpinBox->value() ) {
- mrl += QString( " :%1=%2" ).arg( storedMethod ).
- arg( ui.cacheSpinBox->value() );
- }
- if( ui.startTimeSpinBox->value() ) {
- mrl += " :start-time=" + QString( "%1" ).
- arg( ui.startTimeSpinBox->value() );
+ mrl += QString( " :%1=%2" ).arg( storedMethod ).
+ arg( ui.cacheSpinBox->value() );
+ if( ui.startTimeTimeEdit->time() != ui.startTimeTimeEdit->minimumTime() ) {
+ mrl += QString( " :start-time=%1.%2" )
+ .arg( QString::number(
+ ui.startTimeTimeEdit->minimumTime().secsTo(
+ ui.startTimeTimeEdit->time()
+ ) ) )
+ .arg( ui.startTimeTimeEdit->time().msec(), 3, 10, QChar('0') );
}
ui.advancedLineInput->setText( mrl );
+ ui.mrlLine->setText( itemsMRL.join( " " ) );
}
-void OpenDialog::newCachingMethod( QString method )
+/* Change the caching combobox */
+void OpenDialog::newCachingMethod( const QString& method )
{
if( method != storedMethod ) {
storedMethod = method;
- int i_value = config_GetInt( p_intf, qta( storedMethod ) );
+ int i_value = var_InheritInteger( p_intf, qtu( storedMethod ) );
ui.cacheSpinBox->setValue( i_value );
}
}
-QStringList OpenDialog::SeparateEntries( QString entries )
+/* Split the entries
+ * FIXME! */
+QStringList OpenDialog::SeparateEntries( const QString& entries )
{
bool b_quotes_mode = false;
QString entry;
int index = 0;
- while( index < entries.size() )
+ while( index < entries.count() )
{
int delim_pos = entries.indexOf( QRegExp( "\\s+|\"" ), index );
- if( delim_pos < 0 ) delim_pos = entries.size() - 1;
+ if( delim_pos < 0 ) delim_pos = entries.count() - 1;
entry += entries.mid( index, delim_pos - index + 1 );
index = delim_pos + 1;
if( !b_quotes_mode && entry.endsWith( "\"" ) )
{
/* Enters quotes mode */
- entry.truncate( entry.size() - 1 );
+ entry.truncate( entry.count() - 1 );
b_quotes_mode = true;
}
else if( b_quotes_mode && entry.endsWith( "\"" ) )
{
/* Finished the quotes mode */
- entry.truncate( entry.size() - 1 );
+ entry.truncate( entry.count() - 1 );
b_quotes_mode = false;
}
else if( !b_quotes_mode && !entry.endsWith( "\"" ) )
{
/* we found a non-quoted standalone string */
- if( index < entries.size() ||
+ if( index < entries.count() ||
entry.endsWith( " " ) || entry.endsWith( "\t" ) ||
entry.endsWith( "\r" ) || entry.endsWith( "\n" ) )
- entry.truncate( entry.size() - 1 );
+ entry.truncate( entry.count() - 1 );
if( !entry.isEmpty() ) entries_array.append( entry );
entry.clear();
}
return entries_array;
}
+