]> git.sesse.net Git - vlc/blob - modules/gui/qt4/main_interface.cpp
Fix Playlist resizing and positionning.
[vlc] / modules / gui / qt4 / main_interface.cpp
1 /*****************************************************************************
2  * main_interface.cpp : Main interface
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  *          Ilkka Ollakka <ileoo@videolan.org>
10  *
11  * This program is free software; you can redistribute it and/or modify
12  * it under the terms of the GNU General Public License as published by
13  * the Free Software Foundation; either version 2 of the License, or
14  * (at your option) any later version.
15  *
16  * This program is distributed in the hope that it will be useful,
17  * but WITHOUT ANY WARRANTY; without even the implied warranty of
18  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
19  * GNU General Public License for more details.
20  *
21  * You should have received a copy of the GNU General Public License
22  * along with this program; if not, write to the Free Software
23  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
24  *****************************************************************************/
25
26 #ifdef HAVE_CONFIG_H
27 # include "config.h"
28 #endif
29
30 #include "qt4.hpp"
31 #include "main_interface.hpp"
32 #include "input_manager.hpp"
33 #include "util/qvlcframe.hpp"
34 #include "util/customwidgets.hpp"
35 #include "dialogs_provider.hpp"
36 #include "components/interface_widgets.hpp"
37 #include "components/playlist/playlist.hpp"
38 #include "dialogs/extended.hpp"
39 #include "dialogs/playlist.hpp"
40 #include "menus.hpp"
41
42 #include <QMenuBar>
43 #include <QCloseEvent>
44 #include <QPushButton>
45 #include <QStatusBar>
46 #include <QKeyEvent>
47 #include <QUrl>
48 #include <QSystemTrayIcon>
49 #include <QSize>
50 #include <QMenu>
51 #include <QLabel>
52 #include <QSlider>
53 #include <QWidgetAction>
54 #include <QToolBar>
55 #include <QGroupBox>
56 #include <QDate>
57
58 #include <assert.h>
59 #include <vlc_keys.h>
60 #include <vlc_vout.h>
61
62 /* Callback prototypes */
63 static int PopupMenuCB( vlc_object_t *p_this, const char *psz_variable,
64                         vlc_value_t old_val, vlc_value_t new_val, void *param );
65 static int IntfShowCB( vlc_object_t *p_this, const char *psz_variable,
66                        vlc_value_t old_val, vlc_value_t new_val, void *param );
67 static int InteractCallback( vlc_object_t *, const char *, vlc_value_t,
68                              vlc_value_t, void *);
69
70 MainInterface::MainInterface( intf_thread_t *_p_intf ) : QVLCMW( _p_intf )
71 {
72     /* Variables initialisation */
73     // need_components_update = false;
74     bgWidget             = NULL;
75     videoWidget          = NULL;
76     playlistWidget       = NULL;
77     sysTray              = NULL;
78     videoIsActive        = false;
79     playlistVisible      = false;
80     input_name           = "";
81     fullscreenControls   = NULL;
82
83     /* Ask for privacy */
84     askForPrivacy();
85
86     /**
87      *  Configuration and settings
88      *  Pre-building of interface
89      **/
90     /* Main settings */
91     setFocusPolicy( Qt::StrongFocus );
92     setAcceptDrops( true );
93     setWindowIcon( QApplication::windowIcon() );
94     setWindowOpacity( config_GetFloat( p_intf, "qt-opacity" ) );
95
96     /* Set The Video In emebedded Mode or not */
97     videoEmbeddedFlag = config_GetInt( p_intf, "embedded-video" );
98
99     /* Are we in the enhanced always-video mode or not ? */
100     i_visualmode = config_GetInt( p_intf, "qt-display-mode" );
101
102     /* Set the other interface settings */
103     settings = getSettings();
104     settings->beginGroup( "MainWindow" );
105
106     /* Visualisation, not really used yet */
107     visualSelectorEnabled = settings->value( "visual-selector", false).toBool();
108
109     /* Do we want anoying popups or not */
110     notificationEnabled = (bool)config_GetInt( p_intf, "qt-notification" );
111
112     /**************************
113      *  UI and Widgets design
114      **************************/
115     setVLCWindowsTitle();
116     handleMainUi( settings );
117
118 #if 0
119     /* Create a Dock to get the playlist */
120     dockPL = new QDockWidget( qtr( "Playlist" ), this );
121     dockPL->setSizePolicy( QSizePolicy::Preferred,
122                            QSizePolicy::Expanding );
123     dockPL->setFeatures( QDockWidget::AllDockWidgetFeatures );
124     dockPL->setAllowedAreas( Qt::LeftDockWidgetArea
125                            | Qt::RightDockWidgetArea
126                            | Qt::BottomDockWidgetArea );
127     dockPL->hide();
128 #endif
129
130     /**************************
131      * Menu Bar and Status Bar
132      **************************/
133     QVLCMenu::createMenuBar( this, p_intf, visualSelectorEnabled );
134     /* StatusBar Creation */
135     createStatusBar();
136
137
138     /********************
139      * Input Manager    *
140      ********************/
141     MainInputManager::getInstance( p_intf );
142
143     /**************************
144      * Various CONNECTs on IM *
145      **************************/
146     /* Connect the input manager to the GUI elements it manages */
147
148     /* It is also connected to the control->slider, see the ControlsWidget */
149     CONNECT( THEMIM->getIM(), positionUpdated( float, int, int ),
150              this, setDisplayPosition( float, int, int ) );
151     /* Change the SpeedRate in the Status */
152     CONNECT( THEMIM->getIM(), rateChanged( int ), this, setRate( int ) );
153
154     /**
155      * Connects on nameChanged()
156      * Those connects are not merged because different options can trigger
157      * them down.
158      */
159     /* Naming in the controller statusbar */
160     CONNECT( THEMIM->getIM(), nameChanged( QString ), this,
161              setName( QString ) );
162     /* and in the systray */
163     if( sysTray )
164     {
165         CONNECT( THEMIM->getIM(), nameChanged( QString ), this,
166                  updateSystrayTooltipName( QString ) );
167     }
168     /* and in the title of the controller */
169     if( config_GetInt( p_intf, "qt-name-in-title" ) )
170     {
171         CONNECT( THEMIM->getIM(), nameChanged( QString ), this,
172              setVLCWindowsTitle( QString ) );
173     }
174
175     /**
176      * CONNECTS on PLAY_STATUS
177      **/
178     /* Status on the main controller */
179     CONNECT( THEMIM->getIM(), statusChanged( int ), this, setStatus( int ) );
180     /* and in the systray */
181     if( sysTray )
182     {
183         CONNECT( THEMIM->getIM(), statusChanged( int ), this,
184                  updateSystrayTooltipStatus( int ) );
185     }
186
187     /* END CONNECTS ON IM */
188
189
190     /** OnTimeOut **/
191     /* TODO Remove this function, but so far, there is no choice because there
192        is no intf-should-die variable #1365 */
193     ON_TIMEOUT( updateOnTimer() );
194
195     /************
196      * Callbacks
197      ************/
198     var_Create( p_intf, "interaction", VLC_VAR_ADDRESS );
199     var_AddCallback( p_intf, "interaction", InteractCallback, this );
200     p_intf->b_interaction = true;
201
202     var_AddCallback( p_intf->p_libvlc, "intf-show", IntfShowCB, p_intf );
203
204     /* Register callback for the intf-popupmenu variable */
205     var_AddCallback( p_intf->p_libvlc, "intf-popupmenu", PopupMenuCB, p_intf );
206
207
208     /* VideoWidget connects to avoid different threads speaking to each other */
209     CONNECT( this, askReleaseVideo( void * ),
210              this, releaseVideoSlot( void * ) );
211     if( videoWidget )
212         CONNECT( this, askVideoToResize( unsigned int, unsigned int ),
213                  videoWidget, SetSizing( unsigned int, unsigned int ) );
214
215     CONNECT( this, askUpdate(), this, doComponentsUpdate() );
216
217     /* Size and placement of interface */
218     QVLCTools::restoreWidgetPosition( settings, this, QSize(380, 60) );
219
220
221     /* Playlist */
222     if( settings->value( "playlist-visible", 0 ).toInt() ) togglePlaylist();
223     settings->endGroup();
224
225     /* Final sizing and showing */
226     setMinimumWidth( __MAX( controls->sizeHint().width(),
227                             menuBar()->sizeHint().width() ) );
228     show();
229
230     /* And switch to minimal view if needed
231        Must be called after the show() */
232     if( i_visualmode == QT_MINIMAL_MODE )
233         toggleMinimalView();
234
235     /* Update the geometry TODO: is it useful ?*/
236     updateGeometry();
237     //    resize( sizeHint() );
238
239     /*****************************************************
240      * End everything by creating the Systray Management *
241      *****************************************************/
242     initSystray();
243 }
244
245 MainInterface::~MainInterface()
246 {
247     msg_Dbg( p_intf, "Destroying the main interface" );
248
249     if( playlistWidget )
250     {
251         if( !isDocked() )
252             QVLCTools::saveWidgetPosition( p_intf, "Playlist", playlistWidget );
253     }
254
255     settings->beginGroup( "MainWindow" );
256
257     settings->setValue( "pl-dock-status", (int)i_pl_dock );
258     settings->setValue( "playlist-visible", (int)playlistVisible );
259     settings->setValue( "adv-controls",
260                         getControlsVisibilityStatus() & CONTROLS_ADVANCED );
261
262     if( !videoIsActive )
263     {
264         QVLCTools::saveWidgetPosition(settings, this);
265     }
266     else
267     {
268         msg_Dbg( p_intf, "Not saving because video is in use." );
269     }
270
271     if( bgWidget )
272         settings->setValue( "backgroundSize", bgWidget->size() );
273
274     settings->endGroup();
275
276     var_DelCallback( p_intf->p_libvlc, "intf-show", IntfShowCB, p_intf );
277
278     /* Unregister callback for the intf-popupmenu variable */
279     var_DelCallback( p_intf->p_libvlc, "intf-popupmenu", PopupMenuCB, p_intf );
280
281     p_intf->b_interaction = false;
282     var_DelCallback( p_intf, "interaction", InteractCallback, this );
283
284     p_intf->p_sys->p_mi = NULL;
285 }
286
287 /*****************************
288  *   Main UI handling        *
289  *****************************/
290
291 inline void MainInterface::createStatusBar()
292 {
293     /****************
294      *  Status Bar  *
295      ****************/
296     /* Widgets Creation*/
297     b_remainingTime = false;
298     timeLabel = new TimeLabel;
299     timeLabel->setText( " --:--/--:-- " );
300     timeLabel->setAlignment( Qt::AlignRight | Qt::AlignVCenter );
301     timeLabel->setToolTip( qtr( "Toggle between elapsed and remaining time" ) );
302     nameLabel = new QLabel;
303     nameLabel->setTextInteractionFlags( Qt::TextSelectableByMouse
304                                       | Qt::TextSelectableByKeyboard );
305     speedLabel = new SpeedLabel( p_intf, "1.00x" );
306     speedLabel->setToolTip(
307             qtr( "Current playback speed.\nRight click to adjust" ) );
308     speedLabel->setContextMenuPolicy ( Qt::CustomContextMenu );
309
310     /* Styling those labels */
311     timeLabel->setFrameStyle( QFrame::Sunken | QFrame::Panel );
312     speedLabel->setFrameStyle( QFrame::Sunken | QFrame::Panel );
313     nameLabel->setFrameStyle( QFrame::Sunken | QFrame::StyledPanel);
314
315     /* and adding those */
316     statusBar()->addWidget( nameLabel, 8 );
317     statusBar()->addPermanentWidget( speedLabel, 0 );
318     statusBar()->addPermanentWidget( timeLabel, 0 );
319
320     /* timeLabel behaviour:
321        - double clicking opens the goto time dialog
322        - right-clicking and clicking just toggle between remaining and
323          elapsed time.*/
324     CONNECT( timeLabel, timeLabelClicked(), this, toggleTimeDisplay() );
325     CONNECT( timeLabel, timeLabelDoubleClicked(), THEDP, gotoTimeDialog() );
326     CONNECT( timeLabel, timeLabelDoubleClicked(), this, toggleTimeDisplay() );
327
328     /* Speed Label behaviour:
329        - right click gives the vertical speed slider */
330     CONNECT( speedLabel, customContextMenuRequested( QPoint ),
331              this, showSpeedMenu( QPoint ) );
332 }
333
334 inline void MainInterface::initSystray()
335 {
336     bool b_createSystray = false;
337     bool b_systrayAvailable = QSystemTrayIcon::isSystemTrayAvailable();
338     if( config_GetInt( p_intf, "qt-start-minimized") )
339     {
340         if( b_systrayAvailable )
341         {
342             b_createSystray = true;
343             hide();
344         }
345         else msg_Err( p_intf, "You can't minimize if you haven't a system "
346                 "tray bar" );
347     }
348     if( config_GetInt( p_intf, "qt-system-tray") )
349         b_createSystray = true;
350
351     if( b_systrayAvailable && b_createSystray )
352             createSystray();
353 }
354
355 /**
356  * Give the decorations of the Main Window a correct Name.
357  * If nothing is given, set it to VLC...
358  **/
359 void MainInterface::setVLCWindowsTitle( QString aTitle )
360 {
361     if( aTitle.isEmpty() )
362     {
363         setWindowTitle( qtr( "VLC media player" ) );
364     }
365     else
366     {
367         setWindowTitle( aTitle + " - " + qtr( "VLC media player" ) );
368     }
369 }
370
371 void MainInterface::handleMainUi( QSettings *settings )
372 {
373     /* Create the main Widget and the mainLayout */
374     QWidget *main = new QWidget;
375     setCentralWidget( main );
376     mainLayout = new QVBoxLayout( main );
377
378     /* Margins, spacing */
379     main->setContentsMargins( 0, 0, 0, 0 );
380     main->setSizePolicy( QSizePolicy::Preferred, QSizePolicy::Maximum );
381     mainLayout->setSpacing( 0 );
382     mainLayout->setMargin( 0 );
383
384     /* Create the CONTROLS Widget */
385     bool b_shiny = config_GetInt( p_intf, "qt-blingbling" );
386     controls = new ControlsWidget( p_intf, this,
387                    settings->value( "adv-controls", false ).toBool(),
388                    b_shiny );
389     CONNECT( controls, advancedControlsToggled( bool ),
390              this, doComponentsUpdate() );
391
392 #ifdef WIN32
393     if ( depth() > 8 )
394 #endif
395     /* Create the FULLSCREEN CONTROLS Widget */
396     if( config_GetInt( p_intf, "qt-fs-controller" ) )
397     {
398         fullscreenControls = new FullscreenControllerWidget( p_intf, this,
399                 settings->value( "adv-controls", false ).toBool(),
400                 b_shiny );
401         CONNECT( fullscreenControls, advancedControlsToggled( bool ),
402                 this, doComponentsUpdate() );
403     }
404
405     /* Add the controls Widget to the main Widget */
406     mainLayout->insertWidget( 0, controls, 0, Qt::AlignBottom );
407
408     /* Create the Speed Control Widget */
409     speedControl = new SpeedControlWidget( p_intf );
410     speedControlMenu = new QMenu( this );
411
412     QWidgetAction *widgetAction = new QWidgetAction( speedControl );
413     widgetAction->setDefaultWidget( speedControl );
414     speedControlMenu->addAction( widgetAction );
415
416     /* Visualisation */
417     /* Disabled for now, they SUCK */
418     #if 0
419     visualSelector = new VisualSelector( p_intf );
420     mainLayout->insertWidget( 0, visualSelector );
421     visualSelector->hide();
422     #endif
423
424     /* Bg Cone */
425     bgWidget = new BackgroundWidget( p_intf );
426     bgWidget->resize(
427             settings->value( "backgroundSize", QSize( 300, 200 ) ).toSize() );
428     bgWidget->updateGeometry();
429     mainLayout->insertWidget( 0, bgWidget );
430     CONNECT( this, askBgWidgetToToggle(), bgWidget, toggle() );
431
432     if( i_visualmode != QT_ALWAYS_VIDEO_MODE &&
433         i_visualmode != QT_MINIMAL_MODE )
434     {
435         bgWidget->hide();
436     }
437
438     /* And video Outputs */
439     if( videoEmbeddedFlag )
440     {
441         videoWidget = new VideoWidget( p_intf );
442         mainLayout->insertWidget( 0, videoWidget, 10 );
443     }
444
445     /* Finish the sizing */
446     main->updateGeometry();
447 }
448
449 inline void MainInterface::askForPrivacy()
450 {
451     /**
452      * Ask for the network policy on FIRST STARTUP
453      **/
454     if( config_GetInt( p_intf, "qt-privacy-ask") )
455     {
456         QList<ConfigControl *> controls;
457         if( privacyDialog( &controls ) == QDialog::Accepted )
458         {
459             QList<ConfigControl *>::Iterator i;
460             for(  i = controls.begin() ; i != controls.end() ; i++ )
461             {
462                 ConfigControl *c = qobject_cast<ConfigControl *>(*i);
463                 c->doApply( p_intf );
464             }
465
466             config_PutInt( p_intf,  "qt-privacy-ask" , 0 );
467             /* We have to save here because the user may not launch Prefs */
468             config_SaveConfigFile( p_intf, NULL );
469         }
470     }
471 }
472
473 int MainInterface::privacyDialog( QList<ConfigControl *> *controls )
474 {
475     QDialog *privacy = new QDialog();
476
477     privacy->setWindowTitle( qtr( "Privacy and Network Policies" ) );
478
479     QGridLayout *gLayout = new QGridLayout( privacy );
480
481     QGroupBox *blabla = new QGroupBox( qtr( "Privacy and Network Warning" ) );
482     QGridLayout *blablaLayout = new QGridLayout( blabla );
483     QLabel *text = new QLabel( qtr(
484         "<p>The <i>VideoLAN Team</i> doesn't like when an application goes "
485         "online without authorization.</p>\n "
486         "<p><i>VLC media player</i> can request limited information on "
487         "the Internet, especially to get CD covers or to know "
488         "if updates are available.</p>\n"
489         "<p><i>VLC media player</i> <b>DOES NOT</b> send or collect <b>ANY</b> "
490         "information, even anonymously, about your usage.</p>\n"
491         "<p>Therefore please check the following options, the default being "
492         "almost no access on the web.</p>\n") );
493     text->setWordWrap( true );
494     text->setTextFormat( Qt::RichText );
495
496     blablaLayout->addWidget( text, 0, 0 ) ;
497
498     QGroupBox *options = new QGroupBox;
499     QGridLayout *optionsLayout = new QGridLayout( options );
500
501     gLayout->addWidget( blabla, 0, 0, 1, 3 );
502     gLayout->addWidget( options, 1, 0, 1, 3 );
503     module_config_t *p_config;
504     ConfigControl *control;
505     int line = 0;
506
507 #define CONFIG_GENERIC( option, type )                            \
508     p_config =  config_FindConfig( VLC_OBJECT(p_intf), option );  \
509     if( p_config )                                                \
510     {                                                             \
511         control =  new type ## ConfigControl( VLC_OBJECT(p_intf), \
512                 p_config, options, false, optionsLayout, line );  \
513         controls->append( control );                               \
514     }
515
516 #define CONFIG_GENERIC_NOBOOL( option, type )                     \
517     p_config =  config_FindConfig( VLC_OBJECT(p_intf), option );  \
518     if( p_config )                                                \
519     {                                                             \
520         control =  new type ## ConfigControl( VLC_OBJECT(p_intf), \
521                 p_config, options, optionsLayout, line );  \
522         controls->append( control );                               \
523     }
524
525     CONFIG_GENERIC( "album-art", IntegerList ); line++;
526 #ifdef UPDATE_CHECK
527     CONFIG_GENERIC_NOBOOL( "qt-updates-notif", Bool ); line++;
528     CONFIG_GENERIC_NOBOOL( "qt-updates-days", Integer ); line++;
529 #endif
530
531     QPushButton *ok = new QPushButton( qtr( "OK" ) );
532
533     gLayout->addWidget( ok, 2, 2 );
534
535     CONNECT( ok, clicked(), privacy, accept() );
536     return privacy->exec();
537 }
538
539
540 /**********************************************************************
541  * Handling of sizing of the components
542  **********************************************************************/
543
544 /* This function is probably wrong, but we don't have many many choices...
545    Since we can't know from the playlist Widget if we are inside a dock or not,
546    because the playlist Widget can be called by THEDP, as a separate windows for
547    the skins.
548    Maybe the other solution is to redefine the sizeHint() of the playlist and
549    ask _parent->isFloating()...
550    If you think this would be better, please FIXME it...
551 */
552
553 QSize MainInterface::sizeHint() const
554 {
555     int nwidth  = controls->sizeHint().width();
556     int nheight = controls->isVisible() ?
557                   controls->size().height()
558                   + menuBar()->size().height()
559                   + statusBar()->size().height()
560                   : 0 ;
561
562     if( VISIBLE( bgWidget ) )
563     {
564         nheight += bgWidget->size().height();
565         nwidth  = bgWidget->size().width();
566     }
567     else if( videoIsActive && videoWidget->isVisible() )
568     {
569         nheight += videoWidget->sizeHint().height();
570         nwidth  = videoWidget->sizeHint().width();
571     }
572 #if 0
573     if( !dockPL->isFloating() && dockPL->isVisible() && dockPL->widget()  )
574     {
575         nheight += dockPL->size().height();
576         nwidth = __MAX( nwidth, dockPL->size().width() );
577         msg_Warn( p_intf, "3 %i %i", nheight, nwidth );
578     }
579 #endif
580     return QSize( nwidth, nheight );
581 }
582
583 void MainInterface::toggleFSC()
584 {
585    if( !fullscreenControls ) return;
586
587    IMEvent *eShow = new IMEvent( FullscreenControlToggle_Type, 0 );
588    QApplication::postEvent( fullscreenControls, static_cast<QEvent *>(eShow) );
589 }
590
591 void MainInterface::debug()
592 {
593 #ifndef NDEBUG
594     msg_Dbg( p_intf, "size: %i - %i", size().height(), size().width() );
595     msg_Dbg( p_intf, "sizeHint: %i - %i", sizeHint().height(), sizeHint().width() );
596     if( videoWidget && videoWidget->isVisible() )
597     {
598         //    sleep( 10 );
599         msg_Dbg( p_intf, "size: %i - %i", size().height(), size().width() );
600         msg_Dbg( p_intf, "sizeHint: %i - %i", sizeHint().height(), sizeHint().width() );
601     }
602     adjustSize();
603 #endif
604 }
605
606 /****************************************************************************
607  * Small right-click menu for rate control
608  ****************************************************************************/
609 void MainInterface::showSpeedMenu( QPoint pos )
610 {
611     speedControlMenu->exec( QCursor::pos() - pos
612                           + QPoint( 0, speedLabel->height() ) );
613 }
614
615 /****************************************************************************
616  * Video Handling
617  ****************************************************************************/
618
619 /* This event is used to deal with the fullscreen and always on top
620    issue conflict (bug in wx) */
621 class SetVideoOnTopQtEvent : public QEvent
622 {
623 public:
624     SetVideoOnTopQtEvent( bool _onTop ) :
625       QEvent( (QEvent::Type)SetVideoOnTopEvent_Type ), onTop( _onTop)
626     {}
627
628     bool OnTop() const { return onTop; }
629
630 private:
631     bool onTop;
632 };
633
634 /**
635  * README
636  * Thou shall not call/resize/hide widgets from on another thread.
637  * This is wrong, and this is TEH reason to emit signals on those Video Functions
638  **/
639 void *MainInterface::requestVideo( vout_thread_t *p_nvout, int *pi_x,
640                                    int *pi_y, unsigned int *pi_width,
641                                    unsigned int *pi_height )
642 {
643     /* Request the videoWidget */
644     void *ret = videoWidget->request( p_nvout,pi_x, pi_y, pi_width, pi_height );
645     if( ret ) /* The videoWidget is available */
646     {
647         /* Did we have a bg ? Hide it! */
648         if( VISIBLE( bgWidget) )
649         {
650             bgWasVisible = true;
651             emit askBgWidgetToToggle();
652         }
653         else
654             bgWasVisible = false;
655
656         /* Consider the video active now */
657         videoIsActive = true;
658
659         emit askUpdate();
660
661         if( fullscreenControls ) fullscreenControls->attachVout( p_nvout );
662     }
663     return ret;
664 }
665
666 /* Call from the WindowClose function */
667 void MainInterface::releaseVideo( void *p_win )
668 {
669     if( fullscreenControls ) fullscreenControls->detachVout();
670     if( p_win )
671         emit askReleaseVideo( p_win );
672 }
673
674 /* Function that is CONNECTED to the previous emit */
675 void MainInterface::releaseVideoSlot( void *p_win )
676 {
677     videoWidget->release( p_win );
678
679     if( bgWasVisible )
680     {
681         /* Reset the bg state */
682         bgWasVisible = false;
683         bgWidget->show();
684     }
685
686     videoIsActive = false;
687
688     /* Try to resize, except when you are in Fullscreen mode */
689     if( !isFullScreen() ) doComponentsUpdate();
690 }
691
692 /* Call from WindowControl function */
693 int MainInterface::controlVideo( void *p_window, int i_query, va_list args )
694 {
695     int i_ret = VLC_SUCCESS;
696     switch( i_query )
697     {
698         case VOUT_GET_SIZE:
699         {
700             unsigned int *pi_width  = va_arg( args, unsigned int * );
701             unsigned int *pi_height = va_arg( args, unsigned int * );
702             *pi_width = videoWidget->videoSize.width();
703             *pi_height = videoWidget->videoSize.height();
704             break;
705         }
706         case VOUT_SET_SIZE:
707         {
708             unsigned int i_width  = va_arg( args, unsigned int );
709             unsigned int i_height = va_arg( args, unsigned int );
710             emit askVideoToResize( i_width, i_height );
711             emit askUpdate();
712             break;
713         }
714         case VOUT_SET_STAY_ON_TOP:
715         {
716             int i_arg = va_arg( args, int );
717             QApplication::postEvent( this, new SetVideoOnTopQtEvent( i_arg ) );
718             break;
719         }
720         default:
721             i_ret = VLC_EGENERIC;
722             msg_Warn( p_intf, "unsupported control query" );
723             break;
724     }
725     return i_ret;
726 }
727
728 /*****************************************************************************
729  * Playlist, Visualisation and Menus handling
730  *****************************************************************************/
731 /**
732  * Toggle the playlist widget or dialog
733  **/
734 void MainInterface::togglePlaylist()
735 {
736     /* CREATION
737     If no playlist exist, then create one and attach it to the DockPL*/
738     if( !playlistWidget )
739     {
740         playlistWidget = new PlaylistWidget( p_intf, this );
741
742         i_pl_dock = (pl_dock_e)getSettings()
743                          ->value( "pl-dock-status", PL_UNDOCKED ).toInt();
744         if( i_pl_dock == PL_UNDOCKED )
745         {
746             playlistWidget->setWindowFlags( Qt::Window );
747
748             QVLCTools::restoreWidgetPosition( p_intf, "Playlist",
749                     playlistWidget, QSize( 600, 300 ) );
750         }
751         else
752         {
753             mainLayout->insertWidget( 4, playlistWidget );
754         }
755         playlistVisible = true;
756
757         playlistWidget->show();
758     }
759     else
760     {
761     /* toggle the visibility of the playlist */
762        TOGGLEV( playlistWidget );
763        resize( sizeHint() );
764        playlistVisible = !playlistVisible;
765     }
766 }
767
768 /* Function called from the menu to undock the playlist */
769 void MainInterface::undockPlaylist()
770 {
771 //    dockPL->setFloating( true );
772     adjustSize();
773 }
774
775 void MainInterface::dockPlaylist( pl_dock_e i_pos )
776 {
777 }
778
779 void MainInterface::toggleMinimalView()
780 {
781     /* HACK for minimalView, see menus.cpp */
782     if( !menuBar()->isVisible() ) QVLCMenu::minimalViewAction->toggle();
783
784     if( i_visualmode != QT_ALWAYS_VIDEO_MODE &&
785         i_visualmode != QT_MINIMAL_MODE )
786     { /* NORMAL MODE then */
787         if( videoWidget->isHidden() ) emit askBgWidgetToToggle();
788         else
789         {
790             /* If video is visible, then toggle the status of bgWidget */
791             bgWasVisible = !bgWasVisible;
792         }
793     }
794
795     TOGGLEV( menuBar() );
796     TOGGLEV( controls );
797     TOGGLEV( statusBar() );
798     doComponentsUpdate();
799 }
800
801 /* Video widget cannot do this synchronously as it runs in another thread */
802 /* Well, could it, actually ? Probably dangerous ... */
803 void MainInterface::doComponentsUpdate()
804 {
805     msg_Dbg( p_intf, "Updating the geometry" );
806     //    resize( sizeHint() );
807     adjustSize();
808 }
809
810 /* toggling advanced controls buttons */
811 void MainInterface::toggleAdvanced()
812 {
813     controls->toggleAdvanced();
814     if( fullscreenControls ) fullscreenControls->toggleAdvanced();
815 }
816
817 /* Get the visibility status of the controls (hidden or not, advanced or not) */
818 int MainInterface::getControlsVisibilityStatus()
819 {
820     return( (controls->isVisible() ? CONTROLS_VISIBLE : CONTROLS_HIDDEN )
821                 + CONTROLS_ADVANCED * controls->b_advancedVisible );
822 }
823
824 #if 0
825 void MainInterface::visual()
826 {
827     if( !VISIBLE( visualSelector) )
828     {
829         visualSelector->show();
830         if( !THEMIM->getIM()->hasVideo() )
831         {
832             /* Show the background widget */
833         }
834         visualSelectorEnabled = true;
835     }
836     else
837     {
838         /* Stop any currently running visualization */
839         visualSelector->hide();
840         visualSelectorEnabled = false;
841     }
842     doComponentsUpdate();
843 }
844 #endif
845
846 /************************************************************************
847  * Other stuff
848  ************************************************************************/
849 void MainInterface::setDisplayPosition( float pos, int time, int length )
850 {
851     char psz_length[MSTRTIME_MAX_SIZE], psz_time[MSTRTIME_MAX_SIZE];
852     secstotimestr( psz_length, length );
853     secstotimestr( psz_time, ( b_remainingTime && length ) ? length - time
854                                                            : time );
855
856     QString timestr;
857     timestr.sprintf( "%s/%s", psz_time,
858                             ( !length && time ) ? "--:--" : psz_length );
859
860     /* Add a minus to remaining time*/
861     if( b_remainingTime && length ) timeLabel->setText( " -"+timestr+" " );
862     else timeLabel->setText( " "+timestr+" " );
863 }
864
865 void MainInterface::toggleTimeDisplay()
866 {
867     b_remainingTime = !b_remainingTime;
868 }
869
870 void MainInterface::setName( QString name )
871 {
872     input_name = name; /* store it for the QSystray use */
873     /* Display it in the status bar, but also as a Tooltip in case it doesn't
874        fit in the label */
875     nameLabel->setText( " " + name + " " );
876     nameLabel->setToolTip( " " + name +" " );
877 }
878
879 void MainInterface::setStatus( int status )
880 {
881     msg_Dbg( p_intf, "Updating the stream status: %i", status );
882
883     /* Forward the status to the controls to toggle Play/Pause */
884     controls->setStatus( status );
885     controls->updateInput();
886
887     if( fullscreenControls )
888     {
889         fullscreenControls->setStatus( status );
890         fullscreenControls->updateInput();
891     }
892
893     speedControl->setEnable( THEMIM->getIM()->hasInput() );
894
895     /* And in the systray for the menu */
896     if( sysTray )
897         QVLCMenu::updateSystrayMenu( this, p_intf );
898 }
899
900 void MainInterface::setRate( int rate )
901 {
902     QString str;
903     str.setNum( ( 1000 / (double)rate ), 'f', 2 );
904     str.append( "x" );
905     speedLabel->setText( str );
906     speedLabel->setToolTip( str );
907     speedControl->updateControls( rate );
908 }
909
910 void MainInterface::updateOnTimer()
911 {
912     /* No event for dying */
913     if( intf_ShouldDie( p_intf ) )
914     {
915         QApplication::closeAllWindows();
916         QApplication::quit();
917     }
918 }
919
920 /*****************************************************************************
921  * Systray Icon and Systray Menu
922  *****************************************************************************/
923
924 /**
925  * Create a SystemTray icon and a menu that would go with it.
926  * Connects to a click handler on the icon.
927  **/
928 void MainInterface::createSystray()
929 {
930     QIcon iconVLC;
931     if( QDate::currentDate().dayOfYear() >= 354 )
932         iconVLC =  QIcon( QPixmap( ":/vlc128-christmas.png" ) );
933     else
934         iconVLC =  QIcon( QPixmap( ":/vlc128.png" ) );
935     sysTray = new QSystemTrayIcon( iconVLC, this );
936     sysTray->setToolTip( qtr( "VLC media player" ));
937
938     systrayMenu = new QMenu( qtr( "VLC media player" ), this );
939     systrayMenu->setIcon( iconVLC );
940
941     QVLCMenu::updateSystrayMenu( this, p_intf, true );
942     sysTray->show();
943
944     CONNECT( sysTray, activated( QSystemTrayIcon::ActivationReason ),
945             this, handleSystrayClick( QSystemTrayIcon::ActivationReason ) );
946 }
947
948 /**
949  * Updates the Systray Icon's menu and toggle the main interface
950  */
951 void MainInterface::toggleUpdateSystrayMenu()
952 {
953     /* If hidden, show it */
954     if( isHidden() )
955     {
956         show();
957         activateWindow();
958     }
959     else if( isMinimized() )
960     {
961         /* Minimized */
962         showNormal();
963         activateWindow();
964     }
965     else
966     {
967         /* Visible */
968 #ifdef WIN32
969         /* check if any visible window is above vlc in the z-order,
970          * but ignore the ones always on top */
971         WINDOWINFO wi;
972         HWND hwnd;
973         wi.cbSize = sizeof( WINDOWINFO );
974         for( hwnd = GetNextWindow( internalWinId(), GW_HWNDPREV );
975                 hwnd && !IsWindowVisible( hwnd );
976                 hwnd = GetNextWindow( hwnd, GW_HWNDPREV ) );
977         if( !hwnd || !GetWindowInfo( hwnd, &wi ) ||
978                 (wi.dwExStyle&WS_EX_TOPMOST) )
979 #else
980         if( isActiveWindow() )
981 #endif
982         {
983             hide();
984         }
985         else
986         {
987             activateWindow();
988         }
989     }
990     QVLCMenu::updateSystrayMenu( this, p_intf );
991 }
992
993 void MainInterface::handleSystrayClick(
994                                     QSystemTrayIcon::ActivationReason reason )
995 {
996     switch( reason )
997     {
998         case QSystemTrayIcon::Trigger:
999             toggleUpdateSystrayMenu();
1000             break;
1001         case QSystemTrayIcon::MiddleClick:
1002             sysTray->showMessage( qtr( "VLC media player" ),
1003                     qtr( "Control menu for the player" ),
1004                     QSystemTrayIcon::Information, 3000 );
1005             break;
1006     }
1007 }
1008
1009 /**
1010  * Updates the name of the systray Icon tooltip.
1011  * Doesn't check if the systray exists, check before you call it.
1012  **/
1013 void MainInterface::updateSystrayTooltipName( QString name )
1014 {
1015     if( name.isEmpty() )
1016     {
1017         sysTray->setToolTip( qtr( "VLC media player" ) );
1018     }
1019     else
1020     {
1021         sysTray->setToolTip( name );
1022         if( notificationEnabled && ( isHidden() || isMinimized() ) )
1023         {
1024             sysTray->showMessage( qtr( "VLC media player" ), name,
1025                     QSystemTrayIcon::NoIcon, 3000 );
1026         }
1027     }
1028 }
1029
1030 /**
1031  * Updates the status of the systray Icon tooltip.
1032  * Doesn't check if the systray exists, check before you call it.
1033  **/
1034 void MainInterface::updateSystrayTooltipStatus( int i_status )
1035 {
1036     switch( i_status )
1037     {
1038         case  0:
1039         case  END_S:
1040             {
1041                 sysTray->setToolTip( qtr( "VLC media player" ) );
1042                 break;
1043             }
1044         case PLAYING_S:
1045             {
1046                 sysTray->setToolTip( input_name );
1047                 break;
1048             }
1049         case PAUSE_S:
1050             {
1051                 sysTray->setToolTip( input_name + " - "
1052                         + qtr( "Paused") );
1053                 break;
1054             }
1055     }
1056 }
1057
1058 /************************************************************************
1059  * D&D Events
1060  ************************************************************************/
1061 void MainInterface::dropEvent(QDropEvent *event)
1062 {
1063      const QMimeData *mimeData = event->mimeData();
1064
1065      /* D&D of a subtitles file, add it on the fly */
1066      if( mimeData->urls().size() == 1 )
1067      {
1068         if( THEMIM->getIM()->hasInput() )
1069         {
1070             if( input_AddSubtitles( THEMIM->getInput(),
1071                                     qtu( mimeData->urls()[0].toString() ),
1072                                     true ) )
1073             {
1074                 event->acceptProposedAction();
1075                 return;
1076             }
1077         }
1078      }
1079      bool first = true;
1080      foreach( QUrl url, mimeData->urls() )
1081      {
1082         QString s = url.toLocalFile();
1083         if( s.length() > 0 ) {
1084             playlist_Add( THEPL, qtu(s), NULL,
1085                           PLAYLIST_APPEND | (first ? PLAYLIST_GO:0),
1086                           PLAYLIST_END, true, false );
1087             first = false;
1088         }
1089      }
1090      event->acceptProposedAction();
1091 }
1092 void MainInterface::dragEnterEvent(QDragEnterEvent *event)
1093 {
1094      event->acceptProposedAction();
1095 }
1096 void MainInterface::dragMoveEvent(QDragMoveEvent *event)
1097 {
1098      event->acceptProposedAction();
1099 }
1100 void MainInterface::dragLeaveEvent(QDragLeaveEvent *event)
1101 {
1102      event->accept();
1103 }
1104
1105 /************************************************************************
1106  * Events stuff
1107  ************************************************************************/
1108 void MainInterface::customEvent( QEvent *event )
1109 {
1110 #if 0
1111     if( event->type() == PLDockEvent_Type )
1112     {
1113         PlaylistDialog::killInstance();
1114         playlistEmbeddedFlag = true;
1115         menuBar()->clear();
1116         QVLCMenu::createMenuBar(this, p_intf, true, visualSelectorEnabled);
1117         togglePlaylist();
1118     }
1119 #endif
1120     /*else */
1121     if ( event->type() == SetVideoOnTopEvent_Type )
1122     {
1123         SetVideoOnTopQtEvent* p_event = (SetVideoOnTopQtEvent*)event;
1124         if( p_event->OnTop() )
1125             setWindowFlags(windowFlags() | Qt::WindowStaysOnTopHint);
1126         else
1127             setWindowFlags(windowFlags() & ~Qt::WindowStaysOnTopHint);
1128         show(); /* necessary to apply window flags?? */
1129     }
1130 }
1131
1132 void MainInterface::keyPressEvent( QKeyEvent *e )
1133 {
1134     if( ( e->modifiers() &  Qt::ControlModifier ) && ( e->key() & Qt::Key_H )
1135           && menuBar()->isHidden() )
1136     {
1137         toggleMinimalView();
1138         e->accept();
1139     }
1140
1141     int i_vlck = qtEventToVLCKey( e );
1142     if( i_vlck > 0 )
1143     {
1144         var_SetInteger( p_intf->p_libvlc, "key-pressed", i_vlck );
1145         e->accept();
1146     }
1147     else
1148         e->ignore();
1149 }
1150
1151 void MainInterface::wheelEvent( QWheelEvent *e )
1152 {
1153     int i_vlckey = qtWheelEventToVLCKey( e );
1154     var_SetInteger( p_intf->p_libvlc, "key-pressed", i_vlckey );
1155     e->accept();
1156 }
1157
1158 void MainInterface::closeEvent( QCloseEvent *e )
1159 {
1160     hide();
1161     THEDP->quit();
1162 }
1163
1164 void MainInterface::toggleFullScreen( void )
1165 {
1166     if( isFullScreen() )
1167     {
1168         showNormal();
1169         emit askUpdate(); // Needed if video was launched after the F11
1170     }
1171     else
1172         showFullScreen();
1173 }
1174
1175 /*****************************************************************************
1176  * Callbacks
1177  *****************************************************************************/
1178 static int InteractCallback( vlc_object_t *p_this,
1179                              const char *psz_var, vlc_value_t old_val,
1180                              vlc_value_t new_val, void *param )
1181 {
1182     intf_dialog_args_t *p_arg = new intf_dialog_args_t;
1183     p_arg->p_dialog = (interaction_dialog_t *)(new_val.p_address);
1184     DialogEvent *event = new DialogEvent( INTF_DIALOG_INTERACTION, 0, p_arg );
1185     QApplication::postEvent( THEDP, static_cast<QEvent*>(event) );
1186     return VLC_SUCCESS;
1187 }
1188
1189 /*****************************************************************************
1190  * PopupMenuCB: callback triggered by the intf-popupmenu playlist variable.
1191  *  We don't show the menu directly here because we don't want the
1192  *  caller to block for a too long time.
1193  *****************************************************************************/
1194 static int PopupMenuCB( vlc_object_t *p_this, const char *psz_variable,
1195                         vlc_value_t old_val, vlc_value_t new_val, void *param )
1196 {
1197     intf_thread_t *p_intf = (intf_thread_t *)param;
1198
1199     if( p_intf->pf_show_dialog )
1200     {
1201         p_intf->pf_show_dialog( p_intf, INTF_DIALOG_POPUPMENU,
1202                                 new_val.b_bool, 0 );
1203     }
1204
1205     return VLC_SUCCESS;
1206 }
1207
1208 /*****************************************************************************
1209  * IntfShowCB: callback triggered by the intf-show libvlc variable.
1210  *****************************************************************************/
1211 static int IntfShowCB( vlc_object_t *p_this, const char *psz_variable,
1212                        vlc_value_t old_val, vlc_value_t new_val, void *param )
1213 {
1214     intf_thread_t *p_intf = (intf_thread_t *)param;
1215     p_intf->p_sys->p_mi->toggleFSC();
1216
1217     /* Show event */
1218      return VLC_SUCCESS;
1219 }