/*****************************************************************************
* Controller.cpp : Controller for the main interface
****************************************************************************
- * Copyright ( C ) 2006-2008 the VideoLAN team
+ * Copyright (C) 2006-2009 the VideoLAN team
* $Id$
*
- * Authors: Clément Stenac <zorglub@videolan.org>
- * Jean-Baptiste Kempf <jb@videolan.org>
- * Rafaël Carré <funman@videolanorg>
+ * Authors: Jean-Baptiste Kempf <jb@videolan.org>
* Ilkka Ollakka <ileoo@videolan.org>
*
* This program is free software; you can redistribute it and/or modify
#endif
#include <vlc_vout.h>
+#include <vlc_keys.h>
-#include "dialogs_provider.hpp"
+#include "components/controller.hpp"
+#include "components/controller_widget.hpp"
#include "components/interface_widgets.hpp"
-#include "main_interface.hpp"
+
+#include "dialogs_provider.hpp" /* Opening Dialogs */
#include "input_manager.hpp"
-#include "menus.hpp"
-#include "util/input_slider.hpp"
-#include "util/customwidgets.hpp"
+#include "actions_manager.hpp"
+
+#include "util/input_slider.hpp" /* InputSlider */
+#include "util/customwidgets.hpp" /* qEventToKey */
-#include <QLabel>
#include <QSpacerItem>
-#include <QCursor>
-#include <QToolButton>
#include <QToolButton>
#include <QHBoxLayout>
-#include <QMenu>
-#include <QPalette>
-#include <QResizeEvent>
-#include <QDate>
#include <QSignalMapper>
+#include <QTimer>
-#define I_PLAY_TOOLTIP N_("Play\nIf the playlist is empty, open a media")
+//#define DEBUG_LAYOUT 1
/**********************************************************************
* TEH controls
**********************************************************************/
/******
- * This is an abstract Toolbar/Controller
- * This has helper to create any toolbar, any buttons and to manage the actions
- *
- *****/
-AbstractController::AbstractController( intf_thread_t * _p_i ) : QFrame( NULL )
+ * This is an abstract Toolbar/Controller
+ * This has helper to create any toolbar, any buttons and to manage the actions
+ *
+ *****/
+AbstractController::AbstractController( intf_thread_t * _p_i, QWidget *_parent )
+ : QFrame( _parent )
{
p_intf = _p_i;
-
- /* We need one layout. An controller without layout is stupid with
- current architecture */
- controlLayout = new QGridLayout( this );
+ advControls = NULL;
/* Main action provider */
- toolbarActionsMapper = new QSignalMapper();
+ toolbarActionsMapper = new QSignalMapper( this );
CONNECT( toolbarActionsMapper, mapped( int ),
- this, doAction( int ) );
+ ActionsManager::getInstance( p_intf ), doAction( int ) );
CONNECT( THEMIM->getIM(), statusChanged( int ), this, setStatus( int ) );
+
+ setSizePolicy( QSizePolicy::MinimumExpanding, QSizePolicy::Fixed );
}
+/* Reemit some signals on status Change to activate some buttons */
void AbstractController::setStatus( int status )
{
bool b_hasInput = THEMIM->getIM()->hasInput();
emit inputIsRecordable( b_hasInput &&
var_GetBool( THEMIM->getInput(), "can-record" ) );
+
+ emit inputIsTrickPlayable( b_hasInput &&
+ var_GetBool( THEMIM->getInput(), "can-rewind" ) );
}
+/* Generic button setup */
void AbstractController::setupButton( QAbstractButton *aButton )
{
static QSizePolicy sizePolicy( QSizePolicy::Fixed, QSizePolicy::Fixed );
aButton->setFocusPolicy( Qt::NoFocus );
}
+/* Open the generic config line for the toolbar, parse it
+ * and create the widgets accordingly */
+void AbstractController::parseAndCreate( const QString& config,
+ QBoxLayout *controlLayout )
+{
+ QStringList list = config.split( ";", QString::SkipEmptyParts ) ;
+ for( int i = 0; i < list.size(); i++ )
+ {
+ QStringList list2 = list.at( i ).split( "-" );
+ if( list2.size() < 1 )
+ {
+ msg_Warn( p_intf, "Parsing error. Report this" );
+ continue;
+ }
+
+ bool ok;
+ int i_option = WIDGET_NORMAL;
+ buttonType_e i_type = (buttonType_e)list2.at( 0 ).toInt( &ok );
+ if( !ok )
+ {
+ msg_Warn( p_intf, "Parsing error 0. Please report this" );
+ continue;
+ }
+
+ if( list2.size() > 1 )
+ {
+ i_option = list2.at( 1 ).toInt( &ok );
+ if( !ok )
+ {
+ msg_Warn( p_intf, "Parsing error 1. Please report this" );
+ continue;
+ }
+ }
+
+ createAndAddWidget( controlLayout, -1, i_type, i_option );
+ }
+}
+
+void AbstractController::createAndAddWidget( QBoxLayout *controlLayout,
+ int i_index,
+ buttonType_e i_type,
+ int i_option )
+{
+ /* Special case for SPACERS, who aren't QWidgets */
+ if( i_type == WIDGET_SPACER )
+ {
+ controlLayout->insertSpacing( i_index, 12 );
+ return;
+ }
+
+ if( i_type == WIDGET_SPACER_EXTEND )
+ {
+ controlLayout->insertStretch( i_index, 12 );
+ return;
+ }
+
+ QWidget *widg = createWidget( i_type, i_option );
+ if( !widg ) return;
+
+ controlLayout->insertWidget( i_index, widg );
+}
+
+
#define CONNECT_MAP( a ) CONNECT( a, clicked(), toolbarActionsMapper, map() )
#define SET_MAPPING( a, b ) toolbarActionsMapper->setMapping( a , b )
#define CONNECT_MAP_SET( a, b ) \
CONNECT_MAP( a ); \
SET_MAPPING( a, b );
-#define BUTTON_SET_BAR( button, image, tooltip ) \
+#define BUTTON_SET_BAR( a_button ) \
+ a_button->setToolTip( qtr( tooltipL[button] ) ); \
+ a_button->setIcon( QIcon( iconL[button] ) );
+#define BUTTON_SET_BAR2( button, image, tooltip ) \
button->setToolTip( tooltip ); \
button->setIcon( QIcon( ":/"#image ) );
CONNECT( this, inputExists( bool ), a, setEnabled( bool ) ); \
a->setEnabled( THEMIM->getIM()->hasInput() ); /* TODO: is this necessary? when input is started before the interface? */
-QWidget *AbstractController::createWidget( buttonType_e button, bool b_flat,
- bool b_big, bool b_shiny )
+#define NORMAL_BUTTON( name ) \
+ QToolButton * name ## Button = new QToolButton; \
+ setupButton( name ## Button ); \
+ CONNECT_MAP_SET( name ## Button, name ## _ACTION ); \
+ BUTTON_SET_BAR( name ## Button ); \
+ widget = name ## Button;
+
+QWidget *AbstractController::createWidget( buttonType_e button, int options )
{
+
+ bool b_flat = options & WIDGET_FLAT;
+ bool b_big = options & WIDGET_BIG;
+ bool b_shiny = options & WIDGET_SHINY;
+ bool b_special = false;
+
QWidget *widget = NULL;
switch( button )
{
case PLAY_BUTTON: {
PlayButton *playButton = new PlayButton;
setupButton( playButton );
- BUTTON_SET_BAR( playButton, play_b, qtr( I_PLAY_TOOLTIP ) );
+ BUTTON_SET_BAR( playButton );
CONNECT_MAP_SET( playButton, PLAY_ACTION );
CONNECT( this, inputPlaying( bool ),
playButton, updateButton( bool ));
}
break;
case STOP_BUTTON:{
- QToolButton *stopButton = new QToolButton;
- setupButton( stopButton );
- CONNECT_MAP_SET( stopButton, STOP_ACTION );
- BUTTON_SET_BAR( stopButton, stop_b, qtr( "Stop playback" ) );
- widget = stopButton;
+ NORMAL_BUTTON( STOP );
+ }
+ break;
+ case OPEN_BUTTON:{
+ NORMAL_BUTTON( OPEN );
}
break;
case PREVIOUS_BUTTON:{
- QToolButton *prevButton = new QToolButton;
- setupButton( prevButton );
- CONNECT_MAP_SET( prevButton, PREVIOUS_ACTION );
- BUTTON_SET_BAR( prevButton, previous_b,
- qtr( "Previous media in the playlist" ) );
- widget = prevButton;
+ NORMAL_BUTTON( PREVIOUS );
}
break;
- case NEXT_BUTTON:
- {
- QToolButton *nextButton = new QToolButton;
- setupButton( nextButton );
- CONNECT_MAP_SET( nextButton, NEXT_ACTION );
- BUTTON_SET_BAR( nextButton, next_b,
- qtr( "Next media in the playlist" ) );
- widget = nextButton;
+ case NEXT_BUTTON: {
+ NORMAL_BUTTON( NEXT );
}
break;
case SLOWER_BUTTON:{
- QToolButton *slowerButton = new QToolButton;
- setupButton( slowerButton );
- CONNECT_MAP_SET( slowerButton, SLOWER_ACTION );
- BUTTON_SET_BAR( slowerButton, slower, qtr( "Slower" ) );
- ENABLE_ON_INPUT( slowerButton );
- widget = slowerButton;
+ NORMAL_BUTTON( SLOWER );
+ ENABLE_ON_INPUT( SLOWERButton );
}
break;
case FASTER_BUTTON:{
- QToolButton *fasterButton = new QToolButton;
- setupButton( fasterButton );
- CONNECT_MAP_SET( fasterButton, SLOWER_ACTION );
- BUTTON_SET_BAR( fasterButton, faster, qtr( "Faster" ) );
- ENABLE_ON_INPUT( fasterButton );
- widget = fasterButton;
+ NORMAL_BUTTON( FASTER );
+ ENABLE_ON_INPUT( FASTERButton );
}
break;
-#if 0
case FRAME_BUTTON: {
- QToolButton *frameButton = new QToolButton( "Fr" );
- setupButton( frameButton );
- BUTTON_SET_BAR( frameButton, "", qtr( "Frame by frame" ) );
- ENABLE_ON_INPUT( frameButton );
- widget = frameButton;
+ NORMAL_BUTTON( FRAME );
+ ENABLE_ON_VIDEO( FRAMEButton );
}
break;
-#endif
- case FULLSCREEN_BUTTON:{
- QToolButton *fullscreenButton = new QToolButton;
- setupButton( fullscreenButton );
- CONNECT_MAP_SET( fullscreenButton, FULLSCREEN_ACTION );
- BUTTON_SET_BAR( fullscreenButton, fullscreen,
- qtr( "Toggle the video in fullscreen" ) );
- ENABLE_ON_VIDEO( fullscreenButton );
- widget = fullscreenButton;
+ case FULLSCREEN_BUTTON:
+ case DEFULLSCREEN_BUTTON:
+ {
+ NORMAL_BUTTON( FULLSCREEN );
+ ENABLE_ON_VIDEO( FULLSCREENButton );
}
break;
case EXTENDED_BUTTON:{
- QToolButton *extSettingsButton = new QToolButton;
- setupButton( extSettingsButton );
- CONNECT_MAP_SET( extSettingsButton, EXTENDED_ACTION );
- BUTTON_SET_BAR( extSettingsButton, extended,
- qtr( "Show extended settings" ) );
- widget = extSettingsButton;
+ NORMAL_BUTTON( EXTENDED );
}
break;
case PLAYLIST_BUTTON:{
- QToolButton *playlistButton = new QToolButton;
- setupButton( playlistButton );
- CONNECT_MAP_SET( playlistButton, PLAYLIST_ACTION );
- BUTTON_SET_BAR( playlistButton, playlist,
- qtr( "Show playlist" ) );
- widget = playlistButton;
+ NORMAL_BUTTON( PLAYLIST );
}
break;
case SNAPSHOT_BUTTON:{
- QToolButton *snapshotButton = new QToolButton;
- setupButton( snapshotButton );
- CONNECT_MAP_SET( snapshotButton, SNAPSHOT_ACTION );
- BUTTON_SET_BAR( snapshotButton, snapshot, qtr( "Take a snapshot" ) );
- ENABLE_ON_VIDEO( snapshotButton );
- widget = snapshotButton;
+ NORMAL_BUTTON( SNAPSHOT );
+ ENABLE_ON_VIDEO( SNAPSHOTButton );
}
break;
case RECORD_BUTTON:{
QToolButton *recordButton = new QToolButton;
setupButton( recordButton );
CONNECT_MAP_SET( recordButton, RECORD_ACTION );
- BUTTON_SET_BAR( recordButton, record, qtr( "Record" ) );
+ BUTTON_SET_BAR( recordButton );
ENABLE_ON_INPUT( recordButton );
+ recordButton->setCheckable( true );
+ CONNECT( THEMIM->getIM(), recordingStateChanged( bool ),
+ recordButton, setChecked( bool ) );
widget = recordButton;
}
break;
case ATOB_BUTTON: {
AtoB_Button *ABButton = new AtoB_Button;
setupButton( ABButton );
- BUTTON_SET_BAR( ABButton, atob_nob, qtr( "Loop from point A to point "
- "B continuously.\nClick to set point A" ) );
+ ABButton->setShortcut( qtr("Shift+L") );
+ BUTTON_SET_BAR( ABButton );
ENABLE_ON_INPUT( ABButton );
CONNECT_MAP_SET( ABButton, ATOB_ACTION );
CONNECT( THEMIM->getIM(), AtoBchanged( bool, bool),
break;
case INPUT_SLIDER: {
InputSlider *slider = new InputSlider( Qt::Horizontal, NULL );
+
/* Update the position when the IM has changed */
- CONNECT( THEMIM->getIM(), positionUpdated( float, int, int ),
- slider, setPosition( float, int, int ) );
+ CONNECT( THEMIM->getIM(), positionUpdated( float, int64_t, int ),
+ slider, setPosition( float, int64_t, int ) );
/* And update the IM, when the position has changed */
CONNECT( slider, sliderDragged( float ),
- THEMIM->getIM(), sliderUpdate( float ) );
+ THEMIM->getIM(), sliderUpdate( float ) );
widget = slider;
}
break;
widget = telexFrame();
widget->hide();
break;
+ case VOLUME_SPECIAL:
+ b_special = true;
case VOLUME:
{
- SoundWidget *snd = new SoundWidget( p_intf, b_shiny );
+ SoundWidget *snd = new SoundWidget( this, p_intf, b_shiny, b_special );
widget = snd;
}
break;
+ case TIME_LABEL:
+ {
+ TimeLabel *timeLabel = new TimeLabel( p_intf );
+ widget = timeLabel;
+ }
+ break;
+ case SPLITTER:
+ {
+ QFrame *line = new QFrame;
+ line->setFrameShape( QFrame::VLine );
+ line->setFrameShadow( QFrame::Raised );
+ line->setLineWidth( 0 );
+ line->setMidLineWidth( 1 );
+ widget = line;
+ }
+ break;
+ case ADVANCED_CONTROLLER:
+ {
+ advControls = new AdvControlsWidget( p_intf, this );
+ widget = advControls;
+ }
+ break;
+ case REVERSE_BUTTON:{
+ QToolButton *reverseButton = new QToolButton;
+ setupButton( reverseButton );
+ CONNECT_MAP_SET( reverseButton, REVERSE_ACTION );
+ BUTTON_SET_BAR( reverseButton );
+ reverseButton->setCheckable( true );
+ /* You should, of COURSE change this to the correct event,
+ when/if we have one, that tells us if trickplay is possible . */
+ CONNECT( this, inputIsTrickPlayable( bool ), reverseButton, setVisible( bool ) );
+ reverseButton->setVisible( false );
+ widget = reverseButton;
+ }
+ break;
+ case SKIP_BACK_BUTTON: {
+ NORMAL_BUTTON( SKIP_BACK );
+ ENABLE_ON_INPUT( SKIP_BACKButton );
+ }
+ break;
+ case SKIP_FW_BUTTON: {
+ NORMAL_BUTTON( SKIP_FW );
+ ENABLE_ON_INPUT( SKIP_FWButton );
+ }
+ break;
+ case QUIT_BUTTON: {
+ NORMAL_BUTTON( QUIT );
+ }
+ break;
+ case RANDOM_BUTTON: {
+ NORMAL_BUTTON( RANDOM );
+ RANDOMButton->setCheckable( true );
+ RANDOMButton->setChecked( var_GetBool( THEPL, "random" ) );
+ CONNECT( THEMIM, randomChanged( bool ),
+ RANDOMButton, setChecked( bool ) );
+ }
+ break;
+ case LOOP_BUTTON:{
+ LoopButton *loopButton = new LoopButton;
+ setupButton( loopButton );
+ loopButton->setToolTip( qtr( "Click to toggle between loop one, loop all" ) );
+ loopButton->setCheckable( true );
+ loopButton->updateIcons( NORMAL );
+ CONNECT( THEMIM, repeatLoopChanged( int ), loopButton, updateIcons( int ) );
+ CONNECT( loopButton, clicked(), THEMIM, loopRepeatLoopStatus() );
+ widget = loopButton;
+ }
+ break;
+ case INFO_BUTTON: {
+ NORMAL_BUTTON( INFO );
+ }
+ break;
default:
- msg_Warn( p_intf, "This should not happen" );
+ msg_Warn( p_intf, "This should not happen %i", button );
break;
}
/* Customize Buttons */
if( b_flat || b_big )
{
- QToolButton *tmpButton = qobject_cast<QToolButton *>(widget);
- if( tmpButton )
+ QFrame *frame = qobject_cast<QFrame *>(widget);
+ if( frame )
{
- if( b_flat )
- tmpButton->setAutoRaise( b_flat );
- if( b_big )
- {
- tmpButton->setFixedSize( QSize( 32, 32 ) );
- tmpButton->setIconSize( QSize( 26, 26 ) );
- }
+ QList<QToolButton *> allTButtons = frame->findChildren<QToolButton *>();
+ for( int i = 0; i < allTButtons.size(); i++ )
+ applyAttributes( allTButtons[i], b_flat, b_big );
+ }
+ else
+ {
+ QToolButton *tmpButton = qobject_cast<QToolButton *>(widget);
+ if( tmpButton )
+ applyAttributes( tmpButton, b_flat, b_big );
}
}
return widget;
}
+#undef NORMAL_BUTTON
-QWidget *AbstractController::discFrame()
+void AbstractController::applyAttributes( QToolButton *tmpButton, bool b_flat, bool b_big )
+{
+ if( tmpButton )
+ {
+ if( b_flat )
+ tmpButton->setAutoRaise( b_flat );
+ if( b_big )
+ {
+ tmpButton->setFixedSize( QSize( 32, 32 ) );
+ tmpButton->setIconSize( QSize( 26, 26 ) );
+ }
+ }
+}
+
+QFrame *AbstractController::discFrame()
{
/** Disc and Menus handling */
- QWidget *discFrame = new QWidget( this );
+ QFrame *discFrame = new QFrame( this );
QHBoxLayout *discLayout = new QHBoxLayout( discFrame );
discLayout->setSpacing( 0 ); discLayout->setMargin( 0 );
QToolButton *prevSectionButton = new QToolButton( discFrame );
setupButton( prevSectionButton );
- BUTTON_SET_BAR( prevSectionButton, dvd_prev,
+ BUTTON_SET_BAR2( prevSectionButton, toolbar/dvd_prev,
qtr("Previous Chapter/Title" ) );
discLayout->addWidget( prevSectionButton );
QToolButton *menuButton = new QToolButton( discFrame );
setupButton( menuButton );
discLayout->addWidget( menuButton );
- BUTTON_SET_BAR( menuButton, dvd_menu, qtr( "Menu" ) );
+ BUTTON_SET_BAR2( menuButton, toolbar/dvd_menu, qtr( "Menu" ) );
QToolButton *nextSectionButton = new QToolButton( discFrame );
setupButton( nextSectionButton );
discLayout->addWidget( nextSectionButton );
- BUTTON_SET_BAR( nextSectionButton, dvd_next,
+ BUTTON_SET_BAR2( nextSectionButton, toolbar/dvd_next,
qtr("Next Chapter/Title" ) );
-
/* Change the navigation button display when the IM
navigation changes */
CONNECT( THEMIM->getIM(), titleChanged( bool ),
return discFrame;
}
-QWidget *AbstractController::telexFrame()
+QFrame *AbstractController::telexFrame()
{
/**
* Telextext QFrame
**/
- TeletextController *telexFrame = new TeletextController;
+ QFrame *telexFrame = new QFrame( this );
QHBoxLayout *telexLayout = new QHBoxLayout( telexFrame );
telexLayout->setSpacing( 0 ); telexLayout->setMargin( 0 );
CONNECT( THEMIM->getIM(), teletextPossible( bool ),
/* On/Off button */
QToolButton *telexOn = new QToolButton;
- telexFrame->telexOn = telexOn;
setupButton( telexOn );
- BUTTON_SET_BAR( telexOn, tv, qtr( "Teletext Activation" ) );
+ BUTTON_SET_BAR2( telexOn, toolbar/tv, qtr( "Teletext Activation" ) );
+ telexOn->setEnabled( false );
+ telexOn->setCheckable( true );
+
telexLayout->addWidget( telexOn );
/* Teletext Activation and set */
CONNECT( telexOn, clicked( bool ),
THEMIM->getIM(), activateTeletext( bool ) );
- CONNECT( THEMIM->getIM(), teletextActivated( bool ),
- telexFrame, enableTeletextButtons( bool ) );
-
+ CONNECT( THEMIM->getIM(), teletextPossible( bool ),
+ telexOn, setEnabled( bool ) );
/* Transparency button */
QToolButton *telexTransparent = new QToolButton;
- telexFrame->telexTransparent = telexTransparent;
setupButton( telexTransparent );
- BUTTON_SET_BAR( telexTransparent, tvtelx,
- qtr( "Toggle Transparency " ) );
+ BUTTON_SET_BAR2( telexTransparent, toolbar/tvtelx,
+ qtr( "Toggle Transparency " ) );
telexTransparent->setEnabled( false );
+ telexTransparent->setCheckable( true );
telexLayout->addWidget( telexTransparent );
/* Transparency change and set */
CONNECT( telexTransparent, clicked( bool ),
THEMIM->getIM(), telexSetTransparency( bool ) );
CONNECT( THEMIM->getIM(), teletextTransparencyActivated( bool ),
- telexFrame, toggleTeletextTransparency( bool ) );
+ telexTransparent, setChecked( bool ) );
/* Page setting */
- QSpinBox *telexPage = new QSpinBox;
- telexFrame->telexPage = telexPage;
+ QSpinBox *telexPage = new QSpinBox( telexFrame );
telexPage->setRange( 0, 999 );
telexPage->setValue( 100 );
telexPage->setAccelerated( true );
CONNECT( THEMIM->getIM(), newTelexPageSet( int ),
telexPage, setValue( int ) );
+ CONNECT( THEMIM->getIM(), teletextActivated( bool ), telexPage, setEnabled( bool ) );
+ CONNECT( THEMIM->getIM(), teletextActivated( bool ), telexTransparent, setEnabled( bool ) );
+ CONNECT( THEMIM->getIM(), teletextActivated( bool ), telexOn, setChecked( bool ) );
return telexFrame;
}
#undef CONNECT_MAP
#undef SET_MAPPING
+#undef CONNECT_MAP_SET
#undef BUTTON_SET_BAR
+#undef BUTTON_SET_BAR2
+#undef ENABLE_ON_VIDEO
+#undef ENABLE_ON_INPUT
-SoundWidget::SoundWidget( intf_thread_t * _p_intf, bool b_shiny )
- : b_my_volume( false )
-{
- p_intf = _p_intf;
- QHBoxLayout *layout = new QHBoxLayout( this );
- layout->setSpacing( 0 ); layout->setMargin( 0 );
- hVolLabel = new VolumeClickHandler( p_intf, this );
-
- volMuteLabel = new QLabel;
- volMuteLabel->setPixmap( QPixmap( ":/volume-medium" ) );
- volMuteLabel->installEventFilter( hVolLabel );
- layout->addWidget( volMuteLabel );
-
- if( b_shiny )
- {
- volumeSlider = new SoundSlider( this,
- config_GetInt( p_intf, "volume-step" ),
- config_GetInt( p_intf, "qt-volume-complete" ),
- config_GetPsz( p_intf, "qt-slider-colours" ) );
- }
- else
- {
- volumeSlider = new QSlider( this );
- volumeSlider->setOrientation( Qt::Horizontal );
- }
- volumeSlider->setMaximumSize( QSize( 200, 40 ) );
- volumeSlider->setMinimumSize( QSize( 85, 30 ) );
- volumeSlider->setFocusPolicy( Qt::NoFocus );
- layout->addWidget( volumeSlider );
-
- /* Set the volume from the config */
- volumeSlider->setValue( ( config_GetInt( p_intf, "volume" ) ) *
- VOLUME_MAX / (AOUT_VOLUME_MAX/2) );
-
- /* Force the update at build time in order to have a muted icon if needed */
- updateVolume( volumeSlider->value() );
-
- /* Volume control connection */
- CONNECT( volumeSlider, valueChanged( int ), this, updateVolume( int ) );
- CONNECT( THEMIM, volumeChanged( void ), this, updateVolume( void ) );
-}
-
-void SoundWidget::updateVolume( int i_sliderVolume )
-{
- if( !b_my_volume )
- {
- int i_res = i_sliderVolume * (AOUT_VOLUME_MAX / 2) / VOLUME_MAX;
- aout_VolumeSet( p_intf, i_res );
- }
- if( i_sliderVolume == 0 )
- {
- volMuteLabel->setPixmap( QPixmap(":/volume-muted" ) );
- volMuteLabel->setToolTip( qtr( "Unmute" ) );
- return;
- }
-
- if( i_sliderVolume < VOLUME_MAX / 3 )
- volMuteLabel->setPixmap( QPixmap( ":/volume-low" ) );
- else if( i_sliderVolume > (VOLUME_MAX * 2 / 3 ) )
- volMuteLabel->setPixmap( QPixmap( ":/volume-high" ) );
- else volMuteLabel->setPixmap( QPixmap( ":/volume-medium" ) );
- volMuteLabel->setToolTip( qtr( "Mute" ) );
-}
-
-void SoundWidget::updateVolume()
-{
- /* Audio part */
- audio_volume_t i_volume;
- aout_VolumeGet( p_intf, &i_volume );
- i_volume = ( i_volume * VOLUME_MAX )/ (AOUT_VOLUME_MAX/2);
- int i_gauge = volumeSlider->value();
- b_my_volume = false;
- if( i_volume - i_gauge > 1 || i_gauge - i_volume > 1 )
- {
- b_my_volume = true;
- volumeSlider->setValue( i_volume );
- b_my_volume = false;
- }
-}
-
-void TeletextController::toggleTeletextTransparency( bool b_transparent )
-{
- telexTransparent->setIcon( b_transparent ? QIcon( ":/tvtelx" )
- : QIcon( ":/tvtelx-trans" ) );
-}
-
-void TeletextController::enableTeletextButtons( bool b_enabled )
-{
- telexOn->setChecked( b_enabled );
- telexTransparent->setEnabled( b_enabled );
- telexPage->setEnabled( b_enabled );
-}
-
-void PlayButton::updateButton( bool b_playing )
-{
- setIcon( b_playing ? QIcon( ":/pause_b" ) : QIcon( ":/play_b" ) );
- setToolTip( b_playing ? qtr( "Pause the playback" )
- : qtr( I_PLAY_TOOLTIP ) );
-}
-
-void AtoB_Button::setIcons( bool timeA, bool timeB )
-{
- if( !timeA && !timeB)
- {
- setIcon( QIcon( ":/atob_nob" ) );
- setToolTip( qtr( "Loop from point A to point B continuously\n"
- "Click to set point A" ) );
- }
- else if( timeA && !timeB )
- {
- setIcon( QIcon( ":/atob_noa" ) );
- setToolTip( qtr( "Click to set point B" ) );
- }
- else if( timeA && timeB )
- {
- setIcon( QIcon( ":/atob" ) );
- setToolTip( qtr( "Stop the A to B loop" ) );
- }
-}
-
-
-//* Actions */
-void AbstractController::doAction( int id_action )
-{
- switch( id_action )
- {
- case PLAY_ACTION:
- play(); break;
- case PREVIOUS_ACTION:
- prev(); break;
- case NEXT_ACTION:
- next(); break;
- case STOP_ACTION:
- stop(); break;
- case SLOWER_ACTION:
- slower(); break;
- case FASTER_ACTION:
- faster(); break;
- case FULLSCREEN_ACTION:
- fullscreen(); break;
- case EXTENDED_ACTION:
- extSettings(); break;
- case PLAYLIST_ACTION:
- playlist(); break;
- case SNAPSHOT_ACTION:
- snapshot(); break;
- case RECORD_ACTION:
- record(); break;
- case ATOB_ACTION:
- THEMIM->getIM()->setAtoB(); break;
- default:
- msg_Dbg( p_intf, "Action: %i", id_action );
- break;
- }
-}
-
-void AbstractController::stop()
-{
- THEMIM->stop();
-}
-
-void AbstractController::play()
-{
- if( THEPL->current.i_size == 0 )
- {
- /* The playlist is empty, open a file requester */
- THEDP->openFileDialog();
- return;
- }
- THEMIM->togglePlayPause();
-}
-
-void AbstractController::prev()
-{
- THEMIM->prev();
-}
-
-void AbstractController::next()
-{
- THEMIM->next();
-}
-
-/**
- * TODO
- * This functions toggle the fullscreen mode
- * If there is no video, it should first activate Visualisations...
- * This has also to be fixed in enableVideo()
- */
-void AbstractController::fullscreen()
-{
- vout_thread_t *p_vout =
- (vout_thread_t *)vlc_object_find( p_intf, VLC_OBJECT_VOUT, FIND_ANYWHERE );
- if( p_vout)
- {
- var_SetBool( p_vout, "fullscreen", !var_GetBool( p_vout, "fullscreen" ) );
- vlc_object_release( p_vout );
- }
-}
-
-void AbstractController::snapshot()
-{
- vout_thread_t *p_vout =
- (vout_thread_t *)vlc_object_find( p_intf, VLC_OBJECT_VOUT, FIND_ANYWHERE );
- if( p_vout )
- {
- vout_Control( p_vout, VOUT_SNAPSHOT );
- vlc_object_release( p_vout );
- }
-}
-
-
-void AbstractController::extSettings()
-{
- THEDP->extendedDialog();
-}
-
-void AbstractController::slower()
-{
- THEMIM->getIM()->slower();
-}
-
-void AbstractController::faster()
-{
- THEMIM->getIM()->faster();
-}
-
-void AbstractController::playlist()
-{
- if( p_intf->p_sys->p_mi ) p_intf->p_sys->p_mi->togglePlaylist();
-}
-
-void AbstractController::record()
-{
- input_thread_t *p_input = THEMIM->getInput();
- if( p_input )
- {
- /* This method won't work fine if the stream can't be cut anywhere */
- const bool b_recording = var_GetBool( p_input, "record" );
- var_SetBool( p_input, "record", !b_recording );
-#if 0
- else
- {
- /* 'record' access-filter is not loaded, we open Save dialog */
- input_item_t *p_item = input_GetItem( p_input );
- if( !p_item )
- return;
-
- char *psz = input_item_GetURI( p_item );
- if( psz )
- THEDP->streamingDialog( NULL, psz, true );
- }
-#endif
- }
-}
-
-#if 0
-//TODO Frame by frame function
-void AbstractController::frame(){}
-#endif
-
-
+#include <QHBoxLayout>
/*****************************
* DA Control Widget !
*****************************/
ControlsWidget::ControlsWidget( intf_thread_t *_p_i,
- bool b_advControls ) :
- AbstractController( _p_i )
+ bool b_advControls,
+ QWidget *_parent ) :
+ AbstractController( _p_i, _parent )
{
- setSizePolicy( QSizePolicy::Preferred , QSizePolicy::Maximum );
-
/* advanced Controls handling */
b_advancedVisible = b_advControls;
+#if DEBUG_LAYOUT
+ setStyleSheet( " background: red ");
+#endif
- advControls = new AdvControlsWidget( p_intf );
- if( !b_advancedVisible ) advControls->hide();
-
+ QVBoxLayout *controlLayout = new QVBoxLayout( this );
+ controlLayout->setContentsMargins( 4, 1, 4, 0 );
controlLayout->setSpacing( 0 );
- controlLayout->setLayoutMargins( 7, 5, 7, 3, 6 );
-
- controlLayout->addWidget( createWidget( INPUT_SLIDER ), 0, 1, 1, 18 );
- controlLayout->addWidget( createWidget( SLOWER_BUTTON, true ), 0, 0 );
- controlLayout->addWidget( createWidget( FASTER_BUTTON, true ), 0, 19 );
-
- controlLayout->addWidget( createWidget( MENU_BUTTONS ), 1, 8, 2, 3 );
- controlLayout->addWidget( createWidget( TELETEXT_BUTTONS ), 1, 8, 2, 5, Qt::AlignBottom );
-
- controlLayout->addWidget( createWidget( PLAY_BUTTON, false, true ), 2, 0, 2, 2, Qt::AlignBottom );
- controlLayout->setColumnMinimumWidth( 2, 10 );
- controlLayout->setColumnStretch( 2, 0 );
+ QHBoxLayout *controlLayout1 = new QHBoxLayout;
+ controlLayout1->setSpacing( 0 ); controlLayout1->setMargin( 0 );
- controlLayout->addWidget( createWidget( PREVIOUS_BUTTON ), 3, 3, Qt::AlignBottom );
- controlLayout->addWidget( createWidget( STOP_BUTTON ), 3, 4, Qt::AlignBottom );
- controlLayout->addWidget( createWidget( NEXT_BUTTON ), 3, 5, Qt::AlignBottom );
+ QString line1 = getSettings()->value( "MainToolbar1", MAIN_TB1_DEFAULT )
+ .toString();
+ parseAndCreate( line1, controlLayout1 );
- /* Column 6 is unused */
- controlLayout->setColumnStretch( 6, 0 );
- controlLayout->setColumnStretch( 7, 0 );
- controlLayout->setColumnMinimumWidth( 7, 10 );
+ QHBoxLayout *controlLayout2 = new QHBoxLayout;
+ controlLayout2->setSpacing( 0 ); controlLayout2->setMargin( 0 );
+ QString line2 = getSettings()->value( "MainToolbar2", MAIN_TB2_DEFAULT )
+ .toString();
+ parseAndCreate( line2, controlLayout2 );
- controlLayout->addWidget( createWidget( FULLSCREEN_BUTTON ), 3, 8, Qt::AlignBottom );
- controlLayout->addWidget( createWidget( PLAYLIST_BUTTON ), 3, 9, Qt::AlignBottom );
- controlLayout->addWidget( createWidget( EXTENDED_BUTTON ), 3, 10, Qt::AlignBottom );
- controlLayout->setColumnStretch( 11, 0 ); /* telex alignment */
+ if( !b_advancedVisible && advControls ) advControls->hide();
- controlLayout->setColumnStretch( 12, 0 );
- controlLayout->setColumnMinimumWidth( 12, 10 );
-
- controlLayout->addWidget( advControls, 3, 13, 1, 3, Qt::AlignBottom );
-
- controlLayout->setColumnStretch( 16, 10 );
- controlLayout->setColumnMinimumWidth( 16, 10 );
-
- controlLayout->addWidget( createWidget( VOLUME, false, false, true ),
- 3, 17, 1, 3, Qt::AlignBottom );
+ controlLayout->addLayout( controlLayout1 );
+ controlLayout->addLayout( controlLayout2 );
}
ControlsWidget::~ControlsWidget()
void ControlsWidget::toggleAdvanced()
{
- if( advControls && !b_advancedVisible )
+ if( !advControls ) return;
+
+ if( !b_advancedVisible )
{
advControls->show();
b_advancedVisible = true;
emit advancedControlsToggled( b_advancedVisible );
}
-AdvControlsWidget::AdvControlsWidget( intf_thread_t *_p_i ) :
- AbstractController( _p_i )
+AdvControlsWidget::AdvControlsWidget( intf_thread_t *_p_i, QWidget *_parent ) :
+ AbstractController( _p_i, _parent )
{
+ controlLayout = new QHBoxLayout( this );
controlLayout->setMargin( 0 );
controlLayout->setSpacing( 0 );
+#if DEBUG_LAYOUT
+ setStyleSheet( " background: orange ");
+#endif
- controlLayout->addWidget( createWidget( RECORD_BUTTON ), 0, 0 );
- controlLayout->addWidget( createWidget( SNAPSHOT_BUTTON ), 0, 1 );
- controlLayout->addWidget( createWidget( ATOB_BUTTON ), 0, 2 );
-}
-AdvControlsWidget::~AdvControlsWidget()
-{}
+ QString line = getSettings()->value( "AdvToolbar", ADV_TB_DEFAULT )
+ .toString();
+ parseAndCreate( line, controlLayout );
+}
+InputControlsWidget::InputControlsWidget( intf_thread_t *_p_i, QWidget *_parent ) :
+ AbstractController( _p_i, _parent )
+{
+ controlLayout = new QHBoxLayout( this );
+ controlLayout->setMargin( 0 );
+ controlLayout->setSpacing( 0 );
+#if DEBUG_LAYOUT
+ setStyleSheet( " background: green ");
+#endif
+ QString line = getSettings()->value( "InputToolbar", INPT_TB_DEFAULT ).toString();
+ parseAndCreate( line, controlLayout );
+}
/**********************************************************************
* Fullscrenn control widget
**********************************************************************/
-FullscreenControllerWidget::FullscreenControllerWidget( intf_thread_t *_p_i )
- : AbstractController( _p_i )
+FullscreenControllerWidget::FullscreenControllerWidget( intf_thread_t *_p_i, QWidget *_parent )
+ : AbstractController( _p_i, _parent )
{
i_mouse_last_x = -1;
i_mouse_last_y = -1;
#endif
b_fullscreen = false;
i_hide_timeout = 1;
- p_vout = NULL;
+ i_screennumber = -1;
+
+ vout.clear();
setWindowFlags( Qt::ToolTip );
setMinimumWidth( 600 );
setFrameStyle( QFrame::Sunken );
setSizePolicy( QSizePolicy::Minimum, QSizePolicy::Minimum );
- controlLayout->setLayoutMargins( 5, 2, 5, 2, 5 );
+ QVBoxLayout *controlLayout2 = new QVBoxLayout( this );
+ controlLayout2->setContentsMargins( 4, 6, 4, 2 );
/* First line */
- controlLayout->addWidget( createWidget( SLOWER_BUTTON ), 0, 0 );
- controlLayout->addWidget( createWidget( INPUT_SLIDER ), 0, 1, 1, 13 );
- controlLayout->addWidget( createWidget( FASTER_BUTTON ), 0, 14 );
-
- /* Second line */
- controlLayout->addWidget( createWidget( PLAY_BUTTON, false, true ), 1, 0, 1, 2 );
- controlLayout->addWidget( createWidget( PREVIOUS_BUTTON ), 1, 2 );
- controlLayout->addWidget( createWidget( STOP_BUTTON ), 1, 3 );
- controlLayout->addWidget( createWidget( NEXT_BUTTON ), 1, 4 );
+ InputControlsWidget *inputC = new InputControlsWidget( p_intf, this );
+ controlLayout2->addWidget( inputC );
- controlLayout->addWidget( createWidget( MENU_BUTTONS ), 1, 5 );
- controlLayout->addWidget( createWidget( TELETEXT_BUTTONS ), 1, 6 );
- QToolButton *fullscreenButton =
- qobject_cast<QToolButton *>(createWidget( FULLSCREEN_BUTTON ) );
- fullscreenButton->setIcon( QIcon( ":/defullscreen" ) );
- controlLayout->addWidget( fullscreenButton, 1, 7 );
-
- controlLayout->setColumnStretch( 9, 10 );
-
- TimeLabel *timeLabel = new TimeLabel( p_intf );
-
- controlLayout->addWidget( timeLabel, 1, 10 );
- controlLayout->addWidget( createWidget( VOLUME, false, false, true ),
- 1, 11, 1, 3 );
+ controlLayout = new QHBoxLayout;
+ QString line = getSettings()->value( "MainWindow/FSCtoolbar", FSC_TB_DEFAULT ).toString();
+ parseAndCreate( line, controlLayout );
+ controlLayout2->addLayout( controlLayout );
/* hiding timer */
p_hideTimer = new QTimer( this );
CONNECT( p_slowHideTimer, timeout(), this, slowHideFSC() );
#endif
- adjustSize (); /* need to get real width and height for moving */
+ vlc_mutex_init_recursive( &lock );
-#ifdef WIN32TRICK
- setWindowOpacity( 0.0 );
- b_fscHidden = true;
- adjustSize();
- show();
-#endif
+ DCONNECT( THEMIM->getIM(), voutListChanged( vout_thread_t **, int ),
+ this, setVoutList( vout_thread_t **, int ) );
+
+ /* First Move */
+ QRect rect1 = getSettings()->value( "FullScreen/screen" ).toRect();
+ QPoint pos1 = getSettings()->value( "FullScreen/pos" ).toPoint();
+ int number = var_InheritInteger( p_intf, "qt-fullscreen-screennumber" );
+ if( number == -1 || number > QApplication::desktop()->numScreens() )
+ number = QApplication::desktop()->screenNumber( p_intf->p_sys->p_mi );
+
+ QRect rect = QApplication::desktop()->screenGeometry( number );
+ if( rect == rect1 && rect.contains( pos1, true ) )
+ {
+ move( pos1 );
+ i_screennumber = number;
+ screenRes = QApplication::desktop()->screenGeometry(number);
+ }
+ else
+ {
+ centerFSC( number );
+ }
- vlc_mutex_init_recursive( &lock );
}
FullscreenControllerWidget::~FullscreenControllerWidget()
{
- getSettings()->beginGroup( "FullScreen" );
- getSettings()->setValue( "pos", pos() );
- getSettings()->endGroup();
- detachVout();
+ QPoint pos1 = pos();
+ QRect rect1 = QApplication::desktop()->screenGeometry( pos1 );
+ getSettings()->setValue( "FullScreen/pos", pos1 );
+ getSettings()->setValue( "FullScreen/screen", rect1 );
+
+ setVoutList( NULL, 0 );
vlc_mutex_destroy( &lock );
}
+void FullscreenControllerWidget::centerFSC( int number )
+{
+ screenRes = QApplication::desktop()->screenGeometry(number);
+
+ /* screen has changed, calculate new position */
+ QPoint pos = QPoint( screenRes.x() + (screenRes.width() / 2) - (sizeHint().width() / 2),
+ screenRes.y() + screenRes.height() - sizeHint().height());
+ move( pos );
+
+ i_screennumber = number;
+}
+
/**
* Show fullscreen controller
*/
void FullscreenControllerWidget::showFSC()
{
adjustSize();
- /* center down */
+
int number = QApplication::desktop()->screenNumber( p_intf->p_sys->p_mi );
- int totalCount = QApplication::desktop()->numScreens();
- QRect screenRes = QApplication::desktop()->screenGeometry(number);
- int offset_x = 0;
- int offset_y = 0;
- /* Loop all screens to get needed offset_x/y for
- * physical screen center.
- */
- for(int i=0; i <= totalCount ; i++)
- {
- QRect displayRect = QApplication::desktop()->screenGeometry(i);
- if (displayRect.width()+offset_x <= screenRes.x())
- {
- offset_x += displayRect.width();
- }
- if ( displayRect.height()+offset_y <= screenRes.y())
- {
- offset_y += displayRect.height();
- }
- }
- QPoint pos = QPoint( offset_x + (screenRes.width() / 2) - (width() / 2),
- offset_y + screenRes.height() - height());
- move( pos );
-#ifdef WIN32TRICK
- // after quiting and going to fs, we need to call show()
- if( isHidden() )
- show();
- if( b_fscHidden )
+
+ if( number != i_screennumber ||
+ screenRes != QApplication::desktop()->screenGeometry(number) )
{
- b_fscHidden = false;
- setWindowOpacity( 1.0 );
+ centerFSC( number );
+ msg_Dbg( p_intf, "Recentering the Fullscreen Controller" );
}
-#else
- show();
-#endif
#if HAVE_TRANSPARENCY
- setWindowOpacity( DEFAULT_OPACITY );
+ setWindowOpacity( var_InheritFloat( p_intf, "qt-fs-opacity" ) );
#endif
-}
-/**
- * Hide fullscreen controller
- * FIXME: under windows it have to be done by moving out of screen
- * because hide() doesnt work
- */
-void FullscreenControllerWidget::hideFSC()
-{
-#ifdef WIN32TRICK
- b_fscHidden = true;
- setWindowOpacity( 0.0 ); // simulate hidding
-#else
- hide();
-#endif
+ show();
}
/**
}
else
{
-#ifdef WIN32TRICK
- if ( windowOpacity() > 0.0 && !b_fscHidden )
-#else
if ( windowOpacity() > 0.0 )
-#endif
{
/* we should use 0.01 because of 100 pieces ^^^
but than it cannt be done in time */
/**
* event handling
- * events: show, hide, start timer for hidding
+ * events: show, hide, start timer for hiding
*/
void FullscreenControllerWidget::customEvent( QEvent *event )
{
switch( event->type() )
{
+ /* This is used when the 'i' hotkey is used, to force quick toggle */
case FullscreenControlToggle_Type:
vlc_mutex_lock( &lock );
b_fs = b_fullscreen;
vlc_mutex_unlock( &lock );
+
if( b_fs )
-#ifdef WIN32TRICK
- if( b_fscHidden )
-#else
+ {
if( isHidden() )
-#endif
{
p_hideTimer->stop();
showFSC();
}
else
hideFSC();
+ }
break;
+ /* Event called to Show the FSC on mouseChanged() */
case FullscreenControlShow_Type:
vlc_mutex_lock( &lock );
b_fs = b_fullscreen;
vlc_mutex_unlock( &lock );
-#ifdef WIN32TRICK
- if( b_fs && b_fscHidden )
-#else
- if( b_fs && !isVisible() )
-#endif
+ if( b_fs )
showFSC();
+
break;
- case FullscreenControlHide_Type:
- hideFSC();
- break;
+ /* Start the timer to hide later, called usually with above case */
case FullscreenControlPlanHide_Type:
if( !b_mouse_over ) // Only if the mouse is not over FSC
planHideFSC();
break;
+ /* Hide */
+ case FullscreenControlHide_Type:
+ hideFSC();
+ break;
+ default:
+ break;
}
}
*/
void FullscreenControllerWidget::mouseMoveEvent( QMouseEvent *event )
{
- if ( event->buttons() == Qt::LeftButton )
+ if( event->buttons() == Qt::LeftButton )
{
+ if( i_mouse_last_x == -1 || i_mouse_last_y == -1 )
+ return;
+
int i_moveX = event->globalX() - i_mouse_last_x;
int i_moveY = event->globalY() - i_mouse_last_y;
{
i_mouse_last_x = event->globalX();
i_mouse_last_y = event->globalY();
+ event->accept();
+}
+
+void FullscreenControllerWidget::mouseReleaseEvent( QMouseEvent *event )
+{
+ i_mouse_last_x = -1;
+ i_mouse_last_y = -1;
+ event->accept();
}
/**
p_hideTimer->stop();
#if HAVE_TRANSPARENCY
p_slowHideTimer->stop();
+ setWindowOpacity( DEFAULT_OPACITY );
#endif
+ event->accept();
}
/**
planHideFSC();
b_mouse_over = false;
+ event->accept();
}
/**
* When you get pressed key, send it to video output
- * FIXME: clearing focus by clearFocus() to not getting
- * key press events didnt work
*/
void FullscreenControllerWidget::keyPressEvent( QKeyEvent *event )
{
- int i_vlck = qtEventToVLCKey( event );
- if( i_vlck > 0 )
- {
- var_SetInteger( p_intf->p_libvlc, "key-pressed", i_vlck );
- event->accept();
- }
- else
- event->ignore();
+ emit keyPressed( event );
}
/* */
vlc_value_t new_val, void *data )
{
vout_thread_t *p_vout = (vout_thread_t *) vlc_object;
+
msg_Dbg( p_vout, "Qt4: Fullscreen state changed" );
FullscreenControllerWidget *p_fs = (FullscreenControllerWidget *)data;
- p_fs->fullscreenChanged( p_vout, new_val.b_bool,
- var_GetInteger( p_vout, "mouse-hide-timeout" ) );
+ p_fs->fullscreenChanged( p_vout, new_val.b_bool, var_GetInteger( p_vout, "mouse-hide-timeout" ) );
return VLC_SUCCESS;
}
vlc_value_t old_val, vlc_value_t new_val,
void *data )
{
+ vout_thread_t *p_vout = (vout_thread_t *)vlc_object;
FullscreenControllerWidget *p_fs = (FullscreenControllerWidget *)data;
- int i_mousex, i_mousey;
- bool b_toShow = false;
-
/* Get the value from the Vout - Trust the vout more than Qt */
- i_mousex = var_GetInteger( p_fs->p_vout, "mouse-x" );
- i_mousey = var_GetInteger( p_fs->p_vout, "mouse-y" );
-
- /* First time */
- if( p_fs->i_mouse_last_move_x == -1 || p_fs->i_mouse_last_move_y == -1 )
- {
- p_fs->i_mouse_last_move_x = i_mousex;
- p_fs->i_mouse_last_move_y = i_mousey;
- b_toShow = true;
- }
- /* All other times */
- else
- {
- /* Trigger only if move > 3 px dans une direction */
- if( abs( p_fs->i_mouse_last_move_x - i_mousex ) > 2 ||
- abs( p_fs->i_mouse_last_move_y - i_mousey ) > 2 )
- {
- b_toShow = true;
- p_fs->i_mouse_last_move_x = i_mousex;
- p_fs->i_mouse_last_move_y = i_mousey;
- }
- }
-
- if( b_toShow )
- {
- /* Show event */
- IMEvent *eShow = new IMEvent( FullscreenControlShow_Type, 0 );
- QApplication::postEvent( p_fs, static_cast<QEvent *>(eShow) );
-
- /* Plan hide event */
- IMEvent *eHide = new IMEvent( FullscreenControlPlanHide_Type, 0 );
- QApplication::postEvent( p_fs, static_cast<QEvent *>(eHide) );
- }
+ p_fs->mouseChanged( p_vout, new_val.coords.x, new_val.coords.y );
return VLC_SUCCESS;
}
-
/**
- * It is called when video start
+ * It is call to update the list of vout handled by the fullscreen controller
*/
-void FullscreenControllerWidget::attachVout( vout_thread_t *p_nvout )
+void FullscreenControllerWidget::setVoutList( vout_thread_t **pp_vout, int i_vout )
{
- assert( p_nvout && !p_vout );
+ QList<vout_thread_t*> del;
+ QList<vout_thread_t*> add;
- p_vout = p_nvout;
+ QList<vout_thread_t*> set;
- msg_Dbg( p_vout, "Qt FS: Attaching Vout" );
- vlc_mutex_lock( &lock );
+ /* */
+ for( int i = 0; i < i_vout; i++ )
+ set += pp_vout[i];
- var_AddCallback( p_vout, "fullscreen",
- FullscreenControllerWidgetFullscreenChanged, this );
- /* I miss a add and fire */
- fullscreenChanged( p_vout, var_GetBool( p_vout, "fullscreen" ),
- var_GetInteger( p_vout, "mouse-hide-timeout" ) );
+ /* Vout to remove */
+ vlc_mutex_lock( &lock );
+ foreach( vout_thread_t *p_vout, vout )
+ {
+ if( !set.contains( p_vout ) )
+ del += p_vout;
+ }
vlc_mutex_unlock( &lock );
-}
-/**
- * It is called after turn off video.
- */
-void FullscreenControllerWidget::detachVout()
-{
- if( p_vout )
+
+ foreach( vout_thread_t *p_vout, del )
{
- msg_Dbg( p_vout, "Qt FS: Detaching Vout" );
var_DelCallback( p_vout, "fullscreen",
- FullscreenControllerWidgetFullscreenChanged, this );
+ FullscreenControllerWidgetFullscreenChanged, this );
vlc_mutex_lock( &lock );
fullscreenChanged( p_vout, false, 0 );
+ vout.removeAll( p_vout );
vlc_mutex_unlock( &lock );
- p_vout = NULL;
+
+ vlc_object_release( VLC_OBJECT(p_vout) );
}
-}
+ /* Vout to track */
+ vlc_mutex_lock( &lock );
+ foreach( vout_thread_t *p_vout, set )
+ {
+ if( !vout.contains( p_vout ) )
+ add += p_vout;
+ }
+ vlc_mutex_unlock( &lock );
+
+ foreach( vout_thread_t *p_vout, add )
+ {
+ vlc_object_hold( VLC_OBJECT(p_vout) );
+
+ vlc_mutex_lock( &lock );
+ vout.append( p_vout );
+ var_AddCallback( p_vout, "fullscreen",
+ FullscreenControllerWidgetFullscreenChanged, this );
+ /* I miss a add and fire */
+ fullscreenChanged( p_vout, var_GetBool( p_vout, "fullscreen" ),
+ var_GetInteger( p_vout, "mouse-hide-timeout" ) );
+ vlc_mutex_unlock( &lock );
+ }
+}
/**
* Register and unregister callback for mouse moving
*/
void FullscreenControllerWidget::fullscreenChanged( vout_thread_t *p_vout,
bool b_fs, int i_timeout )
{
- msg_Dbg( p_vout, "Qt: Entering Fullscreen" );
+ /* FIXME - multiple vout (ie multiple mouse position ?) and thread safety if multiple vout ? */
vlc_mutex_lock( &lock );
/* Entering fullscreen, register callback */
if( b_fs && !b_fullscreen )
{
+ msg_Dbg( p_vout, "Qt: Entering Fullscreen" );
b_fullscreen = true;
i_hide_timeout = i_timeout;
var_AddCallback( p_vout, "mouse-moved",
/* Quitting fullscreen, unregistering callback */
else if( !b_fs && b_fullscreen )
{
+ msg_Dbg( p_vout, "Qt: Quitting Fullscreen" );
b_fullscreen = false;
i_hide_timeout = i_timeout;
var_DelCallback( p_vout, "mouse-moved",
FullscreenControllerWidgetMouseMoved, this );
- /* Force fs hidding */
+ /* Force fs hiding */
IMEvent *eHide = new IMEvent( FullscreenControlHide_Type, 0 );
- QApplication::postEvent( this, static_cast<QEvent *>(eHide) );
+ QApplication::postEvent( this, eHide );
}
vlc_mutex_unlock( &lock );
}
+/**
+ * Mouse change callback (show/hide the controller on mouse movement)
+ */
+void FullscreenControllerWidget::mouseChanged( vout_thread_t *p_vout, int i_mousex, int i_mousey )
+{
+ bool b_toShow;
+
+ /* FIXME - multiple vout (ie multiple mouse position ?) and thread safety if multiple vout ? */
+
+ b_toShow = false;
+ if( ( i_mouse_last_move_x == -1 || i_mouse_last_move_y == -1 ) ||
+ ( abs( i_mouse_last_move_x - i_mousex ) > 2 ||
+ abs( i_mouse_last_move_y - i_mousey ) > 2 ) )
+ {
+ i_mouse_last_move_x = i_mousex;
+ i_mouse_last_move_y = i_mousey;
+ b_toShow = true;
+ }
+
+ if( b_toShow )
+ {
+ /* Show event */
+ IMEvent *eShow = new IMEvent( FullscreenControlShow_Type, 0 );
+ QApplication::postEvent( this, eShow );
+
+ /* Plan hide event */
+ IMEvent *eHide = new IMEvent( FullscreenControlPlanHide_Type, 0 );
+ QApplication::postEvent( this, eHide );
+ }
+}
+