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