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