]> git.sesse.net Git - vlc/blob - modules/gui/qt4/main_interface.cpp
var_InheritInteger -> var_InheritBool
[vlc] / modules / gui / qt4 / main_interface.cpp
1 /*****************************************************************************
2  * main_interface.cpp : Main interface
3  ****************************************************************************
4  * Copyright (C) 2006-2010 VideoLAN and AUTHORS
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 Foundation, Inc.,
23  * 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
32 #include "main_interface.hpp"
33 #include "input_manager.hpp"                    // Creation
34 #include "actions_manager.hpp"                  // killInstance
35 #include "extensions_manager.hpp"               // killInstance
36
37 #include "util/customwidgets.hpp"               // qtEventToVLCKey, QVLCStackedWidget
38 #include "util/qt_dirs.hpp"                     // toNativeSeparators
39
40 #include "components/interface_widgets.hpp"     // bgWidget, videoWidget
41 #include "components/controller.hpp"            // controllers
42 #include "components/playlist/playlist.hpp"     // plWidget
43 #include "dialogs/firstrun.hpp"                 // First Run
44
45 #include "menus.hpp"                            // Menu creation
46 #include "recents.hpp"                          // RecentItems when DnD
47
48 #include <QCloseEvent>
49 #include <QKeyEvent>
50
51 #include <QUrl>
52 #include <QSize>
53 #include <QDate>
54
55 #include <QMenu>
56 #include <QMenuBar>
57 #include <QStatusBar>
58 #include <QLabel>
59 #include <QStackedWidget>
60
61 #include <vlc_keys.h>                       /* Wheel event */
62 #include <vlc_vout_display.h>               /* vout_thread_t and VOUT_ events */
63
64 // #define DEBUG_INTF
65
66 /* Callback prototypes */
67 static int PopupMenuCB( vlc_object_t *p_this, const char *psz_variable,
68                         vlc_value_t old_val, vlc_value_t new_val, void *param );
69 static int IntfShowCB( vlc_object_t *p_this, const char *psz_variable,
70                        vlc_value_t old_val, vlc_value_t new_val, void *param );
71
72 MainInterface::MainInterface( intf_thread_t *_p_intf ) : QVLCMW( _p_intf )
73 {
74     /* Variables initialisation */
75     bgWidget             = NULL;
76     videoWidget          = NULL;
77     playlistWidget       = NULL;
78     stackCentralOldWidget= NULL;
79 #ifndef HAVE_MAEMO
80     sysTray              = NULL;
81 #endif
82     fullscreenControls   = NULL;
83     cryptedLabel         = NULL;
84     controls             = NULL;
85     inputC               = NULL;
86
87     b_hideAfterCreation  = false; // --qt-start-minimized
88     playlistVisible      = false;
89     input_name           = "";
90
91
92     /* Ask for Privacy */
93     FirstRun::CheckAndRun( this, p_intf );
94
95     /**
96      *  Configuration and settings
97      *  Pre-building of interface
98      **/
99     /* Main settings */
100     setFocusPolicy( Qt::StrongFocus );
101     setAcceptDrops( true );
102     setWindowRole( "vlc-main" );
103     setWindowIcon( QApplication::windowIcon() );
104     setWindowOpacity( var_InheritFloat( p_intf, "qt-opacity" ) );
105
106     /* Is video in embedded in the UI or not */
107     b_videoEmbedded = var_InheritBool( p_intf, "embedded-video" );
108
109     /* Does the interface resize to video size or the opposite */
110     b_autoresize = var_InheritBool( p_intf, "qt-video-autoresize" );
111
112     /* Are we in the enhanced always-video mode or not ? */
113     i_visualmode = var_InheritBool( p_intf, "qt-minimal-view" );
114
115     /* Do we want anoying popups or not */
116     b_notificationEnabled = var_InheritBool( p_intf, "qt-notification" );
117
118     /* Set the other interface settings */
119     settings = getSettings();
120     settings->beginGroup( "MainWindow" );
121
122     /* */
123     b_plDocked = getSettings()->value( "pl-dock-status", true ).toBool();
124
125     settings->endGroup( );
126
127     /**************
128      * Status Bar *
129      **************/
130     createStatusBar();
131
132     /**************************
133      *  UI and Widgets design
134      **************************/
135     setVLCWindowsTitle();
136
137     /************
138      * Menu Bar *
139      ************/
140     QVLCMenu::createMenuBar( this, p_intf );
141     CONNECT( THEMIM->getIM(), voutListChanged( vout_thread_t **, int ),
142              this, destroyPopupMenu() );
143
144     createMainWidget( settings );
145     /*********************************
146      * Create the Systray Management *
147      *********************************/
148     initSystray();
149
150     /********************
151      * Input Manager    *
152      ********************/
153     MainInputManager::getInstance( p_intf );
154
155 #ifdef WIN32
156     himl = NULL;
157     p_taskbl = NULL;
158     taskbar_wmsg = RegisterWindowMessage("TaskbarButtonCreated");
159 #endif
160
161     /************************************************************
162      * Connect the input manager to the GUI elements it manages *
163      ************************************************************/
164     /**
165      * Connects on nameChanged()
166      * Those connects are different because options can impeach them to trigger.
167      **/
168     /* Main Interface statusbar */
169     CONNECT( THEMIM->getIM(), nameChanged( const QString& ),
170              this, setName( const QString& ) );
171     /* and systray */
172 #ifndef HAVE_MAEMO
173     if( sysTray )
174     {
175         CONNECT( THEMIM->getIM(), nameChanged( const QString& ),
176                  this, updateSystrayTooltipName( const QString& ) );
177     }
178 #endif
179     /* and title of the Main Interface*/
180     if( var_InheritBool( p_intf, "qt-name-in-title" ) )
181     {
182         CONNECT( THEMIM->getIM(), nameChanged( const QString& ),
183                  this, setVLCWindowsTitle( const QString& ) );
184     }
185
186     /**
187      * CONNECTS on PLAY_STATUS
188      **/
189     /* Status on the systray */
190 #ifndef HAVE_MAEMO
191     if( sysTray )
192     {
193         CONNECT( THEMIM->getIM(), statusChanged( int ),
194                  this, updateSystrayTooltipStatus( int ) );
195     }
196 #endif
197
198     /* END CONNECTS ON IM */
199
200     /* VideoWidget connects for asynchronous calls */
201     b_videoFullScreen = false;
202     b_videoOnTop = false;
203     connect( this, SIGNAL(askGetVideo(WId*,int*,int*,unsigned*,unsigned *)),
204              this, SLOT(getVideoSlot(WId*,int*,int*,unsigned*,unsigned*)),
205              Qt::BlockingQueuedConnection );
206     connect( this, SIGNAL(askReleaseVideo( void )),
207              this, SLOT(releaseVideoSlot( void )),
208              Qt::BlockingQueuedConnection );
209     CONNECT( this, askVideoOnTop(bool), this, setVideoOnTop(bool));
210
211     if( videoWidget )
212     {
213         if( b_autoresize )
214         {
215             CONNECT( this, askVideoToResize( unsigned int, unsigned int ),
216                      this, setVideoSize( unsigned int, unsigned int ) );
217             CONNECT( videoWidget, sizeChanged( int, int ),
218                      this, resizeStack( int,  int ) );
219         }
220         CONNECT( this, askVideoSetFullScreen( bool ),
221                  this, setVideoFullScreen( bool ) );
222         CONNECT( videoWidget, keyPressed( QKeyEvent * ),
223                  this, handleKeyPress( QKeyEvent * ) );
224     }
225
226     CONNECT( THEDP, toolBarConfUpdated(), this, recreateToolbars() );
227
228     /** END of CONNECTS**/
229
230
231     /************
232      * Callbacks
233      ************/
234     var_AddCallback( p_intf->p_libvlc, "intf-show", IntfShowCB, p_intf );
235
236     /* Register callback for the intf-popupmenu variable */
237     var_AddCallback( p_intf->p_libvlc, "intf-popupmenu", PopupMenuCB, p_intf );
238
239     /* Playlist */
240     int i_plVis = settings->value( "MainWindow/playlist-visible", false ).toBool();
241
242     if( i_plVis ) togglePlaylist();
243
244     /**** FINAL SIZING and placement of interface */
245     settings->beginGroup( "MainWindow" );
246     QVLCTools::restoreWidgetPosition( settings, this, QSize(400, 100) );
247     settings->endGroup();
248
249     /* Final sizing and showing */
250     setVisible( !b_hideAfterCreation );
251
252     setMinimumWidth( __MAX( controls->sizeHint().width(),
253                             menuBar()->sizeHint().width() ) + 30 );
254
255     /* Switch to minimal view if needed, must be called after the show() */
256     if( i_visualmode )
257         toggleMinimalView( true );
258 }
259
260 MainInterface::~MainInterface()
261 {
262     /* Unsure we hide the videoWidget before destroying it */
263     if( stackCentralOldWidget == videoWidget )
264         showTab( bgWidget );
265
266 #ifdef WIN32
267     if( himl )
268         ImageList_Destroy( himl );
269     if(p_taskbl)
270         p_taskbl->vt->Release(p_taskbl);
271     CoUninitialize();
272 #endif
273
274     /* Be sure to kill the actionsManager... Only used in the MI and control */
275     ActionsManager::killInstance();
276
277     /* Idem */
278     ExtensionsManager::killInstance();
279
280     /* Delete the FSC controller */
281     delete fullscreenControls;
282
283     /* Save states */
284     settings->beginGroup( "MainWindow" );
285
286     settings->setValue( "pl-dock-status", b_plDocked );
287     /* Save playlist state */
288     if( playlistWidget )
289     {
290         settings->setValue( "playlist-visible",
291                             isPlDocked() ?
292                             playlistVisible :
293                             playlistWidget->isVisible() /* FIXME */ );
294     }
295
296     settings->setValue( "adv-controls",
297                         getControlsVisibilityStatus() & CONTROLS_ADVANCED );
298
299     /* Save the stackCentralW sizes */
300     settings->setValue( "bgSize", stackWidgetsSizes[bgWidget] );
301     settings->setValue( "playlistSize", stackWidgetsSizes[playlistWidget] );
302
303     /* Save this size */
304     QVLCTools::saveWidgetPosition(settings, this);
305
306     settings->endGroup();
307
308     /* Save undocked playlist size */
309     if( playlistWidget && !isPlDocked() )
310         QVLCTools::saveWidgetPosition( p_intf, "Playlist", playlistWidget );
311
312     delete playlistWidget;
313
314     delete statusBar();
315
316     /* Unregister callbacks */
317     var_DelCallback( p_intf->p_libvlc, "intf-show", IntfShowCB, p_intf );
318     var_DelCallback( p_intf->p_libvlc, "intf-popupmenu", PopupMenuCB, p_intf );
319
320     p_intf->p_sys->p_mi = NULL;
321 }
322
323 /*****************************
324  *   Main UI handling        *
325  *****************************/
326 void MainInterface::recreateToolbars()
327 {
328     bool b_adv = getControlsVisibilityStatus() & CONTROLS_ADVANCED;
329
330     settings->beginGroup( "MainWindow" );
331     delete controls;
332     delete inputC;
333
334     controls = new ControlsWidget( p_intf, b_adv, this );
335     inputC = new InputControlsWidget( p_intf, this );
336
337     if( fullscreenControls )
338     {
339         delete fullscreenControls;
340         fullscreenControls = new FullscreenControllerWidget( p_intf, this );
341         CONNECT( fullscreenControls, keyPressed( QKeyEvent * ),
342                  this, handleKeyPress( QKeyEvent * ) );
343     }
344     mainLayout->insertWidget( 2, inputC );
345     mainLayout->insertWidget( settings->value( "ToolbarPos", 0 ).toInt() ? 0: 3,
346                               controls );
347     settings->endGroup();
348 }
349
350 void MainInterface::createMainWidget( QSettings *settings )
351 {
352     /* Create the main Widget and the mainLayout */
353     QWidget *main = new QWidget;
354     setCentralWidget( main );
355     mainLayout = new QVBoxLayout( main );
356     main->setContentsMargins( 0, 0, 0, 0 );
357     mainLayout->setSpacing( 0 ); mainLayout->setMargin( 0 );
358
359     /* */
360     stackCentralW = new QVLCStackedWidget( main );
361
362     /* Bg Cone */
363     bgWidget = new BackgroundWidget( p_intf );
364     stackCentralW->addWidget( bgWidget );
365
366     /* And video Outputs */
367     if( b_videoEmbedded )
368     {
369         videoWidget = new VideoWidget( p_intf );
370         stackCentralW->addWidget( videoWidget );
371     }
372     mainLayout->insertWidget( 1, stackCentralW );
373
374     settings->beginGroup( "MainWindow" );
375     stackWidgetsSizes[bgWidget] = settings->value( "bgSize", QSize( 400, 0 ) ).toSize();
376     /* Resize even if no-auto-resize, because we are at creation */
377     resizeStack( stackWidgetsSizes[bgWidget].width(), stackWidgetsSizes[bgWidget].height() );
378
379     /* Create the CONTROLS Widget */
380     controls = new ControlsWidget( p_intf,
381                    settings->value( "adv-controls", false ).toBool(), this );
382     inputC = new InputControlsWidget( p_intf, this );
383
384     mainLayout->insertWidget( 2, inputC );
385     mainLayout->insertWidget( settings->value( "ToolbarPos", 0 ).toInt() ? 0: 3,
386                               controls );
387
388     /* Visualisation, disabled for now, they SUCK */
389     #if 0
390     visualSelector = new VisualSelector( p_intf );
391     mainLayout->insertWidget( 0, visualSelector );
392     visualSelector->hide();
393     #endif
394
395     settings->endGroup();
396
397     /* Enable the popup menu in the MI */
398     main->setContextMenuPolicy( Qt::CustomContextMenu );
399     CONNECT( main, customContextMenuRequested( const QPoint& ),
400              this, popupMenu( const QPoint& ) );
401
402     if ( depth() > 8 ) /* 8bit depth has too many issues with opacity */
403         /* Create the FULLSCREEN CONTROLS Widget */
404         if( var_InheritBool( p_intf, "qt-fs-controller" ) )
405         {
406             fullscreenControls = new FullscreenControllerWidget( p_intf, this );
407             CONNECT( fullscreenControls, keyPressed( QKeyEvent * ),
408                      this, handleKeyPress( QKeyEvent * ) );
409         }
410 }
411
412 inline void MainInterface::initSystray()
413 {
414 #ifndef HAVE_MAEMO
415     bool b_systrayAvailable = QSystemTrayIcon::isSystemTrayAvailable();
416     bool b_systrayWanted = var_InheritBool( p_intf, "qt-system-tray" );
417
418     if( var_InheritBool( p_intf, "qt-start-minimized") )
419     {
420         if( b_systrayAvailable )
421         {
422             b_systrayWanted = true;
423             b_hideAfterCreation = true;
424         }
425         else
426             msg_Err( p_intf, "cannot start minimized without system tray bar" );
427     }
428
429     if( b_systrayAvailable && b_systrayWanted )
430         createSystray();
431 #endif
432 }
433
434 inline void MainInterface::createStatusBar()
435 {
436     /****************
437      *  Status Bar  *
438      ****************/
439     /* Widgets Creation*/
440     QStatusBar *statusBarr = statusBar();
441
442     TimeLabel *timeLabel = new TimeLabel( p_intf );
443     nameLabel = new QLabel( this );
444     nameLabel->setTextInteractionFlags( Qt::TextSelectableByMouse
445                                       | Qt::TextSelectableByKeyboard );
446     SpeedLabel *speedLabel = new SpeedLabel( p_intf, this );
447
448     /* Styling those labels */
449     timeLabel->setFrameStyle( QFrame::Sunken | QFrame::Panel );
450     speedLabel->setFrameStyle( QFrame::Sunken | QFrame::Panel );
451     nameLabel->setFrameStyle( QFrame::Sunken | QFrame::StyledPanel);
452     timeLabel->setStyleSheet(
453             "QLabel:hover { background-color: rgba(255, 255, 255, 50%) }" );
454     speedLabel->setStyleSheet(
455             "QLabel:hover { background-color: rgba(255, 255, 255, 50%) }" );
456
457     /* and adding those */
458     statusBarr->addWidget( nameLabel, 8 );
459     statusBarr->addPermanentWidget( speedLabel, 0 );
460     statusBarr->addPermanentWidget( timeLabel, 0 );
461
462     /* timeLabel behaviour:
463        - double clicking opens the goto time dialog
464        - right-clicking and clicking just toggle between remaining and
465          elapsed time.*/
466     CONNECT( timeLabel, timeLabelDoubleClicked(), THEDP, gotoTimeDialog() );
467
468     CONNECT( THEMIM->getIM(), encryptionChanged( bool ),
469              this, showCryptedLabel( bool ) );
470
471     CONNECT( THEMIM->getIM(), seekRequested( float ),
472              timeLabel, setDisplayPosition( float ) );
473 }
474
475 /**********************************************************************
476  * Handling of sizing of the components
477  **********************************************************************/
478
479 void MainInterface::debug()
480 {
481 #ifdef DEBUG_INTF
482     msg_Dbg( p_intf, "size: %i - %i", size().height(), size().width() );
483     msg_Dbg( p_intf, "sizeHint: %i - %i", sizeHint().height(), sizeHint().width() );
484     msg_Dbg( p_intf, "minimumsize: %i - %i", minimumSize().height(), minimumSize().width() );
485
486     msg_Dbg( p_intf, "Stack size: %i - %i", stackCentralW->size().height(), stackCentralW->size().width() );
487     msg_Dbg( p_intf, "Stack sizeHint: %i - %i", stackCentralW->sizeHint().height(), stackCentralW->sizeHint().width() );
488     msg_Dbg( p_intf, "Central size: %i - %i", centralWidget()->size().height(), centralWidget()->size().width() );
489 #endif
490 }
491
492 inline void MainInterface::showVideo() { showTab( videoWidget ); }
493 inline void MainInterface::restoreStackOldWidget()
494             { showTab( stackCentralOldWidget ); }
495
496 inline void MainInterface::showTab( QWidget *widget )
497 {
498 #ifdef DEBUG_INTF
499     msg_Warn( p_intf, "Old stackCentralOldWidget %i", stackCentralW->indexOf( stackCentralOldWidget ) );
500 #endif
501
502     stackCentralOldWidget = stackCentralW->currentWidget();
503     stackWidgetsSizes[stackCentralOldWidget] = stackCentralW->size();
504
505     stackCentralW->setCurrentWidget( widget );
506     if( b_autoresize )
507         resizeStack( stackWidgetsSizes[widget].width(), stackWidgetsSizes[widget].height() );
508
509 #ifdef DEBUG_INTF
510     msg_Warn( p_intf, "State change %i",  stackCentralW->currentIndex() );
511     msg_Warn( p_intf, "New stackCentralOldWidget %i", stackCentralW->indexOf( stackCentralOldWidget ) );
512 #endif
513 }
514
515 void MainInterface::destroyPopupMenu()
516 {
517     QVLCMenu::PopupMenu( p_intf, false );
518 }
519
520 void MainInterface::popupMenu( const QPoint &p )
521 {
522     QVLCMenu::PopupMenu( p_intf, true );
523 }
524
525 void MainInterface::toggleFSC()
526 {
527    if( !fullscreenControls ) return;
528
529    IMEvent *eShow = new IMEvent( FullscreenControlToggle_Type, 0 );
530    QApplication::postEvent( fullscreenControls, eShow );
531 }
532
533 /****************************************************************************
534  * Video Handling
535  ****************************************************************************/
536
537 /**
538  * NOTE:
539  * You must not change the state of this object or other Qt4 UI objects,
540  * from the video output thread - only from the Qt4 UI main loop thread.
541  * All window provider queries must be handled through signals or events.
542  * That's why we have all those emit statements...
543  */
544 WId MainInterface::getVideo( int *pi_x, int *pi_y,
545                              unsigned int *pi_width, unsigned int *pi_height )
546 {
547     if( !videoWidget )
548         return 0;
549
550     /* This is a blocking call signal. Results are returned through pointers.
551      * Beware of deadlocks! */
552     WId id;
553     emit askGetVideo( &id, pi_x, pi_y, pi_width, pi_height );
554     return id;
555 }
556
557 void MainInterface::getVideoSlot( WId *p_id, int *pi_x, int *pi_y,
558                                   unsigned *pi_width, unsigned *pi_height )
559 {
560     /* Request the videoWidget */
561     WId ret = videoWidget->request( pi_x, pi_y,
562                                     pi_width, pi_height, !b_autoresize );
563     *p_id = ret;
564     if( ret ) /* The videoWidget is available */
565     {
566         /* Consider the video active now */
567         showVideo();
568
569         /* Ask videoWidget to resize correctly, if we are in normal mode */
570         if( !isFullScreen() && !isMaximized() && b_autoresize )
571             videoWidget->SetSizing( *pi_width, *pi_height );
572     }
573 }
574
575 /* Asynchronous call from the WindowClose function */
576 void MainInterface::releaseVideo( void )
577 {
578     emit askReleaseVideo();
579 }
580
581 /* Function that is CONNECTED to the previous emit */
582 void MainInterface::releaseVideoSlot( void )
583 {
584     videoWidget->release();
585     setVideoOnTop( false );
586
587     if( stackCentralW->currentWidget() == videoWidget )
588         restoreStackOldWidget();
589
590     /* We don't want to have a blank video to popup */
591     stackCentralOldWidget = bgWidget;
592 }
593
594 void MainInterface::setVideoSize( unsigned int w, unsigned int h )
595 {
596     if( isFullScreen() || isMaximized() )
597         showNormal();
598     videoWidget->SetSizing( w, h );
599 }
600
601 void MainInterface::setVideoFullScreen( bool fs )
602 {
603     b_videoFullScreen = fs;
604     /* refresh main interface on-top status if needed */
605     setVideoOnTop( b_videoOnTop );
606     videoWidget->SetFullScreen( fs, b_videoOnTop );
607 }
608
609 /* Slot to change the video always-on-top flag.
610  * Emit askVideoOnTop() to invoke this from other thread. */
611 void MainInterface::setVideoOnTop( bool on_top )
612 {
613     b_videoOnTop = on_top;
614     /* The main interface is not always-on-top if it does not contain
615      * the video (which is to say in fullscreen mode). */
616     if( b_videoFullScreen )
617         on_top = false;
618
619     Qt::WindowFlags oldflags = windowFlags(), newflags;
620
621     if( on_top )
622         newflags = oldflags | Qt::WindowStaysOnTopHint;
623     else
624         newflags = oldflags & ~Qt::WindowStaysOnTopHint;
625
626     if( newflags != oldflags )
627     {
628         setWindowFlags( newflags );
629         show(); /* necessary to apply window flags */
630     }
631 }
632
633 /* Asynchronous call from WindowControl function */
634 int MainInterface::controlVideo( int i_query, va_list args )
635 {
636     switch( i_query )
637     {
638     case VOUT_WINDOW_SET_SIZE:
639     {
640         unsigned int i_width  = va_arg( args, unsigned int );
641         unsigned int i_height = va_arg( args, unsigned int );
642
643         emit askVideoToResize( i_width, i_height );
644         return VLC_SUCCESS;
645     }
646     case VOUT_WINDOW_SET_STATE:
647     {
648         unsigned i_arg = va_arg( args, unsigned );
649         unsigned on_top = i_arg & VOUT_WINDOW_STATE_ABOVE;
650
651         emit askVideoOnTop( on_top != 0 );
652         return VLC_SUCCESS;
653     }
654     case VOUT_WINDOW_SET_FULLSCREEN:
655     {
656         bool b_fs = va_arg( args, int );
657
658         emit askVideoSetFullScreen( b_fs );
659         return VLC_SUCCESS;
660     }
661     default:
662         msg_Warn( p_intf, "unsupported control query" );
663         return VLC_EGENERIC;
664     }
665 }
666
667 /*****************************************************************************
668  * Playlist, Visualisation and Menus handling
669  *****************************************************************************/
670 /**
671  * Toggle the playlist widget or dialog
672  **/
673 void MainInterface::createPlaylist()
674 {
675     playlistWidget = new PlaylistWidget( p_intf, this );
676
677     if( b_plDocked )
678     {
679         stackCentralW->addWidget( playlistWidget );
680         stackWidgetsSizes[playlistWidget] = settings->value( "playlistSize", QSize( 500, 250 ) ).toSize();
681     }
682     else
683     {
684 #ifdef WIN32
685         playlistWidget->setParent( NULL );
686 #endif
687         playlistWidget->setWindowFlags( Qt::Window );
688
689         /* This will restore the geometry but will not work for position,
690            because of parenting */
691         QVLCTools::restoreWidgetPosition( p_intf, "Playlist",
692                 playlistWidget, QSize( 600, 300 ) );
693     }
694 }
695
696 void MainInterface::togglePlaylist()
697 {
698     if( !playlistWidget )
699     {
700         createPlaylist();
701     }
702
703     if( b_plDocked )
704     {
705         /* Playlist is not visible, show it */
706         if( stackCentralW->currentWidget() != playlistWidget )
707         {
708             showTab( playlistWidget );
709         }
710         else /* Hide it! */
711         {
712             restoreStackOldWidget();
713         }
714         playlistVisible = ( stackCentralW->currentWidget() == playlistWidget );
715     }
716     else
717     {
718 #ifdef WIN32
719         playlistWidget->setParent( NULL );
720 #endif
721         playlistWidget->setWindowFlags( Qt::Window );
722         playlistVisible = !playlistVisible;
723         playlistWidget->setVisible( playlistVisible );
724     }
725     debug();
726 }
727
728 void MainInterface::dockPlaylist( bool p_docked )
729 {
730     if( b_plDocked == p_docked ) return;
731     b_plDocked = p_docked;
732
733     if( !playlistWidget ) return; /* Playlist wasn't created yet */
734     if( !p_docked )
735     {
736         stackCentralW->removeWidget( playlistWidget );
737 #ifdef WIN32
738         playlistWidget->setParent( NULL );
739 #endif
740         playlistWidget->setWindowFlags( Qt::Window );
741         QVLCTools::restoreWidgetPosition( p_intf, "Playlist",
742                 playlistWidget, QSize( 600, 300 ) );
743         playlistWidget->show();
744         restoreStackOldWidget();
745     }
746     else
747     {
748         QVLCTools::saveWidgetPosition( p_intf, "Playlist", playlistWidget );
749         playlistWidget->setWindowFlags( Qt::Widget ); // Probably a Qt bug here
750         // It would be logical that QStackWidget::addWidget reset the flags...
751         stackCentralW->addWidget( playlistWidget );
752         showTab( playlistWidget );
753     }
754     playlistVisible = true;
755 }
756
757 /*
758   If b_switch is false, then we are normalView
759  */
760 void MainInterface::toggleMinimalView( bool b_switch )
761 {
762     if( i_visualmode == 0 && b_autoresize ) /* Normal mode */
763     {
764         if( stackCentralW->currentWidget() == bgWidget )
765         {
766             if( stackCentralW->height() < 16 )
767             {
768                 resizeStack( stackCentralW->width(), 100 );
769             }
770         }
771     }
772
773     menuBar()->setVisible( !b_switch );
774     controls->setVisible( !b_switch );
775     statusBar()->setVisible( !b_switch );
776     inputC->setVisible( !b_switch );
777
778     emit minimalViewToggled( b_switch );
779 }
780
781 /* toggling advanced controls buttons */
782 void MainInterface::toggleAdvancedButtons()
783 {
784     controls->toggleAdvanced();
785 //    if( fullscreenControls ) fullscreenControls->toggleAdvanced();
786 }
787
788 /* Get the visibility status of the controls (hidden or not, advanced or not) */
789 int MainInterface::getControlsVisibilityStatus()
790 {
791     if( !controls ) return 0;
792     return( (controls->isVisible() ? CONTROLS_VISIBLE : CONTROLS_HIDDEN )
793                 + CONTROLS_ADVANCED * controls->b_advancedVisible );
794 }
795
796 #if 0
797 void MainInterface::visual()
798 {
799     if( !VISIBLE( visualSelector) )
800     {
801         visualSelector->show();
802         if( !THEMIM->getIM()->hasVideo() )
803         {
804             /* Show the background widget */
805         }
806         visualSelectorEnabled = true;
807     }
808     else
809     {
810         /* Stop any currently running visualization */
811         visualSelector->hide();
812         visualSelectorEnabled = false;
813     }
814 }
815 #endif
816
817 /************************************************************************
818  * Other stuff
819  ************************************************************************/
820 void MainInterface::setName( const QString& name )
821 {
822     input_name = name; /* store it for the QSystray use */
823     /* Display it in the status bar, but also as a Tooltip in case it doesn't
824        fit in the label */
825     nameLabel->setText( " " + name + " " );
826     nameLabel->setToolTip( " " + name +" " );
827 }
828
829 /**
830  * Give the decorations of the Main Window a correct Name.
831  * If nothing is given, set it to VLC...
832  **/
833 void MainInterface::setVLCWindowsTitle( const QString& aTitle )
834 {
835     if( aTitle.isEmpty() )
836     {
837         setWindowTitle( qtr( "VLC media player" ) );
838     }
839     else
840     {
841         setWindowTitle( aTitle + " - " + qtr( "VLC media player" ) );
842     }
843 }
844
845 void MainInterface::showCryptedLabel( bool b_show )
846 {
847     if( cryptedLabel == NULL )
848     {
849         cryptedLabel = new QLabel;
850         // The lock icon is not the right one for DRM protection/scrambled.
851         //cryptedLabel->setPixmap( QPixmap( ":/lock" ) );
852         cryptedLabel->setText( "DRM" );
853         statusBar()->addWidget( cryptedLabel );
854     }
855
856     cryptedLabel->setVisible( b_show );
857 }
858
859 void MainInterface::showBuffering( float f_cache )
860 {
861     QString amount = QString("Buffering: %1%").arg( (int)(100*f_cache) );
862     statusBar()->showMessage( amount, 1000 );
863 }
864
865 /*****************************************************************************
866  * Systray Icon and Systray Menu
867  *****************************************************************************/
868 #ifndef HAVE_MAEMO
869 /**
870  * Create a SystemTray icon and a menu that would go with it.
871  * Connects to a click handler on the icon.
872  **/
873 void MainInterface::createSystray()
874 {
875     QIcon iconVLC;
876     if( QDate::currentDate().dayOfYear() >= 354 )
877         iconVLC =  QIcon( ":/logo/vlc128-christmas.png" );
878     else
879         iconVLC =  QIcon( ":/logo/vlc128.png" );
880     sysTray = new QSystemTrayIcon( iconVLC, this );
881     sysTray->setToolTip( qtr( "VLC media player" ));
882
883     systrayMenu = new QMenu( qtr( "VLC media player" ), this );
884     systrayMenu->setIcon( iconVLC );
885
886     QVLCMenu::updateSystrayMenu( this, p_intf, true );
887     sysTray->show();
888
889     CONNECT( sysTray, activated( QSystemTrayIcon::ActivationReason ),
890             this, handleSystrayClick( QSystemTrayIcon::ActivationReason ) );
891 }
892
893 /**
894  * Updates the Systray Icon's menu and toggle the main interface
895  */
896 void MainInterface::toggleUpdateSystrayMenu()
897 {
898     /* If hidden, show it */
899     if( isHidden() )
900     {
901         show();
902         activateWindow();
903     }
904     else if( isMinimized() )
905     {
906         /* Minimized */
907         showNormal();
908         activateWindow();
909     }
910     else
911     {
912         /* Visible (possibly under other windows) */
913 #ifdef WIN32
914         /* check if any visible window is above vlc in the z-order,
915          * but ignore the ones always on top
916          * and the ones which can't be activated */
917         WINDOWINFO wi;
918         HWND hwnd;
919         wi.cbSize = sizeof( WINDOWINFO );
920         for( hwnd = GetNextWindow( internalWinId(), GW_HWNDPREV );
921                 hwnd && ( !IsWindowVisible( hwnd ) ||
922                     ( GetWindowInfo( hwnd, &wi ) &&
923                       (wi.dwExStyle&WS_EX_NOACTIVATE) ) );
924                 hwnd = GetNextWindow( hwnd, GW_HWNDPREV ) );
925             if( !hwnd || !GetWindowInfo( hwnd, &wi ) ||
926                 (wi.dwExStyle&WS_EX_TOPMOST) )
927             {
928                 hide();
929             }
930             else
931             {
932                 activateWindow();
933             }
934 #else
935         hide();
936 #endif
937     }
938     QVLCMenu::updateSystrayMenu( this, p_intf );
939 }
940
941 void MainInterface::handleSystrayClick(
942                                     QSystemTrayIcon::ActivationReason reason )
943 {
944     switch( reason )
945     {
946         case QSystemTrayIcon::Trigger:
947         case QSystemTrayIcon::DoubleClick:
948             toggleUpdateSystrayMenu();
949             break;
950         case QSystemTrayIcon::MiddleClick:
951             sysTray->showMessage( qtr( "VLC media player" ),
952                     qtr( "Control menu for the player" ),
953                     QSystemTrayIcon::Information, 3000 );
954             break;
955         default:
956             break;
957     }
958 }
959
960 /**
961  * Updates the name of the systray Icon tooltip.
962  * Doesn't check if the systray exists, check before you call it.
963  **/
964 void MainInterface::updateSystrayTooltipName( const QString& name )
965 {
966     if( name.isEmpty() )
967     {
968         sysTray->setToolTip( qtr( "VLC media player" ) );
969     }
970     else
971     {
972         sysTray->setToolTip( name );
973         if( b_notificationEnabled && ( isHidden() || isMinimized() ) )
974         {
975             sysTray->showMessage( qtr( "VLC media player" ), name,
976                     QSystemTrayIcon::NoIcon, 3000 );
977         }
978     }
979
980     QVLCMenu::updateSystrayMenu( this, p_intf );
981 }
982
983 /**
984  * Updates the status of the systray Icon tooltip.
985  * Doesn't check if the systray exists, check before you call it.
986  **/
987 void MainInterface::updateSystrayTooltipStatus( int i_status )
988 {
989     switch( i_status )
990     {
991     case PLAYING_S:
992         sysTray->setToolTip( input_name );
993         break;
994     case PAUSE_S:
995         sysTray->setToolTip( input_name + " - " + qtr( "Paused") );
996         break;
997     default:
998         sysTray->setToolTip( qtr( "VLC media player" ) );
999         break;
1000     }
1001     QVLCMenu::updateSystrayMenu( this, p_intf );
1002 }
1003 #endif
1004
1005 /************************************************************************
1006  * D&D Events
1007  ************************************************************************/
1008 void MainInterface::dropEvent(QDropEvent *event)
1009 {
1010     dropEventPlay( event, true );
1011 }
1012
1013 void MainInterface::dropEventPlay( QDropEvent *event, bool b_play )
1014 {
1015     event->setDropAction( Qt::CopyAction );
1016     if( !event->possibleActions() & Qt::CopyAction )
1017         return;
1018
1019     const QMimeData *mimeData = event->mimeData();
1020
1021     /* D&D of a subtitles file, add it on the fly */
1022     if( mimeData->urls().size() == 1 && THEMIM->getIM()->hasInput() )
1023     {
1024         if( !input_AddSubtitle( THEMIM->getInput(),
1025                  qtu( toNativeSeparators( mimeData->urls()[0].toLocalFile() ) ),
1026                  true ) )
1027         {
1028             event->accept();
1029             return;
1030         }
1031     }
1032
1033     bool first = b_play;
1034     foreach( const QUrl &url, mimeData->urls() )
1035     {
1036         QString s = toNativeSeparators( url.toLocalFile() );
1037
1038         if( s.length() > 0 ) {
1039             char* psz_uri = make_URI( qtu(s) );
1040             playlist_Add( THEPL, psz_uri, NULL,
1041                           PLAYLIST_APPEND | (first ? PLAYLIST_GO: PLAYLIST_PREPARSE),
1042                           PLAYLIST_END, true, pl_Unlocked );
1043             free( psz_uri );
1044             first = false;
1045             RecentsMRL::getInstance( p_intf )->addRecent( s );
1046         }
1047     }
1048     event->accept();
1049 }
1050 void MainInterface::dragEnterEvent(QDragEnterEvent *event)
1051 {
1052      event->acceptProposedAction();
1053 }
1054 void MainInterface::dragMoveEvent(QDragMoveEvent *event)
1055 {
1056      event->acceptProposedAction();
1057 }
1058 void MainInterface::dragLeaveEvent(QDragLeaveEvent *event)
1059 {
1060      event->accept();
1061 }
1062
1063 /************************************************************************
1064  * Events stuff
1065  ************************************************************************/
1066 void MainInterface::keyPressEvent( QKeyEvent *e )
1067 {
1068     handleKeyPress( e );
1069 }
1070
1071 void MainInterface::handleKeyPress( QKeyEvent *e )
1072 {
1073     if( ( e->modifiers() &  Qt::ControlModifier ) && ( e->key() == Qt::Key_H )
1074           && !menuBar()->isVisible() )
1075     {
1076         toggleMinimalView( false );
1077         e->accept();
1078     }
1079
1080     int i_vlck = qtEventToVLCKey( e );
1081     if( i_vlck > 0 )
1082     {
1083         var_SetInteger( p_intf->p_libvlc, "key-pressed", i_vlck );
1084         e->accept();
1085     }
1086     else
1087         e->ignore();
1088 }
1089
1090 void MainInterface::wheelEvent( QWheelEvent *e )
1091 {
1092     int i_vlckey = qtWheelEventToVLCKey( e );
1093     var_SetInteger( p_intf->p_libvlc, "key-pressed", i_vlckey );
1094     e->accept();
1095 }
1096
1097 void MainInterface::closeEvent( QCloseEvent *e )
1098 {
1099     e->accept();
1100     hide();
1101     THEDP->quit();
1102 }
1103
1104 void MainInterface::toggleFullScreen()
1105 {
1106     if( isFullScreen() )
1107     {
1108         showNormal();
1109         emit fullscreenInterfaceToggled( false );
1110     }
1111     else
1112     {
1113         showFullScreen();
1114         emit fullscreenInterfaceToggled( true );
1115     }
1116 }
1117
1118 /*****************************************************************************
1119  * PopupMenuCB: callback triggered by the intf-popupmenu playlist variable.
1120  *  We don't show the menu directly here because we don't want the
1121  *  caller to block for a too long time.
1122  *****************************************************************************/
1123 static int PopupMenuCB( vlc_object_t *p_this, const char *psz_variable,
1124                         vlc_value_t old_val, vlc_value_t new_val, void *param )
1125 {
1126     intf_thread_t *p_intf = (intf_thread_t *)param;
1127
1128     if( p_intf->pf_show_dialog )
1129     {
1130         p_intf->pf_show_dialog( p_intf, INTF_DIALOG_POPUPMENU,
1131                                 new_val.b_bool, NULL );
1132     }
1133
1134     return VLC_SUCCESS;
1135 }
1136
1137 /*****************************************************************************
1138  * IntfShowCB: callback triggered by the intf-show libvlc variable.
1139  *****************************************************************************/
1140 static int IntfShowCB( vlc_object_t *p_this, const char *psz_variable,
1141                        vlc_value_t old_val, vlc_value_t new_val, void *param )
1142 {
1143     intf_thread_t *p_intf = (intf_thread_t *)param;
1144     p_intf->p_sys->p_mi->toggleFSC();
1145
1146     /* Show event */
1147      return VLC_SUCCESS;
1148 }