]> git.sesse.net Git - vlc/blob - modules/gui/qt4/components/interface_widgets.cpp
Qt4: add XFlush - this does still not work right
[vlc] / modules / gui / qt4 / components / interface_widgets.cpp
1 /*****************************************************************************
2  * interface_widgets.cpp : Custom widgets for the main interface
3  ****************************************************************************
4  * Copyright ( C ) 2006 the VideoLAN team
5  * $Id$
6  *
7  * Authors: Clément Stenac <zorglub@videolan.org>
8  *          Jean-Baptiste Kempf <jb@videolan.org>
9  *          Rafaël Carré <funman@videolanorg>
10  *          Ilkka Ollakka <ileoo@videolan.org>
11  *
12  * This program is free software; you can redistribute it and/or modify
13  * it under the terms of the GNU General Public License as published by
14  * the Free Software Foundation; either version 2 of the License, or
15  * ( at your option ) any later version.
16  *
17  * This program is distributed in the hope that it will be useful,
18  * but WITHOUT ANY WARRANTY; without even the implied warranty of
19  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
20  * GNU General Public License for more details.
21  *
22  * You should have received a copy of the GNU General Public License
23  * along with this program; if not, write to the Free Software
24  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
25  *****************************************************************************/
26
27 #ifdef HAVE_CONFIG_H
28 # include "config.h"
29 #endif
30
31 #include "dialogs_provider.hpp"
32 #include "components/interface_widgets.hpp"
33 #include "main_interface.hpp"
34 #include "input_manager.hpp"
35 #include "menus.hpp"
36 #include "util/input_slider.hpp"
37 #include "util/customwidgets.hpp"
38 #include <vlc_vout.h>
39
40 #include <QLabel>
41 #include <QSpacerItem>
42 #include <QCursor>
43 #include <QPushButton>
44 #include <QToolButton>
45 #include <QHBoxLayout>
46 #include <QMenu>
47 #include <QPalette>
48 #include <QResizeEvent>
49 #include <QDate>
50 #include <QMutexLocker>
51 #ifdef Q_WS_X11
52 # include <X11/Xlib.h>
53 # include <qx11info_x11.h>
54 #endif
55
56 /**********************************************************************
57  * Video Widget. A simple frame on which video is drawn
58  * This class handles resize issues
59  **********************************************************************/
60
61 VideoWidget::VideoWidget( intf_thread_t *_p_i ) : QFrame( NULL ), p_intf( _p_i )
62 {
63     /* Init */
64     vlc_mutex_init( &lock );
65     p_vout = NULL;
66     handleReady = false;
67     hide(); setMinimumSize( 16, 16 );
68     videoSize.rwidth() = -1;
69     videoSize.rheight() = -1;
70
71     /* Black background is more coherent for a Video Widget IMVHO */
72     QPalette plt =  palette();
73     plt.setColor( QPalette::Active, QPalette::Window , Qt::black );
74     plt.setColor( QPalette::Inactive, QPalette::Window , Qt::black );
75     setPalette( plt );
76
77     /* The core can ask through a callback to show the video.
78      * NOTE: We need to block the video output core until the window handle
79      * is ready for use (otherwise an X11 invalid handle failure may occur).
80      * As a side effect, it is illegal to emit askVideoWidgetToShow from
81      * the same thread as the Qt4 thread that owns this. */
82     QObject::connect( this, SIGNAL(askVideoWidgetToShow()), this, SLOT(show()),
83                       Qt::BlockingQueuedConnection );
84
85     /* The core can ask through a callback to resize the video */
86    // CONNECT( this, askResize( int, int ), this, SetSizing( int, int ) );
87 }
88
89 void VideoWidget::paintEvent(QPaintEvent *ev)
90 {
91     QFrame::paintEvent(ev);
92 #ifdef Q_WS_X11
93     XFlush( QX11Info::display() );
94 #endif
95     QMutexLocker locker( &handleLock );
96     handleReady = true;
97     handleWait.wakeAll();
98 }
99
100 VideoWidget::~VideoWidget()
101 {
102     vlc_mutex_lock( &lock );
103     if( p_vout )
104     {
105         if( !p_intf->psz_switch_intf )
106         {
107             if( vout_Control( p_vout, VOUT_CLOSE ) != VLC_SUCCESS )
108                 vout_Control( p_vout, VOUT_REPARENT );
109         }
110         else
111         {
112             if( vout_Control( p_vout, VOUT_REPARENT ) != VLC_SUCCESS )
113                 vout_Control( p_vout, VOUT_CLOSE );
114         }
115     }
116     vlc_mutex_unlock( &lock );
117     vlc_mutex_destroy( &lock );
118 }
119
120 /**
121  * Request the video to avoid the conflicts
122  **/
123 void *VideoWidget::request( vout_thread_t *p_nvout, int *pi_x, int *pi_y,
124                            unsigned int *pi_width, unsigned int *pi_height )
125 {
126     QMutexLocker locker( &handleLock );
127     msg_Dbg( p_intf, "Video was requested %i, %i", *pi_x, *pi_y );
128     emit askVideoWidgetToShow();
129     if( p_vout )
130     {
131         msg_Dbg( p_intf, "embedded video already in use" );
132         return NULL;
133     }
134     p_vout = p_nvout;
135     while( !handleReady )
136     {
137         msg_Dbg( p_intf, "embedded video pending (handle %p)", winId() );
138         handleWait.wait( &handleLock );
139     }
140     msg_Dbg( p_intf, "embedded video ready (handle %p)", winId() );
141     return ( void* )winId();
142 }
143
144 /* Set the Widget to the correct Size */
145 /* Function has to be called by the parent
146    Parent has to care about resizing himself*/
147 void VideoWidget::SetSizing( unsigned int w, unsigned int h )
148 {
149     msg_Dbg( p_intf, "Video is resizing to: %i %i", w, h );
150     videoSize.rwidth() = w;
151     videoSize.rheight() = h;
152     updateGeometry(); // Needed for deinterlace
153 }
154
155 void VideoWidget::release( void *p_win )
156 {
157     msg_Dbg( p_intf, "Video is non needed anymore" );
158     p_vout = NULL;
159     videoSize.rwidth() = 0;
160     videoSize.rheight() = 0;
161     hide();
162     updateGeometry(); // Needed for deinterlace
163 }
164
165 QSize VideoWidget::sizeHint() const
166 {
167     return videoSize;
168 }
169
170 /**********************************************************************
171  * Background Widget. Show a simple image background. Currently,
172  * it's album art if present or cone.
173  **********************************************************************/
174 #define ICON_SIZE 128
175 #define MAX_BG_SIZE 400
176 #define MIN_BG_SIZE 64
177
178 BackgroundWidget::BackgroundWidget( intf_thread_t *_p_i )
179                  :QWidget( NULL ), p_intf( _p_i )
180 {
181     /* We should use that one to take the more size it can */
182 //    setSizePolicy( QSizePolicy::Preferred, QSizePolicy::Preferred );
183
184     /* A dark background */
185     setAutoFillBackground( true );
186     plt =  palette();
187     plt.setColor( QPalette::Active, QPalette::Window , Qt::black );
188     plt.setColor( QPalette::Inactive, QPalette::Window , Qt::black );
189     setPalette( plt );
190
191     /* A cone in the middle */
192     label = new QLabel;
193     label->setMargin( 5 );
194     label->setMaximumHeight( MAX_BG_SIZE );
195     label->setMaximumWidth( MAX_BG_SIZE );
196     label->setMinimumHeight( MIN_BG_SIZE );
197     label->setMinimumWidth( MIN_BG_SIZE );
198     if( QDate::currentDate().dayOfYear() >= 354 )
199         label->setPixmap( QPixmap( ":/vlc128-christmas.png" ) );
200     else
201         label->setPixmap( QPixmap( ":/vlc128.png" ) );
202
203     QGridLayout *backgroundLayout = new QGridLayout( this );
204     backgroundLayout->addWidget( label, 0, 1 );
205     backgroundLayout->setColumnStretch( 0, 1 );
206     backgroundLayout->setColumnStretch( 2, 1 );
207
208     CONNECT( THEMIM->getIM(), artChanged( QString ), this, updateArt( QString ) );
209 }
210
211 BackgroundWidget::~BackgroundWidget()
212 {
213 }
214
215 void BackgroundWidget::resizeEvent( QResizeEvent * event )
216 {
217     if( event->size().height() <= MIN_BG_SIZE )
218         label->hide();
219     else
220         label->show();
221 }
222
223 void BackgroundWidget::updateArt( QString url )
224 {
225     if( url.isEmpty() )
226     {
227         if( QDate::currentDate().dayOfYear() >= 354 )
228             label->setPixmap( QPixmap( ":/vlc128-christmas.png" ) );
229         else
230             label->setPixmap( QPixmap( ":/vlc128.png" ) );
231         return;
232     }
233     else
234     {
235         label->setPixmap( QPixmap( url ) );
236     }
237 }
238
239 void BackgroundWidget::contextMenuEvent( QContextMenuEvent *event )
240 {
241     QVLCMenu::PopupMenu( p_intf, true );
242 }
243
244 /**********************************************************************
245  * Visualization selector panel
246  **********************************************************************/
247 VisualSelector::VisualSelector( intf_thread_t *_p_i ) :
248                                 QFrame( NULL ), p_intf( _p_i )
249 {
250     QHBoxLayout *layout = new QHBoxLayout( this );
251     layout->setMargin( 0 );
252     QPushButton *prevButton = new QPushButton( "Prev" );
253     QPushButton *nextButton = new QPushButton( "Next" );
254     layout->addWidget( prevButton );
255     layout->addWidget( nextButton );
256
257     layout->addItem( new QSpacerItem( 40,20,
258                               QSizePolicy::Expanding, QSizePolicy::Minimum ) );
259     layout->addWidget( new QLabel( qtr( "Current visualization:" ) ) );
260
261     current = new QLabel( qtr( "None" ) );
262     layout->addWidget( current );
263
264     BUTTONACT( prevButton, prev() );
265     BUTTONACT( nextButton, next() );
266
267     setLayout( layout );
268     setMaximumHeight( 35 );
269 }
270
271 VisualSelector::~VisualSelector()
272 {
273 }
274
275 void VisualSelector::prev()
276 {
277     char *psz_new = aout_VisualPrev( p_intf );
278     if( psz_new )
279     {
280         current->setText( qfu( psz_new ) );
281         free( psz_new );
282     }
283 }
284
285 void VisualSelector::next()
286 {
287     char *psz_new = aout_VisualNext( p_intf );
288     if( psz_new )
289     {
290         current->setText( qfu( psz_new ) );
291         free( psz_new );
292     }
293 }
294
295 /**********************************************************************
296  * TEH controls
297  **********************************************************************/
298
299 #define setupSmallButton( aButton ){  \
300     aButton->setMaximumSize( QSize( 26, 26 ) ); \
301     aButton->setMinimumSize( QSize( 26, 26 ) ); \
302     aButton->setIconSize( QSize( 20, 20 ) ); }
303
304 AdvControlsWidget::AdvControlsWidget( intf_thread_t *_p_i ) :
305                                            QFrame( NULL ), p_intf( _p_i )
306 {
307     QHBoxLayout *advLayout = new QHBoxLayout( this );
308     advLayout->setMargin( 0 );
309     advLayout->setSpacing( 0 );
310     advLayout->setAlignment( Qt::AlignBottom );
311
312     /* A to B Button */
313     ABButton = new QPushButton( "AB" );
314     setupSmallButton( ABButton );
315     advLayout->addWidget( ABButton );
316     BUTTON_SET_ACT( ABButton, "AB", qtr( "A to B" ), fromAtoB() );
317     timeA = timeB = 0;
318     CONNECT( THEMIM->getIM(), positionUpdated( float, int, int ),
319              this, AtoBLoop( float, int, int ) );
320 #if 0
321     frameButton = new QPushButton( "Fr" );
322     frameButton->setMaximumSize( QSize( 26, 26 ) );
323     frameButton->setIconSize( QSize( 20, 20 ) );
324     advLayout->addWidget( frameButton );
325     BUTTON_SET_ACT( frameButton, "Fr", qtr( "Frame by Frame" ), frame() );
326 #endif
327
328     recordButton = new QPushButton( "R" );
329     setupSmallButton( recordButton );
330     advLayout->addWidget( recordButton );
331     BUTTON_SET_ACT_I( recordButton, "", record_16px.png,
332             qtr( "Record" ), record() );
333
334     /* Snapshot Button */
335     snapshotButton = new QPushButton( "S" );
336     setupSmallButton( snapshotButton );
337     advLayout->addWidget( snapshotButton );
338     BUTTON_SET_ACT( snapshotButton, "S", qtr( "Take a snapshot" ), snapshot() );
339 }
340
341 AdvControlsWidget::~AdvControlsWidget()
342 {}
343
344 void AdvControlsWidget::enableInput( bool enable )
345 {
346     ABButton->setEnabled( enable );
347     recordButton->setEnabled( enable );
348 }
349
350 void AdvControlsWidget::enableVideo( bool enable )
351 {
352     snapshotButton->setEnabled( enable );
353 #if 0
354     frameButton->setEnabled( enable );
355 #endif
356 }
357
358 void AdvControlsWidget::snapshot()
359 {
360     vout_thread_t *p_vout =
361         (vout_thread_t *)vlc_object_find( p_intf, VLC_OBJECT_VOUT, FIND_ANYWHERE );
362     if( p_vout ) vout_Control( p_vout, VOUT_SNAPSHOT );
363 }
364
365 /* Function called when the button is clicked() */
366 void AdvControlsWidget::fromAtoB()
367 {
368     if( !timeA )
369     {
370         timeA = var_GetTime( THEMIM->getInput(), "time"  );
371         ABButton->setText( "A->..." );
372         return;
373     }
374     if( !timeB )
375     {
376         timeB = var_GetTime( THEMIM->getInput(), "time"  );
377         var_SetTime( THEMIM->getInput(), "time" , timeA );
378         ABButton->setText( "A<=>B" );
379         return;
380     }
381     timeA = 0;
382     timeB = 0;
383     ABButton->setText( "AB" );
384 }
385
386 /* Function called regularly when in an AtoB loop */
387 void AdvControlsWidget::AtoBLoop( float f_pos, int i_time, int i_length )
388 {
389     if( timeB )
390     {
391         if( i_time >= (int)(timeB/1000000) )
392             var_SetTime( THEMIM->getInput(), "time" , timeA );
393     }
394 }
395
396 /* FIXME Record function */
397 void AdvControlsWidget::record(){}
398
399 #if 0
400 //FIXME Frame by frame function
401 void AdvControlsWidget::frame(){}
402 #endif
403
404 /*****************************
405  * DA Control Widget !
406  *****************************/
407 ControlsWidget::ControlsWidget( intf_thread_t *_p_i,
408                                 MainInterface *_p_mi,
409                                 bool b_advControls,
410                                 bool b_shiny,
411                                 bool b_fsCreation) :
412                                 QFrame( _p_mi ), p_intf( _p_i )
413 {
414     controlLayout = new QGridLayout( );
415
416     controlLayout->setSpacing( 0 );
417     controlLayout->setLayoutMargins( 7, 5, 7, 3, 6 );
418
419     if( !b_fsCreation )
420         setLayout( controlLayout );
421
422     setSizePolicy( QSizePolicy::Preferred , QSizePolicy::Maximum );
423
424     /** The main Slider **/
425     slider = new InputSlider( Qt::Horizontal, NULL );
426     controlLayout->addWidget( slider, 0, 1, 1, 16 );
427     /* Update the position when the IM has changed */
428     CONNECT( THEMIM->getIM(), positionUpdated( float, int, int ),
429              slider, setPosition( float, int, int ) );
430     /* And update the IM, when the position has changed */
431     CONNECT( slider, sliderDragged( float ),
432              THEMIM->getIM(), sliderUpdate( float ) );
433
434     /** Slower and faster Buttons **/
435     slowerButton = new QToolButton;
436     slowerButton->setAutoRaise( true );
437     slowerButton->setMaximumSize( QSize( 26, 20 ) );
438
439     BUTTON_SET_ACT( slowerButton, "-", qtr( "Slower" ), slower() );
440     controlLayout->addWidget( slowerButton, 0, 0 );
441
442     fasterButton = new QToolButton;
443     fasterButton->setAutoRaise( true );
444     fasterButton->setMaximumSize( QSize( 26, 20 ) );
445
446     BUTTON_SET_ACT( fasterButton, "+", qtr( "Faster" ), faster() );
447     controlLayout->addWidget( fasterButton, 0, 17 );
448
449     /* advanced Controls handling */
450     b_advancedVisible = b_advControls;
451
452     advControls = new AdvControlsWidget( p_intf );
453     controlLayout->addWidget( advControls, 1, 3, 2, 4, Qt::AlignBottom );
454     if( !b_advancedVisible ) advControls->hide();
455
456     /** Disc and Menus handling */
457     discFrame = new QWidget( this );
458
459     QHBoxLayout *discLayout = new QHBoxLayout( discFrame );
460     discLayout->setSpacing( 0 );
461     discLayout->setMargin( 0 );
462
463     prevSectionButton = new QPushButton( discFrame );
464     setupSmallButton( prevSectionButton );
465     discLayout->addWidget( prevSectionButton );
466
467     menuButton = new QPushButton( discFrame );
468     setupSmallButton( menuButton );
469     discLayout->addWidget( menuButton );
470
471     nextSectionButton = new QPushButton( discFrame );
472     setupSmallButton( nextSectionButton );
473     discLayout->addWidget( nextSectionButton );
474
475     controlLayout->addWidget( discFrame, 1, 10, 2, 3, Qt::AlignBottom );
476
477     BUTTON_SET_IMG( prevSectionButton, "", previous.png, "" );
478     BUTTON_SET_IMG( nextSectionButton, "", next.png, "" );
479     BUTTON_SET_IMG( menuButton, "", previous.png, "" );
480
481     discFrame->hide();
482
483     /* Change the navigation button display when the IM navigation changes */
484     CONNECT( THEMIM->getIM(), navigationChanged( int ),
485              this, setNavigation( int ) );
486     /* Changes the IM navigation when triggered on the nav buttons */
487     CONNECT( prevSectionButton, clicked(), THEMIM->getIM(),
488              sectionPrev() );
489     CONNECT( nextSectionButton, clicked(), THEMIM->getIM(),
490              sectionNext() );
491     CONNECT( menuButton, clicked(), THEMIM->getIM(),
492              sectionMenu() );
493
494     /**
495      * Telextext QFrame
496      * TODO: Merge with upper menu in a StackLayout
497      **/
498     telexFrame = new QWidget( this );
499     QHBoxLayout *telexLayout = new QHBoxLayout( telexFrame );
500     telexLayout->setSpacing( 0 );
501     telexLayout->setMargin( 0 );
502
503     QPushButton  *telexOn = new QPushButton;
504     setupSmallButton( telexOn );
505     telexLayout->addWidget( telexOn );
506
507     telexTransparent = new QPushButton;
508     setupSmallButton( telexTransparent );
509     telexLayout->addWidget( telexTransparent );
510     b_telexTransparent = false;
511
512     telexPage = new QSpinBox;
513     telexPage->setRange( 0, 999 );
514     telexPage->setValue( 100 );
515     telexPage->setAccelerated( true );
516     telexPage->setWrapping( true );
517     telexPage->setAlignment( Qt::AlignRight );
518     telexPage->setSizePolicy( QSizePolicy::Preferred, QSizePolicy::Minimum );
519     telexLayout->addWidget( telexPage );
520
521     controlLayout->addWidget( telexFrame, 1, 10, 2, 4, Qt::AlignBottom );
522     telexFrame->hide(); /* default hidden */
523
524     CONNECT( telexPage, valueChanged( int ), THEMIM->getIM(),
525              telexGotoPage( int ) );
526
527     BUTTON_SET_ACT_I( telexOn, "", tv.png, qtr( "Teletext on" ),
528                       toggleTeletext() );
529     CONNECT( telexOn, clicked( bool ), THEMIM->getIM(),
530              telexToggle( bool ) );
531     telexTransparent->setEnabled( false );
532     telexPage->setEnabled( false );
533
534     BUTTON_SET_ACT_I( telexTransparent, "", tvtelx.png, qtr( "Teletext" ),
535                       toggleTeletextTransparency() );
536     CONNECT( telexTransparent, clicked( bool ),
537              THEMIM->getIM(), telexSetTransparency() );
538     CONNECT( THEMIM->getIM(), teletextEnabled( bool ),
539              telexFrame, setVisible( bool ) );
540
541     /** Play Buttons **/
542     QSizePolicy sizePolicy( QSizePolicy::Fixed, QSizePolicy::Fixed );
543     sizePolicy.setHorizontalStretch( 0 );
544     sizePolicy.setVerticalStretch( 0 );
545
546     /* Play */
547     playButton = new QPushButton;
548     playButton->setSizePolicy( sizePolicy );
549     playButton->setMaximumSize( QSize( 36, 36 ) );
550     playButton->setMinimumSize( QSize( 36, 36 ) );
551     playButton->setIconSize( QSize( 30, 30 ) );
552
553     controlLayout->addWidget( playButton, 2, 0, 2, 2 );
554
555     controlLayout->setColumnMinimumWidth( 2, 20 );
556     controlLayout->setColumnStretch( 2, 0 );
557
558     /** Prev + Stop + Next Block **/
559     controlButLayout = new QHBoxLayout;
560     controlButLayout->setSpacing( 0 ); /* Don't remove that, will be useful */
561
562     /* Prev */
563     QPushButton *prevButton = new QPushButton;
564     prevButton->setSizePolicy( sizePolicy );
565     setupSmallButton( prevButton );
566
567     controlButLayout->addWidget( prevButton );
568
569     /* Stop */
570     QPushButton *stopButton = new QPushButton;
571     stopButton->setSizePolicy( sizePolicy );
572     setupSmallButton( stopButton );
573
574     controlButLayout->addWidget( stopButton );
575
576     /* next */
577     QPushButton *nextButton = new QPushButton;
578     nextButton->setSizePolicy( sizePolicy );
579     setupSmallButton( nextButton );
580
581     controlButLayout->addWidget( nextButton );
582
583     /* Add this block to the main layout */
584     if( !b_fsCreation )
585         controlLayout->addLayout( controlButLayout, 3, 3, 1, 3 );
586
587     BUTTON_SET_ACT_I( playButton, "", play.png, qtr( "Play" ), play() );
588     BUTTON_SET_ACT_I( prevButton, "" , previous.png,
589                       qtr( "Previous" ), prev() );
590     BUTTON_SET_ACT_I( nextButton, "", next.png, qtr( "Next" ), next() );
591     BUTTON_SET_ACT_I( stopButton, "", stop.png, qtr( "Stop" ), stop() );
592
593     controlLayout->setColumnMinimumWidth( 7, 20 );
594     controlLayout->setColumnStretch( 7, 0 );
595     controlLayout->setColumnStretch( 8, 0 );
596     controlLayout->setColumnStretch( 9, 0 );
597
598     /*
599      * Other first Line buttons
600      */
601     /** Fullscreen/Visualisation **/
602     fullscreenButton = new QPushButton( "F" );
603     BUTTON_SET_ACT( fullscreenButton, "F", qtr( "Fullscreen" ), fullscreen() );
604     setupSmallButton( fullscreenButton );
605     controlLayout->addWidget( fullscreenButton, 3, 10, Qt::AlignBottom );
606
607     /** Playlist Button **/
608     playlistButton = new QPushButton;
609     setupSmallButton( playlistButton );
610     controlLayout->addWidget( playlistButton, 3, 11, Qt::AlignBottom );
611     BUTTON_SET_IMG( playlistButton, "" , playlist.png, qtr( "Show playlist" ) );
612     CONNECT( playlistButton, clicked(), _p_mi, togglePlaylist() );
613
614     /** extended Settings **/
615     extSettingsButton = new QPushButton;
616     BUTTON_SET_ACT( extSettingsButton, "Ex", qtr( "Extended Settings" ),
617             extSettings() );
618     setupSmallButton( extSettingsButton );
619     controlLayout->addWidget( extSettingsButton, 3, 12, Qt::AlignBottom );
620
621     controlLayout->setColumnStretch( 13, 0 );
622     controlLayout->setColumnMinimumWidth( 13, 24 );
623     controlLayout->setColumnStretch( 14, 5 );
624
625     /* Volume */
626     hVolLabel = new VolumeClickHandler( p_intf, this );
627
628     volMuteLabel = new QLabel;
629     volMuteLabel->setPixmap( QPixmap( ":/pixmaps/volume-medium.png" ) );
630     volMuteLabel->setToolTip( qtr( "Mute" ) );
631     volMuteLabel->installEventFilter( hVolLabel );
632     controlLayout->addWidget( volMuteLabel, 3, 15, Qt::AlignBottom );
633
634     if( b_shiny )
635     {
636         volumeSlider = new SoundSlider( this,
637             config_GetInt( p_intf, "volume-step" ),
638             config_GetInt( p_intf, "qt-volume-complete" ),
639             config_GetPsz( p_intf, "qt-slider-colours" ) );
640     }
641     else
642     {
643         volumeSlider = new QSlider( this );
644         volumeSlider->setOrientation( Qt::Horizontal );
645     }
646     volumeSlider->setMaximumSize( QSize( 200, 40 ) );
647     volumeSlider->setMinimumSize( QSize( 106, 30 ) );
648     volumeSlider->setFocusPolicy( Qt::NoFocus );
649     controlLayout->addWidget( volumeSlider, 2, 16, 2 , 2, Qt::AlignBottom );
650
651     /* Set the volume from the config */
652     volumeSlider->setValue( ( config_GetInt( p_intf, "volume" ) ) *
653                               VOLUME_MAX / (AOUT_VOLUME_MAX/2) );
654
655     /* Force the update at build time in order to have a muted icon if needed */
656     updateVolume( volumeSlider->value() );
657
658     /* Volume control connection */
659     CONNECT( volumeSlider, valueChanged( int ), this, updateVolume( int ) );
660     CONNECT( THEMIM, volumeChanged( void ), this, updateVolume( void ) );
661
662     updateInput();
663 }
664
665 ControlsWidget::~ControlsWidget()
666 {}
667
668 void ControlsWidget::toggleTeletext()
669 {
670     bool b_enabled = THEMIM->teletextState();
671     if( b_telexEnabled )
672     {
673         telexTransparent->setEnabled( false );
674         telexPage->setEnabled( false );
675         b_telexEnabled = false;
676     }
677     else if( b_enabled )
678     {
679         telexTransparent->setEnabled( true );
680         telexPage->setEnabled( true );
681         b_telexEnabled = true;
682     }
683 }
684
685 void ControlsWidget::toggleTeletextTransparency()
686 {
687     if( b_telexTransparent )
688     {
689         telexTransparent->setIcon( QIcon( ":/pixmaps/tvtelx.png" ) );
690         telexTransparent->setToolTip( qtr( "Teletext" ) );
691         b_telexTransparent = false;
692     }
693     else
694     {
695         telexTransparent->setIcon( QIcon( ":/pixmaps/tvtelx-transparent.png" ) );
696         telexTransparent->setToolTip( qtr( "Transparent" ) );
697         b_telexTransparent = true;
698     }
699 }
700
701 void ControlsWidget::stop()
702 {
703     THEMIM->stop();
704 }
705
706 void ControlsWidget::play()
707 {
708     if( THEPL->current.i_size == 0 )
709     {
710         /* The playlist is empty, open a file requester */
711         THEDP->openFileDialog();
712         setStatus( 0 );
713         return;
714     }
715     THEMIM->togglePlayPause();
716 }
717
718 void ControlsWidget::prev()
719 {
720     THEMIM->prev();
721 }
722
723 void ControlsWidget::next()
724 {
725     THEMIM->next();
726 }
727
728 void ControlsWidget::setNavigation( int navigation )
729 {
730 #define HELP_MENU N_( "Menu" )
731 #define HELP_PCH N_( "Previous chapter" )
732 #define HELP_NCH N_( "Next chapter" )
733 #define HELP_PTR N_( "Previous track" )
734 #define HELP_NTR N_( "Next track" )
735
736     // 1 = chapter, 2 = title, 0 = no
737     if( navigation == 0 )
738     {
739         discFrame->hide();
740     } else if( navigation == 1 ) {
741         prevSectionButton->setToolTip( qfu( HELP_PCH ) );
742         nextSectionButton->setToolTip( qfu( HELP_NCH ) );
743         menuButton->show();
744         discFrame->show();
745     } else {
746         prevSectionButton->setToolTip( qfu( HELP_PCH ) );
747         nextSectionButton->setToolTip( qfu( HELP_NCH ) );
748         menuButton->hide();
749         discFrame->show();
750     }
751 }
752
753 static bool b_my_volume;
754 void ControlsWidget::updateVolume( int i_sliderVolume )
755 {
756     if( !b_my_volume )
757     {
758         int i_res = i_sliderVolume  * (AOUT_VOLUME_MAX / 2) / VOLUME_MAX;
759         aout_VolumeSet( p_intf, i_res );
760     }
761     if( i_sliderVolume == 0 )
762         volMuteLabel->setPixmap( QPixmap(":/pixmaps/volume-muted.png" ) );
763     else if( i_sliderVolume < VOLUME_MAX / 3 )
764         volMuteLabel->setPixmap( QPixmap( ":/pixmaps/volume-low.png" ) );
765     else if( i_sliderVolume > (VOLUME_MAX * 2 / 3 ) )
766         volMuteLabel->setPixmap( QPixmap( ":/pixmaps/volume-high.png" ) );
767     else volMuteLabel->setPixmap( QPixmap( ":/pixmaps/volume-medium.png" ) );
768 }
769
770 void ControlsWidget::updateVolume()
771 {
772     /* Audio part */
773     audio_volume_t i_volume;
774     aout_VolumeGet( p_intf, &i_volume );
775     i_volume = ( i_volume *  VOLUME_MAX )/ (AOUT_VOLUME_MAX/2);
776     int i_gauge = volumeSlider->value();
777     b_my_volume = false;
778     if( i_volume - i_gauge > 1 || i_gauge - i_volume > 1 )
779     {
780         b_my_volume = true;
781         volumeSlider->setValue( i_volume );
782         b_my_volume = false;
783     }
784 }
785
786 void ControlsWidget::updateInput()
787 {
788     /* Activate the interface buttons according to the presence of the input */
789     enableInput( THEMIM->getIM()->hasInput() );
790     enableVideo( THEMIM->getIM()->hasVideo() && THEMIM->getIM()->hasInput() );
791 }
792
793 void ControlsWidget::setStatus( int status )
794 {
795     if( status == PLAYING_S ) /* Playing */
796     {
797         playButton->setIcon( QIcon( ":/pixmaps/pause.png" ) );
798         playButton->setToolTip( qtr( "Pause" ) );
799     }
800     else
801     {
802         playButton->setIcon( QIcon( ":/pixmaps/play.png" ) );
803         playButton->setToolTip( qtr( "Play" ) );
804     }
805 }
806
807 /**
808  * TODO
809  * This functions toggle the fullscreen mode
810  * If there is no video, it should first activate Visualisations...
811  *  This has also to be fixed in enableVideo()
812  */
813 void ControlsWidget::fullscreen()
814 {
815     vout_thread_t *p_vout =
816         (vout_thread_t *)vlc_object_find( p_intf, VLC_OBJECT_VOUT, FIND_ANYWHERE );
817     if( p_vout)
818     {
819         var_SetBool( p_vout, "fullscreen", !var_GetBool( p_vout, "fullscreen" ) );
820         vlc_object_release( p_vout );
821     }
822 }
823
824 void ControlsWidget::extSettings()
825 {
826     THEDP->extendedDialog();
827 }
828
829 void ControlsWidget::slower()
830 {
831     THEMIM->getIM()->slower();
832 }
833
834 void ControlsWidget::faster()
835 {
836     THEMIM->getIM()->faster();
837 }
838
839 void ControlsWidget::enableInput( bool enable )
840 {
841     slowerButton->setEnabled( enable );
842     slider->setEnabled( enable );
843     fasterButton->setEnabled( enable );
844
845     /* Advanced Buttons too */
846     advControls->enableInput( enable );
847 }
848
849 void ControlsWidget::enableVideo( bool enable )
850 {
851     // TODO Later make the fullscreenButton toggle Visualisation and so on.
852     fullscreenButton->setEnabled( enable );
853
854     /* Advanced Buttons too */
855     advControls->enableVideo( enable );
856 }
857
858 void ControlsWidget::toggleAdvanced()
859 {
860     if( !VISIBLE( advControls ) )
861     {
862         advControls->show();
863         b_advancedVisible = true;
864     }
865     else
866     {
867         advControls->hide();
868         b_advancedVisible = false;
869     }
870     emit advancedControlsToggled( b_advancedVisible );
871 }
872
873
874 /**********************************************************************
875  * Fullscrenn control widget
876  **********************************************************************/
877 FullscreenControllerWidget::FullscreenControllerWidget( intf_thread_t *_p_i,
878         MainInterface *_p_mi, bool b_advControls, bool b_shiny )
879         : ControlsWidget( _p_i, _p_mi, b_advControls, b_shiny, true ),
880         i_lastPosX( -1 ), i_lastPosY( -1 ), i_hideTimeout( 1 ),
881         b_mouseIsOver( false )
882 {
883     setWindowFlags( Qt::ToolTip );
884
885     setFrameShape( QFrame::StyledPanel );
886     setFrameStyle( QFrame::Sunken );
887     setSizePolicy( QSizePolicy::Minimum, QSizePolicy::Minimum );
888
889     QGridLayout *fsLayout = new QGridLayout( this );
890     controlLayout->setSpacing( 0 );
891     controlLayout->setLayoutMargins( 5, 1, 5, 1, 5 );
892
893     fsLayout->addWidget( slowerButton, 0, 0 );
894     slider->setSizePolicy( QSizePolicy::Expanding, QSizePolicy::Minimum);
895     fsLayout->addWidget( slider, 0, 1, 1, 6 );
896     fsLayout->addWidget( fasterButton, 0, 7 );
897
898     fsLayout->addWidget( volMuteLabel, 1, 0);
899     fsLayout->addWidget( volumeSlider, 1, 1 );
900
901     fsLayout->addLayout( controlButLayout, 1, 2 );
902
903     fsLayout->addWidget( playButton, 1, 3 );
904
905     fsLayout->addWidget( discFrame, 1, 4 );
906
907     fsLayout->addWidget( telexFrame, 1, 5 );
908
909     fsLayout->addWidget( advControls, 1, 6, Qt::AlignVCenter );
910
911     fsLayout->addWidget( fullscreenButton, 1, 7 );
912
913     /* hiding timer */
914     p_hideTimer = new QTimer( this );
915     CONNECT( p_hideTimer, timeout(), this, hideFSControllerWidget() );
916     p_hideTimer->setSingleShot( true );
917
918     /* slow hiding timer */
919 #if HAVE_TRANSPARENCY
920     p_slowHideTimer = new QTimer( this );
921     CONNECT( p_slowHideTimer, timeout(), this, slowHideFSC() );
922 #endif
923
924     adjustSize ();  /* need to get real width and height for moving */
925
926     /* center down */
927     QDesktopWidget * p_desktop = QApplication::desktop();
928
929     move( p_desktop->width() / 2 - width() / 2,
930           p_desktop->height() - height() );
931
932     #ifdef WIN32TRICK
933     setWindowOpacity( 0.0 );
934     fscHidden = true;
935     show();
936     #endif
937 }
938
939 FullscreenControllerWidget::~FullscreenControllerWidget()
940 {
941 }
942
943 /**
944  * Hide fullscreen controller
945  * FIXME: under windows it have to be done by moving out of screen
946  *        because hide() doesnt work
947  */
948 void FullscreenControllerWidget::hideFSControllerWidget()
949 {
950     #ifdef WIN32TRICK
951     fscHidden = true;
952     setWindowOpacity( 0.0 );    // simulate hidding
953     #else
954     hide();
955     #endif
956 }
957
958 /**
959  * Hidding fullscreen controller slowly
960  * Linux: need composite manager
961  * Windows: it is blinking, so it can be enabled by define TRASPARENCY
962  */
963 void FullscreenControllerWidget::slowHideFSC()
964 {
965 #if HAVE_TRANSPARENCY
966     static bool first_call = true;
967
968     if ( first_call )
969     {
970         first_call = false;
971
972         p_slowHideTimer->stop();
973         /* the last part of time divided to 100 pieces */
974         p_slowHideTimer->start(
975             (int) ( i_hideTimeout / 2 / ( windowOpacity() * 100 ) ) );
976     }
977     else
978     {
979          if ( windowOpacity() > 0.0 )
980          {
981              /* we should use 0.01 because of 100 pieces ^^^
982                 but than it cannt be done in time */
983              setWindowOpacity( windowOpacity() - 0.02 );
984          }
985
986          if ( windowOpacity() == 0.0 )
987          {
988              first_call = true;
989              p_slowHideTimer->stop();
990          }
991     }
992 #endif
993 }
994
995 /**
996  * Get state of visibility of FS controller on screen
997  * On windows control if it is on hidden position
998  */
999 bool FullscreenControllerWidget::isFSCHidden()
1000 {
1001     #ifdef WIN32TRICK
1002     return fscHidden;
1003     #endif
1004
1005     return isHidden();
1006 }
1007
1008 /**
1009  * event handling
1010  * events: show, hide, start timer for hidding
1011  */
1012 void FullscreenControllerWidget::customEvent( QEvent *event )
1013 {
1014     int type = event->type();
1015
1016     if ( type == FullscreenControlShow_Type )
1017     {
1018         #ifdef WIN32TRICK
1019         // after quiting and going to fs, we need to call show()
1020         if ( isHidden() )
1021             show();
1022
1023         if ( fscHidden )
1024         {
1025             fscHidden = false;
1026             setWindowOpacity( 1.0 );
1027         }
1028         #else
1029         show();
1030         #endif
1031
1032 #if HAVE_TRANSPARENCY
1033         setWindowOpacity( DEFAULT_OPACITY );
1034 #endif
1035     }
1036     else if ( type == FullscreenControlHide_Type )
1037     {
1038         hideFSControllerWidget();
1039     }
1040     else if ( type == FullscreenControlPlanHide_Type && !b_mouseIsOver )
1041     {
1042         p_hideTimer->start( i_hideTimeout );
1043 #if HAVE_TRANSPARENCY
1044         p_slowHideTimer->start( i_hideTimeout / 2 );
1045 #endif
1046     }
1047 }
1048
1049 /**
1050  * On mouse move
1051  * moving with FSC
1052  */
1053 void FullscreenControllerWidget::mouseMoveEvent( QMouseEvent *event )
1054 {
1055     if ( event->buttons() == Qt::LeftButton )
1056     {
1057         int i_moveX = event->globalX() - i_lastPosX;
1058         int i_moveY = event->globalY() - i_lastPosY;
1059
1060         move( x() + i_moveX, y() + i_moveY );
1061
1062         i_lastPosX = event->globalX();
1063         i_lastPosY = event->globalY();
1064     }
1065 }
1066
1067 /**
1068  * On mouse press
1069  * store position of cursor
1070  */
1071 void FullscreenControllerWidget::mousePressEvent( QMouseEvent *event )
1072 {
1073     i_lastPosX = event->globalX();
1074     i_lastPosY = event->globalY();
1075 }
1076
1077 /**
1078  * On mouse go above FSC
1079  */
1080 void FullscreenControllerWidget::enterEvent( QEvent *event )
1081 {
1082     p_hideTimer->stop();
1083 #if HAVE_TRANSPARENCY
1084     p_slowHideTimer->stop();
1085 #endif
1086     b_mouseIsOver = true;
1087 }
1088
1089 /**
1090  * On mouse go out from FSC
1091  */
1092 void FullscreenControllerWidget::leaveEvent( QEvent *event )
1093 {
1094     p_hideTimer->start( i_hideTimeout );
1095 #if HAVE_TRANSPARENCY
1096     p_slowHideTimer->start( i_hideTimeout / 2 );
1097 #endif
1098     b_mouseIsOver = false;
1099 }
1100
1101 /**
1102  * When you get pressed key, send it to video output
1103  * FIXME: clearing focus by clearFocus() to not getting
1104  * key press events didnt work
1105  */
1106 void FullscreenControllerWidget::keyPressEvent( QKeyEvent *event )
1107 {
1108     int i_vlck = qtEventToVLCKey( event );
1109     if( i_vlck > 0 )
1110     {
1111         var_SetInteger( p_intf->p_libvlc, "key-pressed", i_vlck );
1112         event->accept();
1113     }
1114     else
1115         event->ignore();
1116 }
1117
1118 /**
1119  * It is called when video start
1120  */
1121 void FullscreenControllerWidget::regFullscreenCallback( vout_thread_t *p_vout )
1122 {
1123     if ( p_vout )
1124     {
1125         var_AddCallback( p_vout, "fullscreen", regMouseMoveCallback, this );
1126     }
1127 }
1128
1129 /**
1130  * It is called after turn off video, because p_vout is NULL now
1131  * we cannt delete callback, just hide if FScontroller is visible
1132  */
1133 void FullscreenControllerWidget::unregFullscreenCallback()
1134 {
1135     if ( isVisible() )
1136         hide();
1137 }
1138
1139 /**
1140  * Register and unregister callback for mouse moving
1141  */
1142 static int regMouseMoveCallback( vlc_object_t *vlc_object, const char *variable,
1143                                  vlc_value_t old_val, vlc_value_t new_val,
1144                                  void *data )
1145 {
1146     vout_thread_t *p_vout = (vout_thread_t *) vlc_object;
1147
1148     static bool b_registered = false;
1149     FullscreenControllerWidget *p_fs = (FullscreenControllerWidget *) data;
1150
1151     if ( var_GetBool( p_vout, "fullscreen" ) && !b_registered )
1152     {
1153         p_fs->SetHideTimeout( var_GetInteger( p_vout, "mouse-hide-timeout" ) );
1154         var_AddCallback( p_vout, "mouse-moved",
1155                         showFullscreenControllCallback, (void *) p_fs );
1156         b_registered = true;
1157     }
1158
1159     if ( !var_GetBool( p_vout, "fullscreen" ) && b_registered )
1160     {
1161         var_DelCallback( p_vout, "mouse-moved",
1162                         showFullscreenControllCallback, (void *) p_fs );
1163         b_registered = false;
1164         p_fs->hide();
1165     }
1166
1167     return VLC_SUCCESS;
1168 }
1169
1170 /**
1171  * Show fullscreen controller after mouse move
1172  * after show immediately plan hide event
1173  */
1174 static int showFullscreenControllCallback( vlc_object_t *vlc_object, const char *variable,
1175                                            vlc_value_t old_val, vlc_value_t new_val,
1176                                            void *data )
1177 {
1178     FullscreenControllerWidget *p_fs = (FullscreenControllerWidget *) data;
1179
1180     if ( p_fs->isFSCHidden() || p_fs->windowOpacity() < DEFAULT_OPACITY )
1181     {
1182         IMEvent *event = new IMEvent( FullscreenControlShow_Type, 0 );
1183         QApplication::postEvent( p_fs, static_cast<QEvent *>(event) );
1184     }
1185
1186     IMEvent *e = new IMEvent( FullscreenControlPlanHide_Type, 0 );
1187     QApplication::postEvent( p_fs, static_cast<QEvent *>(e) );
1188
1189     return VLC_SUCCESS;
1190 }
1191
1192 /**********************************************************************
1193  * Speed control widget
1194  **********************************************************************/
1195 SpeedControlWidget::SpeedControlWidget( intf_thread_t *_p_i ) :
1196                              QFrame( NULL ), p_intf( _p_i )
1197 {
1198     QSizePolicy sizePolicy( QSizePolicy::Maximum, QSizePolicy::Fixed );
1199     sizePolicy.setHorizontalStretch( 0 );
1200     sizePolicy.setVerticalStretch( 0 );
1201
1202     speedSlider = new QSlider;
1203     speedSlider->setSizePolicy( sizePolicy );
1204     speedSlider->setMaximumSize( QSize( 80, 200 ) );
1205     speedSlider->setOrientation( Qt::Vertical );
1206     speedSlider->setTickPosition( QSlider::TicksRight );
1207
1208     speedSlider->setRange( -100, 100 );
1209     speedSlider->setSingleStep( 10 );
1210     speedSlider->setPageStep( 20 );
1211     speedSlider->setTickInterval( 20 );
1212
1213     CONNECT( speedSlider, valueChanged( int ), this, updateRate( int ) );
1214
1215     QToolButton *normalSpeedButton = new QToolButton( this );
1216     normalSpeedButton->setMaximumSize( QSize( 26, 20 ) );
1217     normalSpeedButton->setAutoRaise( true );
1218     normalSpeedButton->setText( "N" );
1219     normalSpeedButton->setToolTip( qtr( "Revert to normal play speed" ) );
1220
1221     CONNECT( normalSpeedButton, clicked(), this, resetRate() );
1222
1223     QVBoxLayout *speedControlLayout = new QVBoxLayout;
1224     speedControlLayout->addWidget( speedSlider );
1225     speedControlLayout->addWidget( normalSpeedButton );
1226     setLayout( speedControlLayout );
1227 }
1228
1229 SpeedControlWidget::~SpeedControlWidget()
1230 {}
1231
1232 void SpeedControlWidget::setEnable( bool b_enable )
1233 {
1234     speedSlider->setEnabled( b_enable );
1235 }
1236
1237 #define RATE_SLIDER_MAXIMUM 3.0
1238 #define RATE_SLIDER_MINIMUM 0.3
1239 #define RATE_SLIDER_LENGTH 100.0
1240
1241 void SpeedControlWidget::updateControls( int rate )
1242 {
1243     if( speedSlider->isSliderDown() )
1244     {
1245         //We don't want to change anything if the user is using the slider
1246         return;
1247     }
1248
1249     int sliderValue;
1250     double speed = INPUT_RATE_DEFAULT / (double)rate;
1251
1252     if( rate >= INPUT_RATE_DEFAULT )
1253     {
1254         if( speed < RATE_SLIDER_MINIMUM )
1255         {
1256             sliderValue = speedSlider->minimum();
1257         }
1258         else
1259         {
1260             sliderValue = (int)( ( speed - 1.0 ) * RATE_SLIDER_LENGTH
1261                                         / ( 1.0 - RATE_SLIDER_MAXIMUM ) );
1262         }
1263     }
1264     else
1265     {
1266         if( speed > RATE_SLIDER_MAXIMUM )
1267         {
1268             sliderValue = speedSlider->maximum();
1269         }
1270         else
1271         {
1272             sliderValue = (int)( ( speed - 1.0 ) * RATE_SLIDER_LENGTH
1273                                         / ( RATE_SLIDER_MAXIMUM - 1.0 ) );
1274         }
1275     }
1276
1277     //Block signals to avoid feedback loop
1278     speedSlider->blockSignals( true );
1279     speedSlider->setValue( sliderValue );
1280     speedSlider->blockSignals( false );
1281 }
1282
1283 void SpeedControlWidget::updateRate( int sliderValue )
1284 {
1285     int rate;
1286
1287     if( sliderValue < 0.0 )
1288     {
1289         rate = (int)(INPUT_RATE_DEFAULT* RATE_SLIDER_LENGTH /
1290             ( sliderValue * ( 1.0 - RATE_SLIDER_MINIMUM ) + RATE_SLIDER_LENGTH ));
1291     }
1292     else
1293     {
1294         rate = (int)(INPUT_RATE_DEFAULT* RATE_SLIDER_LENGTH /
1295             ( sliderValue * ( RATE_SLIDER_MAXIMUM - 1.0 ) + RATE_SLIDER_LENGTH ));
1296     }
1297
1298     THEMIM->getIM()->setRate(rate);
1299 }
1300
1301 void SpeedControlWidget::resetRate()
1302 {
1303     THEMIM->getIM()->setRate(INPUT_RATE_DEFAULT);
1304 }