]> git.sesse.net Git - vlc/blob - modules/gui/qt4/main_interface.cpp
9fe65065468cd00575600e295a73857ab9fe5870
[vlc] / modules / gui / qt4 / main_interface.cpp
1 /*****************************************************************************
2  * main_inteface.cpp : Main interface
3  ****************************************************************************
4  * Copyright (C) 2006 the VideoLAN team
5  * $Id$
6  *
7  * Authors: ClĂ©ment Stenac <zorglub@videolan.org>
8  *
9  * This program is free software; you can redistribute it and/or modify
10  * it under the terms of the GNU General Public License as published by
11  * the Free Software Foundation; either version 2 of the License, or
12  * (at your option) any later version.
13  *
14  * This program is distributed in the hope that it will be useful,
15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17  * GNU General Public License for more details.
18  *
19  * You should have received a copy of the GNU General Public License
20  * along with this program; if not, write to the Free Software
21  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA. *****************************************************************************/
22
23 #include "qt4.hpp"
24 #include "main_interface.hpp"
25 #include "input_manager.hpp"
26 #include "util/input_slider.hpp"
27 #include "util/qvlcframe.hpp"
28 #include "dialogs_provider.hpp"
29 #include "components/interface_widgets.hpp"
30 #include "dialogs/playlist.hpp"
31 #include "menus.hpp"
32
33 #include <QMenuBar>
34 #include <QCloseEvent>
35 #include <QPushButton>
36 #include <QStatusBar>
37 #include <QKeyEvent>
38
39 #include <assert.h>
40 #include <vlc_keys.h>
41 #include <vlc/vout.h>
42
43 #ifdef WIN32
44     #define PREF_W 410
45     #define PREF_H 121
46 #else
47     #define PREF_W 450
48     #define PREF_H 125
49 #endif
50
51 #define BUTTON_SET( button, image, tooltip ) ui.button##Button->setText(""); \
52     ui.button##Button->setIcon( QIcon( ":/pixmaps/"#image ) ); \
53     ui.button##Button->setToolTip( tooltip );
54
55 #define VISIBLE(i) (i && i->isVisible())
56
57 #define SET_WIDTH(i,j) i->widgetSize.setWidth(j)
58 #define SET_HEIGHT(i,j) i->widgetSize.setHeight(j)
59 #define SET_WH( i,j,k) i->widgetSize.setWidth(j); i->widgetSize.setHeight(k);
60
61 #define DS(i) i.width(),i.height()
62
63 static int InteractCallback( vlc_object_t *, const char *, vlc_value_t,
64                              vlc_value_t, void *);
65 /* Video handling */
66 static void *DoRequest( intf_thread_t *p_intf, vout_thread_t *p_vout,
67                         int *pi1, int *pi2, unsigned int*pi3,unsigned int*pi4)
68 {
69     return p_intf->p_sys->p_mi->requestVideo( p_vout, pi1, pi2, pi3, pi4 );
70 }
71 static void DoRelease( intf_thread_t *p_intf, void *p_win )
72 {
73     return p_intf->p_sys->p_mi->releaseVideo( p_win );
74 }
75 static int DoControl( intf_thread_t *p_intf, void *p_win, int i_q, va_list a )
76 {
77     return p_intf->p_sys->p_mi->controlVideo( p_win, i_q, a );
78 }
79
80 bool embeddedPlaylistWasActive;
81 bool videoIsActive;
82 QSize savedVideoSize;
83
84 MainInterface::MainInterface( intf_thread_t *_p_intf ) : QVLCMW( _p_intf )
85 {
86     settings = new QSettings( "VideoLAN", "VLC" );
87     settings->beginGroup( "MainWindow" );
88
89     need_components_update = false;
90     bgWidget = NULL; videoWidget = NULL; playlistWidget = NULL;
91     embeddedPlaylistWasActive = videoIsActive = false;
92
93     setWindowTitle( QString::fromUtf8( _("VLC media player") ) );
94     handleMainUi( settings );
95
96     QVLCMenu::createMenuBar( this, p_intf, playlistEmbeddedFlag );
97
98     /* Status bar */
99     timeLabel = new QLabel( 0 );
100     nameLabel = new QLabel( 0 );
101     statusBar()->addWidget( nameLabel, 4 );
102     statusBar()->addPermanentWidget( timeLabel, 1 );
103
104     setFocusPolicy( Qt::StrongFocus );
105
106     /* Init input manager */
107     MainInputManager::getInstance( p_intf );
108     ON_TIMEOUT( updateOnTimer() );
109
110     /* Volume control */
111     CONNECT( ui.volumeSlider, valueChanged(int), this, updateVolume(int) );
112     /* Connect the input manager to the GUI elements it manages */
113     CONNECT( THEMIM->getIM(), positionUpdated( float, int, int ),
114              slider, setPosition( float,int, int ) );
115     CONNECT( THEMIM->getIM(), positionUpdated( float, int, int ),
116              this, setDisplay( float, int, int ) );
117     CONNECT( THEMIM->getIM(), nameChanged( QString ), this,setName( QString ) );
118     CONNECT( THEMIM->getIM(), statusChanged( int ), this, setStatus( int ) );
119     CONNECT( slider, sliderDragged( float ),
120              THEMIM->getIM(), sliderUpdate( float ) );
121
122     var_Create( p_intf, "interaction", VLC_VAR_ADDRESS );
123     var_AddCallback( p_intf, "interaction", InteractCallback, this );
124     p_intf->b_interaction = VLC_TRUE;
125 }
126
127 MainInterface::~MainInterface()
128 {
129     /// \todo Save everything
130     p_intf->b_interaction = VLC_FALSE;
131     var_DelCallback( p_intf, "interaction", InteractCallback, this );
132
133     p_intf->pf_request_window = NULL;
134     p_intf->pf_release_window = NULL;
135     p_intf->pf_control_window = NULL;
136 }
137
138 void MainInterface::handleMainUi( QSettings *settings )
139 {
140     QWidget *main = new QWidget( this );
141     setCentralWidget( main );
142     ui.setupUi( centralWidget() );
143
144     slider = new InputSlider( Qt::Horizontal, NULL );
145     ui.hboxLayout->insertWidget( 0, slider );
146
147     BUTTON_SET( prev, previous.png, qtr( "Previous" ) );
148     BUTTONACT( ui.prevButton, prev() );
149     BUTTON_SET( next, next.png , qtr( "Next" ) );
150     BUTTONACT( ui.nextButton, next() );
151     BUTTON_SET( play, play.png , qtr( "Play" ) );
152     BUTTONACT( ui.playButton, play() );
153     BUTTON_SET( stop, stop.png , qtr( "Stop" )  );
154     BUTTONACT( ui.stopButton, stop() );
155
156     BUTTON_SET( visual,  stop.png, qtr( "Audio visualizations" ) );
157     BUTTONACT( ui.visualButton, visual() );
158
159     /* Volume */
160     ui.volMuteLabel->setPixmap( QPixmap( ":/pixmaps/volume-low.png" ) );
161     ui.volumeSlider->setMaximum( 100 );
162     ui.volMuteLabel->setToolTip( qtr( "Mute" ) );
163     VolumeClickHandler *h = new VolumeClickHandler( p_intf, this );
164     ui.volMuteLabel->installEventFilter(h);
165     ui.volumeSlider->setFocusPolicy( Qt::NoFocus );
166
167     /* Fetch configuration from settings and vlc config */
168     videoEmbeddedFlag = false;
169     if( config_GetInt( p_intf, "embedded-video" ) )
170         videoEmbeddedFlag = true;
171
172     alwaysVideoFlag = false;
173     if( videoEmbeddedFlag && config_GetInt( p_intf, "qt-always-video" ))
174         alwaysVideoFlag = true;
175
176     playlistEmbeddedFlag = true;
177     /// \todo fetch playlist settings
178
179     BUTTON_SET( playlist, volume-low.png, playlistEmbeddedFlag ?
180                                                 qtr( "Show playlist" ) :
181                                                 qtr( "Open playlist" ) );
182     BUTTONACT( ui.playlistButton, playlist() );
183
184     /* Set initial size */
185     resize ( PREF_W, PREF_H );
186
187     addSize = QSize( ui.vboxLayout->margin() * 2, PREF_H );
188
189     visualSelector = new VisualSelector( p_intf );
190     ui.vboxLayout->insertWidget( 0, visualSelector );
191     visualSelector->hide();
192
193     if( alwaysVideoFlag )
194     {
195         bgWidget = new BackgroundWidget( p_intf );
196         bgWidget->widgetSize = settings->value( "backgroundSize",
197                                                 QSize( 200, 200 ) ).toSize();
198         bgWidget->resize( bgWidget->widgetSize );
199         bgWidget->updateGeometry();
200         ui.vboxLayout->insertWidget( 0, bgWidget );
201     }
202
203     if( videoEmbeddedFlag )
204     {
205         videoWidget = new VideoWidget( p_intf );
206         videoWidget->widgetSize = QSize( 1, 1 );
207         videoWidget->resize( videoWidget->widgetSize );
208         ui.vboxLayout->insertWidget( 0, videoWidget );
209
210         p_intf->pf_request_window  = ::DoRequest;
211         p_intf->pf_release_window  = ::DoRelease;
212         p_intf->pf_control_window  = ::DoControl;
213     }
214
215     calculateInterfaceSize();
216     resize( mainSize );
217
218     setMinimumSize( PREF_W, addSize.height() );
219 }
220
221 /**********************************************************************
222  * Handling of the components
223  **********************************************************************/
224 void MainInterface::calculateInterfaceSize()
225 {
226     int width = 0, height = 0;
227     if( VISIBLE( bgWidget ) )
228     {
229         width = bgWidget->widgetSize.width();
230         height = bgWidget->widgetSize.height();
231         assert( !(playlistWidget && playlistWidget->isVisible() ) );
232     }
233     else if( VISIBLE( playlistWidget ) )
234     {
235         width = playlistWidget->widgetSize.width();
236         height = playlistWidget->widgetSize.height();
237         fprintf( stderr, "Have %ix%i playlist\n", width, height );
238     }
239     else if( videoIsActive )
240     {
241         width =  videoWidget->widgetSize.width() ;
242         height = videoWidget->widgetSize.height();
243         fprintf( stderr, "Video Size %ix%i\n", DS( videoWidget->widgetSize ) );
244     }
245     else
246     {
247         width = PREF_W - addSize.width();
248         height = PREF_H - addSize.height();
249     }
250     if( VISIBLE( visualSelector ) )
251         height += visualSelector->height();
252
253     fprintf( stderr, "Setting to %ix%i\n",
254                      width + addSize.width() , height + addSize.height() );
255
256     mainSize = QSize( width + addSize.width(), height + addSize.height() );
257 }
258
259 void MainInterface::resizeEvent( QResizeEvent *e )
260 {
261     fprintf( stderr, "Resize event to %ix%i\n", DS( e->size() ) );
262     videoWidget->widgetSize.setWidth(  e->size().width() - addSize.width() );
263     if( videoWidget && videoIsActive && videoWidget->widgetSize.height() > 1 )
264     {
265         SET_WH( videoWidget, e->size().width() - addSize.width(),
266                              e->size().height()  - addSize.height() );
267         videoWidget->updateGeometry();
268         fprintf( stderr, "Video set to %ix%i\n", DS( videoWidget->widgetSize) );
269     }
270     if( VISIBLE( playlistWidget ) )
271     {
272         SET_WH( playlistWidget , e->size().width() - addSize.width(),
273                                  e->size().height() - addSize.height() );
274         playlistWidget->updateGeometry();
275         fprintf( stderr, "PL set to %ix%i\n",DS(playlistWidget->widgetSize ) );
276     }
277 }
278
279 void *MainInterface::requestVideo( vout_thread_t *p_nvout, int *pi_x,
280                                    int *pi_y, unsigned int *pi_width,
281                                    unsigned int *pi_height )
282 {
283     void *ret = videoWidget->request( p_nvout,pi_x, pi_y, pi_width, pi_height );
284     if( ret )
285     {
286         videoIsActive = true;
287         if( VISIBLE( playlistWidget ) )
288         {
289             embeddedPlaylistWasActive = true;
290             playlistWidget->hide();
291         }
292         bool bgWasVisible = false;
293         if( VISIBLE( bgWidget) )
294         {
295             bgWasVisible = true;
296             bgWidget->hide();
297         }
298         if( THEMIM->getIM()->hasVideo() || !bgWasVisible )
299         {
300             videoWidget->widgetSize = QSize( *pi_width, *pi_height );
301         }
302         else /* Background widget available, use its size */
303         {
304             /* Ok, our visualizations are bad, so don't do this for the moment
305              * use the requested size anyway */
306             // videoWidget->widgetSize = bgWidget->widgeTSize;
307             videoWidget->widgetSize = QSize( *pi_width, *pi_height );
308         }
309         videoWidget->updateGeometry(); /// FIXME: Needed ?
310         need_components_update = true;
311     }
312     return ret;
313 }
314
315 void MainInterface::releaseVideo( void *p_win )
316 {
317     videoWidget->release( p_win );
318     videoWidget->widgetSize = QSize( 0, 0 );
319     videoWidget->resize( videoWidget->widgetSize );
320
321     if( embeddedPlaylistWasActive )
322         playlistWidget->show();
323     else if( bgWidget )
324         bgWidget->show();
325
326     videoIsActive = false;
327     need_components_update = true;
328 }
329
330 int MainInterface::controlVideo( void *p_window, int i_query, va_list args )
331 {
332     int i_ret = VLC_EGENERIC;
333     switch( i_query )
334     {
335         case VOUT_GET_SIZE:
336         {
337             unsigned int *pi_width  = va_arg( args, unsigned int * );
338             unsigned int *pi_height = va_arg( args, unsigned int * );
339             *pi_width = videoWidget->widgetSize.width();
340             *pi_height = videoWidget->widgetSize.height();
341             i_ret = VLC_SUCCESS;
342             break;
343         }
344         case VOUT_SET_SIZE:
345         {
346             unsigned int i_width  = va_arg( args, unsigned int );
347             unsigned int i_height = va_arg( args, unsigned int );
348 //          if( !i_width && p_vout ) i_width = p_vout->i_window_width;
349 //          if( !i_height && p_vout ) i_height = p_vout->i_window_height;
350             videoWidget->widgetSize = QSize( i_width, i_height );
351             videoWidget->updateGeometry();
352             need_components_update = true;
353             i_ret = VLC_SUCCESS;
354             break;
355         }
356         case VOUT_SET_STAY_ON_TOP:
357         default:
358             msg_Warn( p_intf, "unsupported control query" );
359             break;
360     }
361     return i_ret;
362 }
363
364 void MainInterface::visual()
365 {
366     if( !VISIBLE( visualSelector) )
367     {
368         visualSelector->show();
369         if( !THEMIM->getIM()->hasVideo() )
370         {
371             /* Show the background widget */
372         }
373     }
374     else
375     {
376         /* Stop any currently running visualization */
377         visualSelector->hide();
378     }
379     doComponentsUpdate();
380 }
381
382 void MainInterface::playlist()
383 {
384     // Toggle the playlist dialog
385     if( !playlistEmbeddedFlag )
386     {
387         if( playlistWidget )
388         {
389             /// \todo Destroy it
390         }
391         THEDP->playlistDialog();
392         return;
393     }
394
395     if( !playlistWidget )
396     {
397         PlaylistDialog::killInstance();
398         playlistWidget = new PlaylistWidget( p_intf );
399         ui.vboxLayout->insertWidget( 0, playlistWidget );
400         playlistWidget->widgetSize = settings->value( "playlistSize",
401                                                QSize( 650, 310 ) ).toSize();
402         playlistWidget->hide();
403     }
404     /// Todo, reset its size ?
405     if( VISIBLE( playlistWidget) )
406     {
407         fprintf( stderr, "hiding playlist\n" );
408         playlistWidget->hide();
409         if( videoIsActive )
410         {
411             videoWidget->widgetSize = savedVideoSize;
412             videoWidget->resize( videoWidget->widgetSize );
413             videoWidget->updateGeometry();
414         }
415     }
416     else
417     {
418         fprintf( stderr, "showing playlist\n" );
419         playlistWidget->show();
420         if( videoIsActive )
421         {
422             savedVideoSize = videoWidget->widgetSize;
423             videoWidget->widgetSize.setHeight( 0 );
424             videoWidget->resize( videoWidget->widgetSize );
425             videoWidget->updateGeometry();
426         }
427         if( VISIBLE( bgWidget ) ) bgWidget->hide();
428     }
429     doComponentsUpdate();
430 }
431
432 /* Video widget cannot do this synchronously as it runs in another thread */
433 /* Well, could it, actually ? Probably dangerous ... */
434 void MainInterface::doComponentsUpdate()
435 {
436     calculateInterfaceSize();
437     resize( mainSize );
438 }
439
440 void MainInterface::undockPlaylist()
441 {
442     if( playlistWidget )
443     {
444         playlistWidget->hide();
445         playlistWidget->deleteLater();
446         ui.vboxLayout->removeWidget( playlistWidget );
447         playlistWidget = NULL;
448         playlistEmbeddedFlag = false;
449
450         menuBar()->clear();
451         QVLCMenu::createMenuBar( this, p_intf, false );
452
453         if( videoIsActive )
454         {
455             videoWidget->widgetSize = savedVideoSize;
456             videoWidget->resize( videoWidget->widgetSize );
457             videoWidget->updateGeometry();
458         }
459
460         doComponentsUpdate();
461         THEDP->playlistDialog();
462     }
463 }
464
465 void MainInterface::customEvent( QEvent *event )
466 {
467     if( event->type() == PLDockEvent_Type )
468     {
469         PlaylistDialog::killInstance();
470         playlistEmbeddedFlag = true;
471         menuBar()->clear();
472         QVLCMenu::createMenuBar(this, p_intf, true );
473         playlist();
474     }
475 }
476
477 /************************************************************************
478  * Other stuff
479  ************************************************************************/
480 void MainInterface::keyPressEvent( QKeyEvent *e )
481 {
482     int i_vlck = 0;
483     /* Handle modifiers */
484     if( e->modifiers()& Qt::ShiftModifier ) i_vlck |= KEY_MODIFIER_SHIFT;
485     if( e->modifiers()& Qt::AltModifier ) i_vlck |= KEY_MODIFIER_ALT;
486     if( e->modifiers()& Qt::ControlModifier ) i_vlck |= KEY_MODIFIER_CTRL;
487     if( e->modifiers()& Qt::MetaModifier ) i_vlck |= KEY_MODIFIER_META;
488
489     bool found = false;
490     /* Look for some special keys */
491 #define HANDLE( qt, vk ) case Qt::qt : i_vlck |= vk; found = true;break
492     switch( e->key() )
493     {
494         HANDLE( Key_Left, KEY_LEFT );
495         HANDLE( Key_Right, KEY_RIGHT );
496         HANDLE( Key_Up, KEY_UP );
497         HANDLE( Key_Down, KEY_DOWN );
498         HANDLE( Key_Space, KEY_SPACE );
499         HANDLE( Key_Escape, KEY_ESC );
500         HANDLE( Key_Enter, KEY_ENTER );
501         HANDLE( Key_F1, KEY_F1 );
502         HANDLE( Key_F2, KEY_F2 );
503         HANDLE( Key_F3, KEY_F3 );
504         HANDLE( Key_F4, KEY_F4 );
505         HANDLE( Key_F5, KEY_F5 );
506         HANDLE( Key_F6, KEY_F6 );
507         HANDLE( Key_F7, KEY_F7 );
508         HANDLE( Key_F8, KEY_F8 );
509         HANDLE( Key_F9, KEY_F9 );
510         HANDLE( Key_F10, KEY_F10 );
511         HANDLE( Key_F11, KEY_F11 );
512         HANDLE( Key_F12, KEY_F12 );
513         HANDLE( Key_PageUp, KEY_PAGEUP );
514         HANDLE( Key_PageDown, KEY_PAGEDOWN );
515         HANDLE( Key_Home, KEY_HOME );
516         HANDLE( Key_End, KEY_END );
517         HANDLE( Key_Insert, KEY_INSERT );
518         HANDLE( Key_Delete, KEY_DELETE );
519
520     }
521     if( !found )
522     {
523         /* Force lowercase */
524         if( e->key() >= Qt::Key_A && e->key() <= Qt::Key_Z )
525             i_vlck += e->key() + 32;
526         /* Rest of the ascii range */
527         else if( e->key() >= Qt::Key_Space && e->key() <= Qt::Key_AsciiTilde )
528             i_vlck += e->key();
529     }
530     if( i_vlck >= 0 )
531     {
532         var_SetInteger( p_intf->p_libvlc, "key-pressed", i_vlck );
533         e->accept();
534     }
535     else
536         e->ignore();
537 }
538
539 void MainInterface::stop()
540 {
541     playlist_Stop( THEPL );
542 }
543 void MainInterface::play()
544 {
545     if( !THEPL->i_size || !THEPL->i_enabled )
546     {
547         /* The playlist is empty, open a file requester */
548         THEDP->simpleOpenDialog();
549         setStatus( 0 );
550         return;
551     }
552     THEMIM->togglePlayPause();
553 }
554 void MainInterface::prev()
555 {
556     playlist_Prev( THEPL );
557 }
558 void MainInterface::next()
559 {
560     playlist_Next( THEPL );
561 }
562
563 void MainInterface::setDisplay( float pos, int time, int length )
564 {
565     char psz_length[MSTRTIME_MAX_SIZE], psz_time[MSTRTIME_MAX_SIZE];
566     secstotimestr( psz_length, length );
567     secstotimestr( psz_time, time );
568     QString title;
569     title.sprintf( "%s/%s", psz_time, psz_length );
570     timeLabel->setText( " "+title+" " );
571 }
572
573 void MainInterface::setName( QString name )
574 {
575     nameLabel->setText( " " + name+" " );
576 }
577
578 void MainInterface::setStatus( int status )
579 {
580     if( status == 1 ) // Playing
581         ui.playButton->setIcon( QIcon( ":/pixmaps/pause.png" ) );
582     else
583         ui.playButton->setIcon( QIcon( ":/pixmaps/play.png" ) );
584 }
585
586 static bool b_my_volume;
587
588 void MainInterface::updateOnTimer()
589 {
590     if( p_intf->b_die )
591     {
592         QApplication::closeAllWindows();
593         DialogsProvider::killInstance();
594         QApplication::quit();
595     }
596     if( need_components_update )
597     {
598         doComponentsUpdate();
599         need_components_update = false;
600     }
601
602     audio_volume_t i_volume;
603     aout_VolumeGet( p_intf, &i_volume );
604     i_volume = (i_volume *  200 )/ AOUT_VOLUME_MAX ;
605     int i_gauge = ui.volumeSlider->value();
606     b_my_volume = false;
607     if( i_volume - i_gauge > 1 || i_gauge - i_volume > 1 )
608     {
609         b_my_volume = true;
610         ui.volumeSlider->setValue( i_volume );
611         b_my_volume = false;
612     }
613 }
614
615 void MainInterface::closeEvent( QCloseEvent *e )
616 {
617     hide();
618     p_intf->b_die = VLC_TRUE;
619 }
620
621 void MainInterface::updateVolume( int sliderVolume )
622 {
623     if( !b_my_volume )
624     {
625         int i_res = sliderVolume * AOUT_VOLUME_MAX /
626                             (2*ui.volumeSlider->maximum() );
627         aout_VolumeSet( p_intf, i_res );
628     }
629 }
630
631 static int InteractCallback( vlc_object_t *p_this,
632                              const char *psz_var, vlc_value_t old_val,
633                              vlc_value_t new_val, void *param )
634 {
635     intf_dialog_args_t *p_arg = new intf_dialog_args_t;
636     p_arg->p_dialog = (interaction_dialog_t *)(new_val.p_address);
637
638     MainInterface *p_interface = (MainInterface*)param;
639     DialogEvent *event = new DialogEvent( INTF_DIALOG_INTERACTION, 0, p_arg );
640     QApplication::postEvent( THEDP, static_cast<QEvent*>(event) );
641     return VLC_SUCCESS;
642 }