]> git.sesse.net Git - vlc/blob - modules/gui/qt4/main_interface.cpp
b34261bc491f229976d356ba7c1ebd5959120644
[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( menuBar(), 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::customEvent( QEvent *event )
441 {
442     if( event->type() == PLUndockEvent_Type )
443     {
444         ui.vboxLayout->removeWidget( playlistWidget );
445         playlistWidget = NULL;
446         playlistEmbeddedFlag = false;
447         doComponentsUpdate();
448         menuBar()->clear();
449         QVLCMenu::createMenuBar( menuBar(), p_intf, false );
450     }
451     else if( event->type() == PLDockEvent_Type )
452     {
453         PlaylistDialog::killInstance();
454         playlistEmbeddedFlag = true;
455         menuBar()->clear();
456         QVLCMenu::createMenuBar( menuBar(), p_intf, true );
457         playlist();
458     }
459 }
460
461 /************************************************************************
462  * Other stuff
463  ************************************************************************/
464 void MainInterface::keyPressEvent( QKeyEvent *e )
465 {
466     int i_vlck = 0;
467     /* Handle modifiers */
468     if( e->modifiers()& Qt::ShiftModifier ) i_vlck |= KEY_MODIFIER_SHIFT;
469     if( e->modifiers()& Qt::AltModifier ) i_vlck |= KEY_MODIFIER_ALT;
470     if( e->modifiers()& Qt::ControlModifier ) i_vlck |= KEY_MODIFIER_CTRL;
471     if( e->modifiers()& Qt::MetaModifier ) i_vlck |= KEY_MODIFIER_META;
472
473     bool found = false;
474     /* Look for some special keys */
475 #define HANDLE( qt, vk ) case Qt::qt : i_vlck |= vk; found = true;break
476     switch( e->key() )
477     {
478         HANDLE( Key_Left, KEY_LEFT );
479         HANDLE( Key_Right, KEY_RIGHT );
480         HANDLE( Key_Up, KEY_UP );
481         HANDLE( Key_Down, KEY_DOWN );
482         HANDLE( Key_Space, KEY_SPACE );
483         HANDLE( Key_Escape, KEY_ESC );
484         HANDLE( Key_Enter, KEY_ENTER );
485         HANDLE( Key_F1, KEY_F1 );
486         HANDLE( Key_F2, KEY_F2 );
487         HANDLE( Key_F3, KEY_F3 );
488         HANDLE( Key_F4, KEY_F4 );
489         HANDLE( Key_F5, KEY_F5 );
490         HANDLE( Key_F6, KEY_F6 );
491         HANDLE( Key_F7, KEY_F7 );
492         HANDLE( Key_F8, KEY_F8 );
493         HANDLE( Key_F9, KEY_F9 );
494         HANDLE( Key_F10, KEY_F10 );
495         HANDLE( Key_F11, KEY_F11 );
496         HANDLE( Key_F12, KEY_F12 );
497         HANDLE( Key_PageUp, KEY_PAGEUP );
498         HANDLE( Key_PageDown, KEY_PAGEDOWN );
499         HANDLE( Key_Home, KEY_HOME );
500         HANDLE( Key_End, KEY_END );
501         HANDLE( Key_Insert, KEY_INSERT );
502         HANDLE( Key_Delete, KEY_DELETE );
503
504     }
505     if( !found )
506     {
507         /* Force lowercase */
508         if( e->key() >= Qt::Key_A && e->key() <= Qt::Key_Z )
509             i_vlck += e->key() + 32;
510         /* Rest of the ascii range */
511         else if( e->key() >= Qt::Key_Space && e->key() <= Qt::Key_AsciiTilde )
512             i_vlck += e->key();
513     }
514     if( i_vlck >= 0 )
515     {
516         var_SetInteger( p_intf->p_libvlc, "key-pressed", i_vlck );
517         e->accept();
518     }
519     else
520         e->ignore();
521 }
522
523 void MainInterface::stop()
524 {
525     playlist_Stop( THEPL );
526 }
527 void MainInterface::play()
528 {
529     if( !THEPL->i_size || !THEPL->i_enabled )
530     {
531         /* The playlist is empty, open a file requester */
532         THEDP->simpleOpenDialog();
533         setStatus( 0 );
534         return;
535     }
536     THEMIM->togglePlayPause();
537 }
538 void MainInterface::prev()
539 {
540     playlist_Prev( THEPL );
541 }
542 void MainInterface::next()
543 {
544     playlist_Next( THEPL );
545 }
546
547 void MainInterface::setDisplay( float pos, int time, int length )
548 {
549     char psz_length[MSTRTIME_MAX_SIZE], psz_time[MSTRTIME_MAX_SIZE];
550     secstotimestr( psz_length, length );
551     secstotimestr( psz_time, time );
552     QString title;
553     title.sprintf( "%s/%s", psz_time, psz_length );
554     timeLabel->setText( " "+title+" " );
555 }
556
557 void MainInterface::setName( QString name )
558 {
559     nameLabel->setText( " " + name+" " );
560 }
561
562 void MainInterface::setStatus( int status )
563 {
564     if( status == 1 ) // Playing
565         ui.playButton->setIcon( QIcon( ":/pixmaps/pause.png" ) );
566     else
567         ui.playButton->setIcon( QIcon( ":/pixmaps/play.png" ) );
568 }
569
570 static bool b_my_volume;
571
572 void MainInterface::updateOnTimer()
573 {
574     if( p_intf->b_die )
575     {
576         QApplication::closeAllWindows();
577         DialogsProvider::killInstance();
578         QApplication::quit();
579     }
580     if( need_components_update )
581     {
582         doComponentsUpdate();
583         need_components_update = false;
584     }
585
586     audio_volume_t i_volume;
587     aout_VolumeGet( p_intf, &i_volume );
588     i_volume = (i_volume *  200 )/ AOUT_VOLUME_MAX ;
589     int i_gauge = ui.volumeSlider->value();
590     b_my_volume = false;
591     if( i_volume - i_gauge > 1 || i_gauge - i_volume > 1 )
592     {
593         b_my_volume = true;
594         ui.volumeSlider->setValue( i_volume );
595         b_my_volume = false;
596     }
597 }
598
599 void MainInterface::closeEvent( QCloseEvent *e )
600 {
601     hide();
602     p_intf->b_die = VLC_TRUE;
603 }
604
605 void MainInterface::updateVolume( int sliderVolume )
606 {
607     if( !b_my_volume )
608     {
609         int i_res = sliderVolume * AOUT_VOLUME_MAX /
610                             (2*ui.volumeSlider->maximum() );
611         aout_VolumeSet( p_intf, i_res );
612     }
613 }
614
615 static int InteractCallback( vlc_object_t *p_this,
616                              const char *psz_var, vlc_value_t old_val,
617                              vlc_value_t new_val, void *param )
618 {
619     intf_dialog_args_t *p_arg = new intf_dialog_args_t;
620     p_arg->p_dialog = (interaction_dialog_t *)(new_val.p_address);
621
622     MainInterface *p_interface = (MainInterface*)param;
623     DialogEvent *event = new DialogEvent( INTF_DIALOG_INTERACTION, 0, p_arg );
624     QApplication::postEvent( THEDP, static_cast<QEvent*>(event) );
625     return VLC_SUCCESS;
626 }