You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1676 lines
58 KiB
1676 lines
58 KiB
/*****************************************************************************
|
|
* extended_panels.cpp : Extended controls panels
|
|
****************************************************************************
|
|
* Copyright (C) 2006-2013 the VideoLAN team
|
|
* $Id$
|
|
*
|
|
* Authors: Clément Stenac <zorglub@videolan.org>
|
|
* Antoine Cellerier <dionoea .t videolan d@t org>
|
|
* Jean-Baptiste Kempf <jb@videolan.org>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* ( at your option ) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
|
|
*****************************************************************************/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#include <math.h>
|
|
|
|
#include <QLabel>
|
|
#include <QVariant>
|
|
#include <QString>
|
|
#include <QFont>
|
|
#include <QGridLayout>
|
|
#include <QComboBox>
|
|
#include <QTimer>
|
|
#include <QFileDialog>
|
|
#include <QGraphicsScene>
|
|
#include <QPainter>
|
|
|
|
#include "components/extended_panels.hpp"
|
|
#include "dialogs/preferences.hpp"
|
|
#include "qt.hpp"
|
|
#include "input_manager.hpp"
|
|
#include "util/qt_dirs.hpp"
|
|
#include "util/customwidgets.hpp"
|
|
|
|
#include "../../audio_filter/equalizer_presets.h"
|
|
#include <vlc_vout.h>
|
|
#include <vlc_modules.h>
|
|
#include <vlc_plugin.h>
|
|
|
|
static char *ChangeFiltersString( struct intf_thread_t *p_intf, const char *psz_filter_type, const char *psz_name, bool b_add );
|
|
static void ChangeAFiltersString( struct intf_thread_t *p_intf, const char *psz_name, bool b_add );
|
|
static void ChangeVFiltersString( struct intf_thread_t *p_intf, const char *psz_name, bool b_add );
|
|
|
|
const QString ModuleFromWidgetName( QObject *obj )
|
|
{
|
|
return obj->objectName().replace( "Enable","" );
|
|
}
|
|
|
|
QString OptionFromWidgetName( QObject *obj )
|
|
{
|
|
/* Gruik ? ... nah */
|
|
QString option = obj->objectName().replace( "Slider", "" )
|
|
.replace( "Combo" , "" )
|
|
.replace( "Dial" , "" )
|
|
.replace( "Check" , "" )
|
|
.replace( "Spin" , "" )
|
|
.replace( "Text" , "" );
|
|
for( char a = 'A'; a <= 'Z'; a++ )
|
|
{
|
|
option = option.replace( QString( a ),
|
|
QString( '-' ) + QString( a + 'a' - 'A' ) );
|
|
}
|
|
return option;
|
|
}
|
|
|
|
static inline void setup_vfilter( intf_thread_t *p_intf, const char* psz_name, QWidget *widget )
|
|
{
|
|
vlc_object_t *p_obj = ( vlc_object_t * )
|
|
vlc_object_find_name( p_intf->obj.libvlc, psz_name );
|
|
QCheckBox *checkbox = qobject_cast<QCheckBox*>( widget );
|
|
QGroupBox *groupbox = qobject_cast<QGroupBox*>( widget );
|
|
if( p_obj )
|
|
{
|
|
vlc_object_release( p_obj ); \
|
|
if( checkbox ) checkbox->setChecked( true ); \
|
|
else if (groupbox) groupbox->setChecked( true ); \
|
|
}
|
|
else
|
|
{
|
|
if( checkbox ) checkbox->setChecked( false );
|
|
else if (groupbox) groupbox->setChecked( false );
|
|
}
|
|
}
|
|
|
|
#define SETUP_VFILTER( widget ) \
|
|
setup_vfilter( p_intf, #widget, ui.widget##Enable ); \
|
|
CONNECT( ui.widget##Enable, clicked(), this, updateFilters() );
|
|
|
|
#define SETUP_VFILTER_OPTION( widget, signal ) \
|
|
initComboBoxItems( ui.widget ); \
|
|
setWidgetValue( ui.widget ); \
|
|
CONNECT( ui.widget, signal, this, updateFilterOptions() );
|
|
|
|
ExtVideo::ExtVideo( intf_thread_t *_p_intf, QTabWidget *_parent ) :
|
|
QObject( _parent ), p_intf( _p_intf )
|
|
{
|
|
ui.setupUi( _parent );
|
|
|
|
SETUP_VFILTER( adjust )
|
|
SETUP_VFILTER_OPTION( hueSlider, valueChanged( int ) )
|
|
SETUP_VFILTER_OPTION( contrastSlider, valueChanged( int ) )
|
|
SETUP_VFILTER_OPTION( brightnessSlider, valueChanged( int ) )
|
|
SETUP_VFILTER_OPTION( saturationSlider, valueChanged( int ) )
|
|
SETUP_VFILTER_OPTION( gammaSlider, valueChanged( int ) )
|
|
SETUP_VFILTER_OPTION( brightnessThresholdCheck, stateChanged( int ) )
|
|
|
|
SETUP_VFILTER( extract )
|
|
SETUP_VFILTER_OPTION( extractComponentText, textChanged( const QString& ) )
|
|
|
|
SETUP_VFILTER( posterize )
|
|
|
|
SETUP_VFILTER( colorthres )
|
|
SETUP_VFILTER_OPTION( colorthresColorText, textChanged( const QString& ) )
|
|
SETUP_VFILTER_OPTION( colorthresSaturationthresSlider, valueChanged( int ) )
|
|
SETUP_VFILTER_OPTION( colorthresSimilaritythresSlider, valueChanged( int ) )
|
|
|
|
SETUP_VFILTER( sepia )
|
|
SETUP_VFILTER_OPTION( sepiaIntensitySpin, valueChanged( int ) )
|
|
|
|
SETUP_VFILTER( invert )
|
|
|
|
SETUP_VFILTER( gradient )
|
|
SETUP_VFILTER_OPTION( gradientModeCombo, currentIndexChanged( QString ) )
|
|
SETUP_VFILTER_OPTION( gradientTypeCheck, stateChanged( int ) )
|
|
SETUP_VFILTER_OPTION( gradientCartoonCheck, stateChanged( int ) )
|
|
|
|
SETUP_VFILTER( motionblur )
|
|
SETUP_VFILTER_OPTION( blurFactorSlider, valueChanged( int ) )
|
|
|
|
SETUP_VFILTER( motiondetect )
|
|
|
|
SETUP_VFILTER( psychedelic )
|
|
|
|
SETUP_VFILTER( sharpen )
|
|
SETUP_VFILTER_OPTION( sharpenSigmaSlider, valueChanged( int ) )
|
|
|
|
SETUP_VFILTER( ripple )
|
|
|
|
SETUP_VFILTER( wave )
|
|
|
|
SETUP_VFILTER( transform )
|
|
SETUP_VFILTER_OPTION( transformTypeCombo, currentIndexChanged( QString ) )
|
|
|
|
SETUP_VFILTER( rotate )
|
|
SETUP_VFILTER_OPTION( rotateAngleDial, valueChanged( int ) )
|
|
ui.rotateAngleDial->setWrapping( true );
|
|
ui.rotateAngleDial->setNotchesVisible( true );
|
|
|
|
SETUP_VFILTER( puzzle )
|
|
SETUP_VFILTER_OPTION( puzzleRowsSpin, valueChanged( int ) )
|
|
SETUP_VFILTER_OPTION( puzzleColsSpin, valueChanged( int ) )
|
|
SETUP_VFILTER_OPTION( puzzleBlackSlotCheck, stateChanged( int ) )
|
|
|
|
SETUP_VFILTER( magnify )
|
|
|
|
SETUP_VFILTER( clone )
|
|
SETUP_VFILTER_OPTION( cloneCountSpin, valueChanged( int ) )
|
|
|
|
SETUP_VFILTER( wall )
|
|
SETUP_VFILTER_OPTION( wallRowsSpin, valueChanged( int ) )
|
|
SETUP_VFILTER_OPTION( wallColsSpin, valueChanged( int ) )
|
|
|
|
|
|
SETUP_VFILTER( erase )
|
|
SETUP_VFILTER_OPTION( eraseMaskText, editingFinished() )
|
|
SETUP_VFILTER_OPTION( eraseYSpin, valueChanged( int ) )
|
|
SETUP_VFILTER_OPTION( eraseXSpin, valueChanged( int ) )
|
|
BUTTONACT( ui.eraseBrowseBtn, browseEraseFile() );
|
|
|
|
SETUP_VFILTER( marq )
|
|
SETUP_VFILTER_OPTION( marqMarqueeText, textChanged( const QString& ) )
|
|
SETUP_VFILTER_OPTION( marqPositionCombo, currentIndexChanged( QString ) )
|
|
|
|
SETUP_VFILTER( logo )
|
|
SETUP_VFILTER_OPTION( logoFileText, editingFinished() )
|
|
SETUP_VFILTER_OPTION( logoYSpin, valueChanged( int ) )
|
|
SETUP_VFILTER_OPTION( logoXSpin, valueChanged( int ) )
|
|
SETUP_VFILTER_OPTION( logoOpacitySlider, valueChanged( int ) )
|
|
BUTTONACT( ui.logoBrowseBtn, browseLogo() );
|
|
|
|
SETUP_VFILTER( gradfun )
|
|
SETUP_VFILTER_OPTION( gradfunRadiusSlider, valueChanged( int ) )
|
|
|
|
SETUP_VFILTER( grain )
|
|
SETUP_VFILTER_OPTION( grainVarianceSlider, valueChanged( int ) )
|
|
|
|
SETUP_VFILTER( mirror )
|
|
|
|
SETUP_VFILTER( gaussianblur )
|
|
SETUP_VFILTER_OPTION( gaussianblurSigmaSlider, valueChanged( int ) )
|
|
|
|
SETUP_VFILTER( antiflicker )
|
|
SETUP_VFILTER_OPTION( antiflickerSofteningSizeSlider, valueChanged( int ) )
|
|
|
|
SETUP_VFILTER( hqdn3d )
|
|
SETUP_VFILTER_OPTION( hqdn3dLumaSpatSlider, valueChanged( int ) )
|
|
SETUP_VFILTER_OPTION( hqdn3dLumaTempSlider, valueChanged( int ) )
|
|
SETUP_VFILTER_OPTION( hqdn3dChromaSpatSlider, valueChanged( int ) )
|
|
SETUP_VFILTER_OPTION( hqdn3dChromaTempSlider, valueChanged( int ) )
|
|
|
|
|
|
SETUP_VFILTER( anaglyph )
|
|
|
|
#undef SETUP_VFILTER
|
|
#undef SETUP_VFILTER_OPTION
|
|
|
|
CONNECT( ui.cropTopPx, valueChanged( int ), this, cropChange() );
|
|
CONNECT( ui.cropBotPx, valueChanged( int ), this, cropChange() );
|
|
CONNECT( ui.cropLeftPx, valueChanged( int ), this, cropChange() );
|
|
CONNECT( ui.cropRightPx, valueChanged( int ), this, cropChange() );
|
|
CONNECT( ui.leftRightCropSync, toggled ( bool ), this, cropChange() );
|
|
CONNECT( ui.topBotCropSync, toggled ( bool ), this, cropChange() );
|
|
CONNECT( ui.topBotCropSync, toggled( bool ),
|
|
ui.cropBotPx, setDisabled( bool ) );
|
|
CONNECT( ui.leftRightCropSync, toggled( bool ),
|
|
ui.cropRightPx, setDisabled( bool ) );
|
|
}
|
|
|
|
void ExtVideo::cropChange()
|
|
{
|
|
if( ui.topBotCropSync->isChecked() )
|
|
ui.cropBotPx->setValue( ui.cropTopPx->value() );
|
|
if( ui.leftRightCropSync->isChecked() )
|
|
ui.cropRightPx->setValue( ui.cropLeftPx->value() );
|
|
|
|
vout_thread_t *p_vout = THEMIM->getVout();
|
|
if( p_vout )
|
|
{
|
|
var_SetInteger( p_vout, "crop-top", ui.cropTopPx->value() );
|
|
var_SetInteger( p_vout, "crop-bottom", ui.cropBotPx->value() );
|
|
var_SetInteger( p_vout, "crop-left", ui.cropLeftPx->value() );
|
|
var_SetInteger( p_vout, "crop-right", ui.cropRightPx->value() );
|
|
vlc_object_release( p_vout );
|
|
}
|
|
}
|
|
|
|
void ExtVideo::clean()
|
|
{
|
|
ui.cropTopPx->setValue( 0 );
|
|
ui.cropBotPx->setValue( 0 );
|
|
ui.cropLeftPx->setValue( 0 );
|
|
ui.cropRightPx->setValue( 0 );
|
|
}
|
|
|
|
static char *ChangeFiltersString( struct intf_thread_t *p_intf, const char *psz_filter_type, const char *psz_name, bool b_add )
|
|
{
|
|
char *psz_parser, *psz_string;
|
|
|
|
psz_string = config_GetPsz( p_intf, psz_filter_type );
|
|
|
|
if( !psz_string ) psz_string = strdup( "" );
|
|
|
|
psz_parser = strstr( psz_string, psz_name );
|
|
|
|
if( b_add )
|
|
{
|
|
if( !psz_parser )
|
|
{
|
|
psz_parser = psz_string;
|
|
if( asprintf( &psz_string, ( *psz_string ) ? "%s:%s" : "%s%s",
|
|
psz_string, psz_name ) == -1 )
|
|
{
|
|
free( psz_parser );
|
|
return NULL;
|
|
}
|
|
free( psz_parser );
|
|
}
|
|
else
|
|
{
|
|
free( psz_string );
|
|
return NULL;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if( psz_parser )
|
|
{
|
|
if( *( psz_parser + strlen( psz_name ) ) == ':' )
|
|
{
|
|
memmove( psz_parser, psz_parser + strlen( psz_name ) + 1,
|
|
strlen( psz_parser + strlen( psz_name ) + 1 ) + 1 );
|
|
}
|
|
else
|
|
{
|
|
*psz_parser = '\0';
|
|
}
|
|
|
|
/* Remove trailing : : */
|
|
size_t i_len = strlen( psz_string );
|
|
if( i_len > 0 && *( psz_string + i_len - 1 ) == ':' )
|
|
{
|
|
*( psz_string + i_len - 1 ) = '\0';
|
|
}
|
|
}
|
|
else
|
|
{
|
|
free( psz_string );
|
|
return NULL;
|
|
}
|
|
}
|
|
return psz_string;
|
|
}
|
|
|
|
static void ChangeAFiltersString( struct intf_thread_t *p_intf, const char *psz_name, bool b_add )
|
|
{
|
|
char *psz_string;
|
|
|
|
module_t *p_obj = module_find( psz_name );
|
|
if( !p_obj )
|
|
{
|
|
msg_Err( p_intf, "Unable to find filter module \"%s\".", psz_name );
|
|
return;
|
|
}
|
|
|
|
psz_string = ChangeFiltersString( p_intf, "audio-filter", psz_name, b_add );
|
|
if( !psz_string )
|
|
return;
|
|
|
|
config_PutPsz( p_intf, "audio-filter", psz_string );
|
|
|
|
free( psz_string );
|
|
}
|
|
|
|
static const char* GetVFilterType( struct intf_thread_t *p_intf, const char *psz_name )
|
|
{
|
|
module_t *p_obj = module_find( psz_name );
|
|
if( !p_obj )
|
|
{
|
|
msg_Err( p_intf, "Unable to find filter module \"%s\".", psz_name );
|
|
return NULL;
|
|
}
|
|
|
|
if( module_provides( p_obj, "video splitter" ) )
|
|
return "video-splitter";
|
|
else if( module_provides( p_obj, "video filter" ) )
|
|
return "video-filter";
|
|
else if( module_provides( p_obj, "sub source" ) )
|
|
return "sub-source";
|
|
else if( module_provides( p_obj, "sub filter" ) )
|
|
return "sub-filter";
|
|
else
|
|
{
|
|
msg_Err( p_intf, "Unknown video filter type." );
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
static void ChangeVFiltersString( struct intf_thread_t *p_intf, const char *psz_name, bool b_add )
|
|
{
|
|
char *psz_string;
|
|
const char *psz_filter_type = GetVFilterType( p_intf, psz_name );
|
|
|
|
psz_string = ChangeFiltersString( p_intf, psz_filter_type, psz_name, b_add );
|
|
if( !psz_string )
|
|
return;
|
|
|
|
/* Vout is not kept, so put that in the config */
|
|
config_PutPsz( p_intf, psz_filter_type, psz_string );
|
|
|
|
/* Try to set on the fly */
|
|
if( !strcmp( psz_filter_type, "video-splitter" ) )
|
|
{
|
|
playlist_t *p_playlist = THEPL;
|
|
var_SetString( p_playlist, psz_filter_type, psz_string );
|
|
}
|
|
else
|
|
{
|
|
vout_thread_t *p_vout = THEMIM->getVout();
|
|
if( p_vout )
|
|
{
|
|
var_SetString( p_vout, psz_filter_type, psz_string );
|
|
vlc_object_release( p_vout );
|
|
}
|
|
}
|
|
|
|
free( psz_string );
|
|
}
|
|
|
|
void ExtVideo::updateFilters()
|
|
{
|
|
QString module = ModuleFromWidgetName( sender() );
|
|
|
|
QCheckBox *checkbox = qobject_cast<QCheckBox*>( sender() );
|
|
QGroupBox *groupbox = qobject_cast<QGroupBox*>( sender() );
|
|
|
|
ChangeVFiltersString( p_intf, qtu( module ),
|
|
checkbox ? checkbox->isChecked()
|
|
: groupbox->isChecked() );
|
|
}
|
|
|
|
#define UPDATE_AND_APPLY_TEXT( widget, file ) \
|
|
CONNECT( ui.widget, textChanged( const QString& ), \
|
|
this, updateFilterOptions() ); \
|
|
ui.widget->setText( toNativeSeparators( file ) ); \
|
|
ui.widget->disconnect( SIGNAL( textChanged( const QString& ) ) );
|
|
|
|
void ExtVideo::browseLogo()
|
|
{
|
|
QString file = QFileDialog::getOpenFileName( NULL, qtr( "Logo filenames" ),
|
|
p_intf->p_sys->filepath, "Images (*.png *.jpg);;All (*)" );
|
|
|
|
UPDATE_AND_APPLY_TEXT( logoFileText, file );
|
|
}
|
|
|
|
void ExtVideo::browseEraseFile()
|
|
{
|
|
QString file = QFileDialog::getOpenFileName( NULL, qtr( "Image mask" ),
|
|
p_intf->p_sys->filepath, "Images (*.png *.jpg);;All (*)" );
|
|
|
|
UPDATE_AND_APPLY_TEXT( eraseMaskText, file );
|
|
}
|
|
|
|
#undef UPDATE_AND_APPLY_TEXT
|
|
|
|
void ExtVideo::initComboBoxItems( QObject *widget )
|
|
{
|
|
QComboBox *combobox = qobject_cast<QComboBox*>( widget );
|
|
if( !combobox ) return;
|
|
|
|
QString option = OptionFromWidgetName( widget );
|
|
module_config_t *p_item = config_FindConfig( VLC_OBJECT( p_intf ),
|
|
qtu( option ) );
|
|
if( p_item == NULL )
|
|
{
|
|
msg_Err( p_intf, "Couldn't find option \"%s\".", qtu( option ) );
|
|
return;
|
|
}
|
|
|
|
if( p_item->i_type == CONFIG_ITEM_INTEGER
|
|
|| p_item->i_type == CONFIG_ITEM_BOOL )
|
|
{
|
|
int64_t *values;
|
|
char **texts;
|
|
ssize_t count = config_GetIntChoices( VLC_OBJECT( p_intf ),
|
|
qtu( option ), &values, &texts );
|
|
for( ssize_t i = 0; i < count; i++ )
|
|
{
|
|
combobox->addItem( qtr( texts[i] ), qlonglong(values[i]) );
|
|
free( texts[i] );
|
|
}
|
|
free( texts );
|
|
free( values );
|
|
}
|
|
else if( p_item->i_type == CONFIG_ITEM_STRING )
|
|
{
|
|
char **values;
|
|
char **texts;
|
|
ssize_t count = config_GetPszChoices( VLC_OBJECT( p_intf ),
|
|
qtu( option ), &values, &texts );
|
|
for( ssize_t i = 0; i < count; i++ )
|
|
{
|
|
combobox->addItem( qtr( texts[i] ), qfu(values[i]) );
|
|
free( texts[i] );
|
|
free( values[i] );
|
|
}
|
|
free( texts );
|
|
free( values );
|
|
}
|
|
}
|
|
|
|
void ExtVideo::setWidgetValue( QObject *widget )
|
|
{
|
|
QString module = ModuleFromWidgetName( widget->parent() );
|
|
//std::cout << "Module name: " << module.toStdString() << std::endl;
|
|
QString option = OptionFromWidgetName( widget );
|
|
//std::cout << "Option name: " << option.toStdString() << std::endl;
|
|
|
|
vlc_object_t *p_obj = ( vlc_object_t * )
|
|
vlc_object_find_name( p_intf->obj.libvlc, qtu( module ) );
|
|
int i_type;
|
|
vlc_value_t val;
|
|
|
|
if( !p_obj )
|
|
{
|
|
#if 0
|
|
msg_Dbg( p_intf,
|
|
"Module instance %s not found, looking in config values.",
|
|
qtu( module ) );
|
|
#endif
|
|
i_type = config_GetType( p_intf, qtu( option ) ) & VLC_VAR_CLASS;
|
|
switch( i_type )
|
|
{
|
|
case VLC_VAR_INTEGER:
|
|
case VLC_VAR_BOOL:
|
|
val.i_int = config_GetInt( p_intf, qtu( option ) );
|
|
break;
|
|
case VLC_VAR_FLOAT:
|
|
val.f_float = config_GetFloat( p_intf, qtu( option ) );
|
|
break;
|
|
case VLC_VAR_STRING:
|
|
val.psz_string = config_GetPsz( p_intf, qtu( option ) );
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
i_type = var_Type( p_obj, qtu( option ) ) & VLC_VAR_CLASS;
|
|
var_Get( p_obj, qtu( option ), &val );
|
|
vlc_object_release( p_obj );
|
|
}
|
|
|
|
/* Try to cast to all the widgets we're likely to encounter. Only
|
|
* one of the casts is expected to work. */
|
|
QSlider *slider = qobject_cast<QSlider*> ( widget );
|
|
QCheckBox *checkbox = qobject_cast<QCheckBox*> ( widget );
|
|
QSpinBox *spinbox = qobject_cast<QSpinBox*> ( widget );
|
|
QDoubleSpinBox *doublespinbox = qobject_cast<QDoubleSpinBox*>( widget );
|
|
VLCQDial *dial = qobject_cast<VLCQDial*> ( widget );
|
|
QLineEdit *lineedit = qobject_cast<QLineEdit*> ( widget );
|
|
QComboBox *combobox = qobject_cast<QComboBox*> ( widget );
|
|
|
|
if( i_type == VLC_VAR_INTEGER || i_type == VLC_VAR_BOOL )
|
|
{
|
|
if( slider ) slider->setValue( val.i_int );
|
|
else if( checkbox ) checkbox->setCheckState( val.i_int? Qt::Checked
|
|
: Qt::Unchecked );
|
|
else if( spinbox ) spinbox->setValue( val.i_int );
|
|
else if( dial ) dial->setValue( (360 - val.i_int) % 360 );
|
|
else if( lineedit )
|
|
{
|
|
char str[30];
|
|
snprintf( str, sizeof(str), "%06" PRIX64, val.i_int );
|
|
lineedit->setText( str );
|
|
}
|
|
else if( combobox ) combobox->setCurrentIndex(
|
|
combobox->findData( qlonglong(val.i_int) ) );
|
|
else msg_Warn( p_intf, "Could not find the correct Integer widget" );
|
|
}
|
|
else if( i_type == VLC_VAR_FLOAT )
|
|
{
|
|
if( slider ) slider->setValue( ( int )( val.f_float*( double )slider->tickInterval() ) ); /* hack alert! */
|
|
else if( doublespinbox ) doublespinbox->setValue( val.f_float );
|
|
else if( dial ) dial->setValue( (360 - lroundf(val.f_float)) % 360 );
|
|
else msg_Warn( p_intf, "Could not find the correct Float widget" );
|
|
}
|
|
else if( i_type == VLC_VAR_STRING )
|
|
{
|
|
if( lineedit ) lineedit->setText( qfu( val.psz_string ) );
|
|
else if( combobox ) combobox->setCurrentIndex(
|
|
combobox->findData( qfu( val.psz_string ) ) );
|
|
else msg_Warn( p_intf, "Could not find the correct String widget" );
|
|
free( val.psz_string );
|
|
}
|
|
else
|
|
if( p_obj )
|
|
msg_Err( p_intf,
|
|
"Module %s's %s variable is of an unsupported type ( %d )",
|
|
qtu( module ),
|
|
qtu( option ),
|
|
i_type );
|
|
}
|
|
|
|
void ExtVideo::setFilterOption( struct intf_thread_t *p_intf, const char *psz_module, const char *psz_option,
|
|
int i_int, double f_float, QString val )
|
|
{
|
|
vlc_object_t *p_obj = ( vlc_object_t * )vlc_object_find_name( p_intf->obj.libvlc, psz_module );
|
|
int i_type;
|
|
bool b_is_command;
|
|
|
|
if( !p_obj )
|
|
{
|
|
msg_Warn( p_intf, "Module %s not found. You'll need to restart the filter to take the change into account.", psz_module );
|
|
i_type = config_GetType( p_intf, psz_option );
|
|
b_is_command = false;
|
|
}
|
|
else
|
|
{
|
|
i_type = var_Type( p_obj, psz_option );
|
|
if( i_type == 0 )
|
|
i_type = config_GetType( p_intf, psz_option );
|
|
b_is_command = ( i_type & VLC_VAR_ISCOMMAND );
|
|
}
|
|
|
|
i_type &= VLC_VAR_CLASS;
|
|
if( i_type == VLC_VAR_INTEGER || i_type == VLC_VAR_BOOL )
|
|
{
|
|
if( i_int == -1 )
|
|
msg_Warn( p_intf, "Could not find the correct Integer widget" );
|
|
config_PutInt( p_intf, psz_option, i_int );
|
|
if( b_is_command )
|
|
{
|
|
if( i_type == VLC_VAR_INTEGER )
|
|
var_SetInteger( p_obj, psz_option, i_int );
|
|
else
|
|
var_SetBool( p_obj, psz_option, i_int );
|
|
}
|
|
}
|
|
else if( i_type == VLC_VAR_FLOAT )
|
|
{
|
|
if( f_float == -1 )
|
|
msg_Warn( p_intf, "Could not find the correct Float widget" );
|
|
config_PutFloat( p_intf, psz_option, f_float );
|
|
if( b_is_command )
|
|
var_SetFloat( p_obj, psz_option, f_float );
|
|
}
|
|
else if( i_type == VLC_VAR_STRING )
|
|
{
|
|
if( val.isNull() )
|
|
msg_Warn( p_intf, "Could not find the correct String widget" );
|
|
config_PutPsz( p_intf, psz_option, qtu( val ) );
|
|
if( b_is_command )
|
|
var_SetString( p_obj, psz_option, qtu( val ) );
|
|
}
|
|
else
|
|
msg_Err( p_intf,
|
|
"Module %s's %s variable is of an unsupported type ( %d )",
|
|
psz_module,
|
|
psz_option,
|
|
i_type );
|
|
|
|
if( !b_is_command )
|
|
{
|
|
msg_Warn( p_intf, "Module %s's %s variable isn't a command. Brute-restarting the filter.",
|
|
psz_module,
|
|
psz_option );
|
|
ChangeVFiltersString( p_intf, psz_module, false );
|
|
ChangeVFiltersString( p_intf, psz_module, true );
|
|
}
|
|
|
|
if( p_obj ) vlc_object_release( p_obj );
|
|
}
|
|
|
|
void ExtVideo::updateFilterOptions()
|
|
{
|
|
QString module = ModuleFromWidgetName( sender()->parent() );
|
|
//msg_Dbg( p_intf, "Module name: %s", qtu( module ) );
|
|
QString option = OptionFromWidgetName( sender() );
|
|
//msg_Dbg( p_intf, "Option name: %s", qtu( option ) );
|
|
|
|
/* Try to cast to all the widgets we're likely to encounter. Only
|
|
* one of the casts is expected to work. */
|
|
QSlider *slider = qobject_cast<QSlider*> ( sender() );
|
|
QCheckBox *checkbox = qobject_cast<QCheckBox*> ( sender() );
|
|
QSpinBox *spinbox = qobject_cast<QSpinBox*> ( sender() );
|
|
QDoubleSpinBox *doublespinbox = qobject_cast<QDoubleSpinBox*>( sender() );
|
|
VLCQDial *dial = qobject_cast<VLCQDial*> ( sender() );
|
|
QLineEdit *lineedit = qobject_cast<QLineEdit*> ( sender() );
|
|
QComboBox *combobox = qobject_cast<QComboBox*> ( sender() );
|
|
|
|
int i_int = -1;
|
|
double f_float = -1.;
|
|
QString val;
|
|
|
|
if( slider ) {
|
|
i_int = slider->value();
|
|
f_float = ( double )slider->value() / ( double )slider->tickInterval(); /* hack alert! */
|
|
}
|
|
else if( checkbox ) i_int = checkbox->checkState() == Qt::Checked;
|
|
else if( spinbox ) i_int = spinbox->value();
|
|
else if( doublespinbox ) f_float = doublespinbox->value();
|
|
else if( dial ) {
|
|
i_int = (360 - dial->value()) % 360;
|
|
f_float = i_int;
|
|
}
|
|
else if( lineedit ) {
|
|
i_int = lineedit->text().toInt( NULL,16 );
|
|
f_float = lineedit->text().toDouble();
|
|
val = lineedit->text();
|
|
}
|
|
else if( combobox ) {
|
|
i_int = combobox->itemData( combobox->currentIndex() ).toInt();
|
|
val = combobox->itemData( combobox->currentIndex() ).toString();
|
|
}
|
|
|
|
setFilterOption( p_intf, qtu( module ), qtu( option ), i_int, f_float, val);
|
|
}
|
|
|
|
int ExtVideo::getPostprocessing( struct intf_thread_t *p_intf)
|
|
{
|
|
char *psz_config = config_GetPsz(p_intf, "video-filter");
|
|
int i_q = -1;
|
|
if (psz_config) {
|
|
if (strstr(psz_config, "postproc"))
|
|
i_q = config_GetInt(p_intf, "postproc-q");
|
|
free(psz_config);
|
|
}
|
|
return i_q;
|
|
}
|
|
|
|
void ExtVideo::setPostprocessing( struct intf_thread_t *p_intf, int q)
|
|
{
|
|
const char *psz_name = "postproc";
|
|
|
|
if( q == -1 )
|
|
{
|
|
ChangeVFiltersString( p_intf, psz_name, false );
|
|
}
|
|
else
|
|
{
|
|
ChangeVFiltersString( p_intf, psz_name, false );
|
|
setFilterOption( p_intf, "postproc", "postproc-q", q, -1, QString() );
|
|
}
|
|
}
|
|
|
|
/**********************************************************************
|
|
* v4l2 controls
|
|
**********************************************************************/
|
|
|
|
ExtV4l2::ExtV4l2( intf_thread_t *_p_intf, QWidget *_parent )
|
|
: QWidget( _parent ), p_intf( _p_intf ), box( NULL )
|
|
{
|
|
QVBoxLayout *layout = new QVBoxLayout( this );
|
|
help = new QLabel( qtr("No v4l2 instance found.\n"
|
|
"Please check that the device has been opened with VLC and is playing.\n\n"
|
|
"Controls will automatically appear here.")
|
|
, this );
|
|
help->setAlignment( Qt::AlignHCenter | Qt::AlignVCenter );
|
|
help->setWordWrap( true );
|
|
layout->addWidget( help );
|
|
setLayout( layout );
|
|
}
|
|
|
|
void ExtV4l2::showEvent( QShowEvent *event )
|
|
{
|
|
QWidget::showEvent( event );
|
|
Refresh();
|
|
}
|
|
|
|
void ExtV4l2::Refresh( void )
|
|
{
|
|
vlc_object_t *p_obj = (vlc_object_t*)vlc_object_find_name( THEPL, "v4l2" );
|
|
help->hide();
|
|
if( box )
|
|
{
|
|
layout()->removeWidget( box );
|
|
delete box;
|
|
box = NULL;
|
|
}
|
|
if( p_obj )
|
|
{
|
|
vlc_value_t val, text;
|
|
int i_ret = var_Change( p_obj, "controls", VLC_VAR_GETCHOICES,
|
|
&val, &text );
|
|
if( i_ret < 0 )
|
|
{
|
|
msg_Err( p_intf, "Oops, v4l2 object doesn't have a 'controls' variable." );
|
|
help->show();
|
|
vlc_object_release( p_obj );
|
|
return;
|
|
}
|
|
|
|
box = new QGroupBox( this );
|
|
layout()->addWidget( box );
|
|
QVBoxLayout *layout = new QVBoxLayout( box );
|
|
box->setLayout( layout );
|
|
|
|
for( int i = 0; i < val.p_list->i_count; i++ )
|
|
{
|
|
vlc_value_t vartext;
|
|
const char *psz_var = text.p_list->p_values[i].psz_string;
|
|
|
|
if( var_Change( p_obj, psz_var, VLC_VAR_GETTEXT, &vartext, NULL ) )
|
|
continue;
|
|
|
|
QString name = qtr( vartext.psz_string );
|
|
free( vartext.psz_string );
|
|
msg_Dbg( p_intf, "v4l2 control \"%" PRIx64 "\": %s (%s)",
|
|
val.p_list->p_values[i].i_int, psz_var, qtu( name ) );
|
|
|
|
int i_type = var_Type( p_obj, psz_var );
|
|
switch( i_type & VLC_VAR_TYPE )
|
|
{
|
|
case VLC_VAR_INTEGER:
|
|
{
|
|
QLabel *label = new QLabel( name, box );
|
|
QHBoxLayout *hlayout = new QHBoxLayout();
|
|
hlayout->addWidget( label );
|
|
int i_val = var_GetInteger( p_obj, psz_var );
|
|
if( i_type & VLC_VAR_HASCHOICE )
|
|
{
|
|
QComboBox *combobox = new QComboBox( box );
|
|
combobox->setObjectName( qfu( psz_var ) );
|
|
|
|
vlc_value_t val2, text2;
|
|
var_Change( p_obj, psz_var, VLC_VAR_GETCHOICES,
|
|
&val2, &text2 );
|
|
for( int j = 0; j < val2.p_list->i_count; j++ )
|
|
{
|
|
combobox->addItem(
|
|
text2.p_list->p_values[j].psz_string,
|
|
qlonglong( val2.p_list->p_values[j].i_int) );
|
|
if( i_val == val2.p_list->p_values[j].i_int )
|
|
combobox->setCurrentIndex( j );
|
|
}
|
|
var_FreeList( &val2, &text2 );
|
|
|
|
CONNECT( combobox, currentIndexChanged( int ), this,
|
|
ValueChange( int ) );
|
|
hlayout->addWidget( combobox );
|
|
}
|
|
else
|
|
{
|
|
QSlider *slider = new QSlider( box );
|
|
slider->setObjectName( qfu( psz_var ) );
|
|
slider->setOrientation( Qt::Horizontal );
|
|
vlc_value_t val2;
|
|
var_Change( p_obj, psz_var, VLC_VAR_GETMIN,
|
|
&val2, NULL );
|
|
if( val2.i_int < INT_MIN )
|
|
val2.i_int = INT_MIN; /* FIXME */
|
|
slider->setMinimum( val2.i_int );
|
|
var_Change( p_obj, psz_var, VLC_VAR_GETMAX,
|
|
&val2, NULL );
|
|
if( val2.i_int > INT_MAX )
|
|
val2.i_int = INT_MAX; /* FIXME */
|
|
slider->setMaximum( val2.i_int );
|
|
if( !var_Change( p_obj, psz_var, VLC_VAR_GETSTEP,
|
|
&val2, NULL ) )
|
|
slider->setSingleStep( val2.i_int );
|
|
slider->setValue( i_val );
|
|
CONNECT( slider, valueChanged( int ), this,
|
|
ValueChange( int ) );
|
|
hlayout->addWidget( slider );
|
|
}
|
|
layout->addLayout( hlayout );
|
|
break;
|
|
}
|
|
case VLC_VAR_BOOL:
|
|
{
|
|
QCheckBox *button = new QCheckBox( name, box );
|
|
button->setObjectName( qfu( psz_var ) );
|
|
button->setChecked( var_GetBool( p_obj, psz_var ) );
|
|
|
|
CONNECT( button, clicked( bool ), this,
|
|
ValueChange( bool ) );
|
|
layout->addWidget( button );
|
|
break;
|
|
}
|
|
case VLC_VAR_VOID:
|
|
{
|
|
if( i_type & VLC_VAR_ISCOMMAND )
|
|
{
|
|
QPushButton *button = new QPushButton( name, box );
|
|
button->setObjectName( qfu( psz_var ) );
|
|
|
|
CONNECT( button, clicked( bool ), this,
|
|
ValueChange( bool ) );
|
|
layout->addWidget( button );
|
|
}
|
|
else
|
|
{
|
|
QLabel *label = new QLabel( name, box );
|
|
layout->addWidget( label );
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
msg_Warn( p_intf, "Unhandled var type for %s", psz_var );
|
|
break;
|
|
}
|
|
}
|
|
var_FreeList( &val, &text );
|
|
vlc_object_release( p_obj );
|
|
}
|
|
else
|
|
{
|
|
msg_Dbg( p_intf, "Couldn't find v4l2 instance" );
|
|
help->show();
|
|
if ( isVisible() )
|
|
QTimer::singleShot( 2000, this, SLOT(Refresh()) );
|
|
}
|
|
}
|
|
|
|
void ExtV4l2::ValueChange( bool value )
|
|
{
|
|
ValueChange( (int)value );
|
|
}
|
|
|
|
void ExtV4l2::ValueChange( int value )
|
|
{
|
|
QObject *s = sender();
|
|
vlc_object_t *p_obj = (vlc_object_t*)vlc_object_find_name( THEPL, "v4l2" );
|
|
if( p_obj )
|
|
{
|
|
QString var = s->objectName();
|
|
int i_type = var_Type( p_obj, qtu( var ) );
|
|
switch( i_type & VLC_VAR_TYPE )
|
|
{
|
|
case VLC_VAR_INTEGER:
|
|
if( i_type & VLC_VAR_HASCHOICE )
|
|
{
|
|
QComboBox *combobox = qobject_cast<QComboBox*>( s );
|
|
value = combobox->itemData( value ).toInt();
|
|
}
|
|
var_SetInteger( p_obj, qtu( var ), value );
|
|
break;
|
|
case VLC_VAR_BOOL:
|
|
var_SetBool( p_obj, qtu( var ), value );
|
|
break;
|
|
case VLC_VAR_VOID:
|
|
var_TriggerCallback( p_obj, qtu( var ) );
|
|
break;
|
|
}
|
|
vlc_object_release( p_obj );
|
|
}
|
|
else
|
|
{
|
|
msg_Warn( p_intf, "Oops, v4l2 object isn't available anymore" );
|
|
Refresh();
|
|
}
|
|
}
|
|
|
|
/**********************************************************************
|
|
* Sliders
|
|
**********************************************************************/
|
|
|
|
FilterSliderData::FilterSliderData( QObject *parent, QSlider *_slider ) :
|
|
QObject( parent ), slider( _slider )
|
|
{
|
|
b_save_to_config = false;
|
|
}
|
|
|
|
FilterSliderData::FilterSliderData( QObject *parent,
|
|
intf_thread_t *_p_intf,
|
|
QSlider *_slider,
|
|
QLabel *_label, QLabel *_nameLabel,
|
|
const slider_data_t *_p_data ):
|
|
QObject( parent ), slider( _slider ), valueLabel( _label ),
|
|
nameLabel( _nameLabel ), p_data( _p_data ), p_intf( _p_intf )
|
|
{
|
|
b_save_to_config = false;
|
|
slider->setMinimum( p_data->f_min / p_data->f_resolution );
|
|
slider->setMaximum( p_data->f_max / p_data->f_resolution );
|
|
nameLabel->setText( p_data->descs );
|
|
CONNECT( slider, valueChanged( int ), this, updateText( int ) );
|
|
setValue( initialValue() );
|
|
/* In case current == min|max text would not be first updated */
|
|
if ( slider->value() == slider->maximum() ||
|
|
slider->value() == slider->minimum() )
|
|
updateText( slider->value() );
|
|
CONNECT( slider, valueChanged( int ), this, onValueChanged( int ) );
|
|
}
|
|
|
|
void FilterSliderData::setValue( float f )
|
|
{
|
|
slider->setValue( f / p_data->f_resolution );
|
|
}
|
|
|
|
void FilterSliderData::updateText( int i )
|
|
{
|
|
float f = ((float) i) * p_data->f_resolution * p_data->f_visual_multiplier;
|
|
valueLabel->setText( QString( p_data->units )
|
|
.prepend( "%1 " )
|
|
.arg( QString::number( f, 'f', 1 ) ) );
|
|
}
|
|
|
|
float FilterSliderData::initialValue()
|
|
{
|
|
vlc_object_t *p_aout = (vlc_object_t *) THEMIM->getAout();
|
|
float f = p_data->f_value;
|
|
if( p_aout )
|
|
{
|
|
if ( var_Type( p_aout, qtu(p_data->name) ) == 0 )
|
|
{
|
|
vlc_object_release( p_aout );
|
|
/* Not found, will try in config */
|
|
}
|
|
else
|
|
{
|
|
f = var_GetFloat( p_aout, qtu(p_data->name) );
|
|
vlc_object_release( p_aout );
|
|
return f;
|
|
}
|
|
}
|
|
|
|
if ( ! config_FindConfig( VLC_OBJECT(p_intf), qtu(p_data->name) ) )
|
|
return f;
|
|
|
|
f = config_GetFloat( p_intf, qtu(p_data->name) );
|
|
return f;
|
|
}
|
|
|
|
void FilterSliderData::onValueChanged( int i ) const
|
|
{
|
|
float f = ((float) i) * p_data->f_resolution;
|
|
vlc_object_t *p_aout = (vlc_object_t *) THEMIM->getAout();
|
|
if ( p_aout )
|
|
{
|
|
var_SetFloat( p_aout, qtu(p_data->name), f );
|
|
vlc_object_release( p_aout );
|
|
}
|
|
writeToConfig();
|
|
}
|
|
|
|
void FilterSliderData::writeToConfig() const
|
|
{
|
|
if ( !b_save_to_config ) return;
|
|
float f = ((float) slider->value()) * p_data->f_resolution;
|
|
config_PutFloat( p_intf, qtu(p_data->name), f );
|
|
}
|
|
|
|
void FilterSliderData::setSaveToConfig( bool b )
|
|
{
|
|
b_save_to_config = b;
|
|
}
|
|
|
|
AudioFilterControlWidget::AudioFilterControlWidget
|
|
( intf_thread_t *_p_intf, QWidget *parent, const char *_name ) :
|
|
QWidget( parent ), p_intf( _p_intf ), name( _name ), i_smallfont(0)
|
|
{}
|
|
|
|
void AudioFilterControlWidget::build()
|
|
{
|
|
QFont smallFont = QApplication::font();
|
|
smallFont.setPointSize( smallFont.pointSize() + i_smallfont );
|
|
|
|
QVBoxLayout *layout = new QVBoxLayout( this );
|
|
slidersBox = new QGroupBox( qtr( "Enable" ) );
|
|
slidersBox->setCheckable( true );
|
|
layout->addWidget( slidersBox );
|
|
|
|
QGridLayout *ctrlLayout = new QGridLayout( slidersBox );
|
|
|
|
int i = 0;
|
|
foreach( const FilterSliderData::slider_data_t &data, controls )
|
|
{
|
|
QSlider *slider = new QSlider( Qt::Vertical );
|
|
QLabel *valueLabel = new QLabel();
|
|
valueLabel->setFont( smallFont );
|
|
valueLabel->setAlignment( Qt::AlignHCenter );
|
|
QLabel *nameLabel = new QLabel();
|
|
nameLabel->setFont( smallFont );
|
|
nameLabel->setAlignment( Qt::AlignHCenter );
|
|
FilterSliderData *filter =
|
|
new FilterSliderData( this, p_intf,
|
|
slider, valueLabel, nameLabel, & data );
|
|
ctrlLayout->addWidget( slider, 0, i, Qt::AlignHCenter );
|
|
ctrlLayout->addWidget( valueLabel, 1, i, Qt::AlignHCenter );
|
|
ctrlLayout->addWidget( nameLabel, 2, i, Qt::AlignHCenter );
|
|
i++;
|
|
sliderDatas << filter;
|
|
}
|
|
|
|
vlc_object_t *p_aout = (vlc_object_t *)THEMIM->getAout();
|
|
char *psz_af;
|
|
if( p_aout )
|
|
{
|
|
psz_af = var_GetNonEmptyString( p_aout, "audio-filter" );
|
|
vlc_object_release( p_aout );
|
|
}
|
|
else
|
|
psz_af = config_GetPsz( p_intf, "audio-filter" );
|
|
|
|
if( psz_af && strstr( psz_af, qtu(name) ) != NULL )
|
|
slidersBox->setChecked( true );
|
|
else
|
|
slidersBox->setChecked( false );
|
|
CONNECT( slidersBox, toggled(bool), this, enable(bool) );
|
|
|
|
free( psz_af );
|
|
}
|
|
|
|
AudioFilterControlWidget::~AudioFilterControlWidget()
|
|
{
|
|
|
|
}
|
|
|
|
void AudioFilterControlWidget::enable( bool b_enable ) const
|
|
{
|
|
ChangeAFiltersString( p_intf, qtu(name), b_enable );
|
|
playlist_EnableAudioFilter( THEPL, qtu(name), b_enable );
|
|
}
|
|
|
|
void AudioFilterControlWidget::setSaveToConfig( bool b_save )
|
|
{
|
|
foreach( FilterSliderData *f, sliderDatas )
|
|
f->setSaveToConfig( b_save );
|
|
}
|
|
|
|
/**********************************************************************
|
|
* Equalizer
|
|
**********************************************************************/
|
|
|
|
EqualizerSliderData::EqualizerSliderData( QObject *parent, intf_thread_t *_p_intf,
|
|
QSlider *slider, QLabel *_label,
|
|
QLabel *_nameLabel, const slider_data_t *_p_data,
|
|
int _index )
|
|
: FilterSliderData( parent, slider ), index( _index )
|
|
{
|
|
p_intf = _p_intf;
|
|
valueLabel = _label;
|
|
nameLabel = _nameLabel;
|
|
p_data = _p_data;
|
|
|
|
slider->setMinimum( p_data->f_min / p_data->f_resolution );
|
|
slider->setMaximum( p_data->f_max / p_data->f_resolution );
|
|
nameLabel->setText( p_data->descs );
|
|
CONNECT( slider, valueChanged( int ), this, updateText( int ) );
|
|
setValue( initialValue() );
|
|
updateText( slider->value() );
|
|
CONNECT( slider, valueChanged( int ), this, onValueChanged( int ) );
|
|
}
|
|
|
|
QStringList EqualizerSliderData::getBandsFromAout() const
|
|
{
|
|
vlc_object_t *p_aout = (vlc_object_t *) THEMIM->getAout();
|
|
QStringList bands;
|
|
if( p_aout )
|
|
{
|
|
if ( var_Type( p_aout, qtu(p_data->name) ) == VLC_VAR_STRING )
|
|
{
|
|
char *psz_bands = var_GetString( p_aout, qtu(p_data->name) );
|
|
if ( psz_bands )
|
|
{
|
|
bands = QString( psz_bands ).split( " ", QString::SkipEmptyParts );
|
|
free( psz_bands );
|
|
}
|
|
}
|
|
vlc_object_release( p_aout );
|
|
}
|
|
|
|
if ( bands.count() ) return bands;
|
|
/* Or try config then */
|
|
|
|
if ( ! config_FindConfig( VLC_OBJECT(p_intf), qtu(p_data->name) ) )
|
|
return bands;
|
|
|
|
char *psz_bands = config_GetPsz( p_intf, qtu(p_data->name) );
|
|
if ( psz_bands )
|
|
{
|
|
bands = QString( psz_bands ).split( " ", QString::SkipEmptyParts );
|
|
free( psz_bands );
|
|
}
|
|
|
|
return bands;
|
|
}
|
|
|
|
float EqualizerSliderData::initialValue()
|
|
{
|
|
float f = p_data->f_value;
|
|
QStringList bands = getBandsFromAout();
|
|
|
|
if ( bands.count() > index )
|
|
f = QLocale( QLocale::C ).toFloat( bands[ index ] );
|
|
|
|
return f;
|
|
}
|
|
|
|
void EqualizerSliderData::onValueChanged( int i ) const
|
|
{
|
|
QStringList bands = getBandsFromAout();
|
|
if ( bands.count() > index )
|
|
{
|
|
float f = ((float) i) * p_data->f_resolution;
|
|
bands[ index ] = QLocale( QLocale::C ).toString( f );
|
|
vlc_object_t *p_aout = (vlc_object_t *) THEMIM->getAout();
|
|
if ( p_aout )
|
|
{
|
|
var_SetString( p_aout, qtu(p_data->name), qtu(bands.join( " " )) );
|
|
vlc_object_release( p_aout );
|
|
}
|
|
writeToConfig();
|
|
}
|
|
}
|
|
|
|
void EqualizerSliderData::writeToConfig() const
|
|
{
|
|
if ( !b_save_to_config ) return;
|
|
QStringList bands = getBandsFromAout();
|
|
if ( bands.count() > index )
|
|
{
|
|
float f = (float) slider->value() * p_data->f_resolution;
|
|
bands[ index ] = QLocale( QLocale::C ).toString( f );
|
|
config_PutPsz( p_intf, qtu(p_data->name), qtu(bands.join( " " )) );
|
|
}
|
|
}
|
|
|
|
Equalizer::Equalizer( intf_thread_t *p_intf, QWidget *parent )
|
|
: AudioFilterControlWidget( p_intf, parent, "equalizer" )
|
|
{
|
|
i_smallfont = -3;
|
|
bool b_vlcBands = var_InheritBool( p_intf, "equalizer-vlcfreqs" );
|
|
|
|
const FilterSliderData::slider_data_t vlc_bands[10] =
|
|
{
|
|
{ "equalizer-bands", qtr("60 Hz"), qtr("dB"), -20.0f, 20.0f, 0.0f, 0.1f, 1.0 },
|
|
{ "equalizer-bands", qtr("170 Hz"), qtr("dB"), -20.0f, 20.0f, 0.0f, 0.1f, 1.0 },
|
|
{ "equalizer-bands", qtr("310 Hz"), qtr("dB"), -20.0f, 20.0f, 0.0f, 0.1f, 1.0 },
|
|
{ "equalizer-bands", qtr("600 Hz"), qtr("dB"), -20.0f, 20.0f, 0.0f, 0.1f, 1.0 },
|
|
{ "equalizer-bands", qtr("1 KHz"), qtr("dB"), -20.0f, 20.0f, 0.0f, 0.1f, 1.0 },
|
|
{ "equalizer-bands", qtr("3 KHz"), qtr("dB"), -20.0f, 20.0f, 0.0f, 0.1f, 1.0 },
|
|
{ "equalizer-bands", qtr("6 KHz"), qtr("dB"), -20.0f, 20.0f, 0.0f, 0.1f, 1.0 },
|
|
{ "equalizer-bands", qtr("12 KHz"), qtr("dB"), -20.0f, 20.0f, 0.0f, 0.1f, 1.0 },
|
|
{ "equalizer-bands", qtr("14 KHz"), qtr("dB"), -20.0f, 20.0f, 0.0f, 0.1f, 1.0 },
|
|
{ "equalizer-bands", qtr("16 KHz"), qtr("dB"), -20.0f, 20.0f, 0.0f, 0.1f, 1.0 },
|
|
};
|
|
const FilterSliderData::slider_data_t iso_bands[10] =
|
|
{
|
|
{ "equalizer-bands", qtr("31 Hz"), qtr("dB"), -20.0f, 20.0f, 0.0f, 0.1f, 1.0 },
|
|
{ "equalizer-bands", qtr("63 Hz"), qtr("dB"), -20.0f, 20.0f, 0.0f, 0.1f, 1.0 },
|
|
{ "equalizer-bands", qtr("125 Hz"), qtr("dB"), -20.0f, 20.0f, 0.0f, 0.1f, 1.0 },
|
|
{ "equalizer-bands", qtr("250 Hz"), qtr("dB"), -20.0f, 20.0f, 0.0f, 0.1f, 1.0 },
|
|
{ "equalizer-bands", qtr("500 Hz"), qtr("dB"), -20.0f, 20.0f, 0.0f, 0.1f, 1.0 },
|
|
{ "equalizer-bands", qtr("1 KHz"), qtr("dB"), -20.0f, 20.0f, 0.0f, 0.1f, 1.0 },
|
|
{ "equalizer-bands", qtr("2 KHz"), qtr("dB"), -20.0f, 20.0f, 0.0f, 0.1f, 1.0 },
|
|
{ "equalizer-bands", qtr("4 KHz"), qtr("dB"), -20.0f, 20.0f, 0.0f, 0.1f, 1.0 },
|
|
{ "equalizer-bands", qtr("8 KHz"), qtr("dB"), -20.0f, 20.0f, 0.0f, 0.1f, 1.0 },
|
|
{ "equalizer-bands", qtr("16 KHz"), qtr("dB"), -20.0f, 20.0f, 0.0f, 0.1f, 1.0 },
|
|
};
|
|
const FilterSliderData::slider_data_t preamp_vals =
|
|
{ "equalizer-preamp", qtr("Preamp"), qtr("dB"), -20.0f, 20.0f, 0.0f, 0.1f, 1.0 };
|
|
|
|
for( int i=0; i<10 ;i++ ) controls.append( (b_vlcBands) ? vlc_bands[i] : iso_bands[i] );
|
|
preamp_values = preamp_vals;
|
|
build();
|
|
}
|
|
|
|
void Equalizer::build()
|
|
{
|
|
QFont smallFont = QApplication::font();
|
|
smallFont.setPointSize( smallFont.pointSize() + i_smallfont );
|
|
|
|
Ui::EqualizerWidget ui;
|
|
ui.setupUi( this );
|
|
|
|
QGridLayout *ctrlLayout = new QGridLayout( ui.slidersPlaceholder );
|
|
|
|
/* set up preamp control */
|
|
ui.preampLabel->setFont( smallFont );
|
|
ui.preampValue->setFont( smallFont );
|
|
preamp = new FilterSliderData( this, p_intf,
|
|
ui.preampSlider, ui.preampValue, ui.preampLabel, & preamp_values );
|
|
|
|
/* fix sliders spacing accurately */
|
|
int i_width = qMax( QFontMetrics( smallFont ).width( "500 Hz" ),
|
|
QFontMetrics( smallFont ).width( "-20.0 dB" ) );
|
|
int i = 0;
|
|
foreach( const FilterSliderData::slider_data_t &data, controls )
|
|
{
|
|
QSlider *slider = new QSlider( Qt::Vertical );
|
|
slider->setMinimumWidth( i_width );
|
|
QLabel *valueLabel = new QLabel();
|
|
valueLabel->setFont( smallFont );
|
|
valueLabel->setAlignment( Qt::AlignHCenter );
|
|
QLabel *nameLabel = new QLabel();
|
|
nameLabel->setFont( smallFont );
|
|
nameLabel->setAlignment( Qt::AlignHCenter );
|
|
EqualizerSliderData *filter =
|
|
new EqualizerSliderData( this, p_intf,
|
|
slider, valueLabel, nameLabel, & data, i );
|
|
ctrlLayout->addWidget( slider, 0, i, Qt::AlignHCenter );
|
|
ctrlLayout->addWidget( valueLabel, 2, i, Qt::AlignHCenter );
|
|
ctrlLayout->addWidget( nameLabel, 1, i, Qt::AlignHCenter );
|
|
sliderDatas << filter; /* keep track for applying presets */
|
|
i++;
|
|
}
|
|
|
|
/* Add the listed presets */
|
|
ui.presetsCombo->addItem( "", QVariant() ); /* 1st entry = custom/modified */
|
|
for( i = 0 ; i < NB_PRESETS ; i ++ )
|
|
{
|
|
QGraphicsScene scene;
|
|
QPixmap icon( 40, 40 );
|
|
icon.fill( Qt::transparent );
|
|
QPainter painter( &icon );
|
|
for ( int j = 0; j < eqz_preset_10b[i].i_band; j++ )
|
|
{
|
|
float f_value = eqz_preset_10b[i].f_amp[j];
|
|
if ( f_value > 20.0 ) f_value = 20.0;
|
|
if ( f_value < -20.0 ) f_value = -20.0;
|
|
QRectF shape( j, 20.0 - f_value, 1, f_value );
|
|
scene.addRect( shape, QPen(), palette().brush( QPalette::WindowText ) );
|
|
}
|
|
scene.addLine( 0.0, 20.0, eqz_preset_10b[i].i_band, 20.0,
|
|
palette().color( QPalette::WindowText ) );
|
|
scene.setSceneRect( 0.0, 0.0, eqz_preset_10b[i].i_band , 40.0 );
|
|
scene.render( &painter, icon.rect(), scene.sceneRect(), Qt::IgnoreAspectRatio );
|
|
ui.presetsCombo->addItem( icon, qtr( preset_list_text[i] ),
|
|
QVariant( preset_list[i] ) );
|
|
}
|
|
CONNECT( ui.presetsCombo, activated(int), this, setCorePreset(int) );
|
|
|
|
/* Set enable checkbox */
|
|
vlc_object_t *p_aout = (vlc_object_t *)THEMIM->getAout();
|
|
char *psz_af;
|
|
if( p_aout )
|
|
{
|
|
psz_af = var_GetNonEmptyString( p_aout, "audio-filter" );
|
|
vlc_object_release( p_aout );
|
|
}
|
|
else
|
|
psz_af = config_GetPsz( p_intf, "audio-filter" );
|
|
|
|
/* To enable or disable subwidgets */
|
|
/* If that list grows, better iterate over layout's childs */
|
|
CONNECT( ui.enableCheck, toggled(bool), ui.presetsCombo, setEnabled(bool) );
|
|
CONNECT( ui.enableCheck, toggled(bool), ui.presetLabel, setEnabled(bool) );
|
|
CONNECT( ui.enableCheck, toggled(bool), ui.eq2PassCheck, setEnabled(bool) );
|
|
CONNECT( ui.enableCheck, toggled(bool), ui.slidersPlaceholder, setEnabled(bool) );
|
|
CONNECT( ui.enableCheck, toggled(bool), ui.preampSlider, setEnabled(bool) );
|
|
CONNECT( ui.enableCheck, toggled(bool), ui.preampValue, setEnabled(bool) );
|
|
CONNECT( ui.enableCheck, toggled(bool), ui.preampLabel, setEnabled(bool) );
|
|
|
|
if( psz_af && strstr( psz_af, qtu(name) ) != NULL )
|
|
ui.enableCheck->setChecked( true );
|
|
else
|
|
ui.enableCheck->setChecked( false );
|
|
|
|
/* workaround for non emitted toggle() signal */
|
|
ui.enableCheck->toggle(); ui.enableCheck->toggle();
|
|
|
|
free( psz_af );
|
|
CONNECT( ui.enableCheck, toggled(bool), this, enable(bool) );
|
|
|
|
/* Connect and set 2 Pass checkbox */
|
|
ui.eq2PassCheck->setChecked( var_InheritBool( p_aout, "equalizer-2pass" ) );
|
|
CONNECT( ui.eq2PassCheck, toggled(bool), this, enable2Pass(bool) );
|
|
}
|
|
|
|
void Equalizer::setSaveToConfig( bool b_save )
|
|
{
|
|
AudioFilterControlWidget::setSaveToConfig( b_save );
|
|
preamp->setSaveToConfig( b_save );
|
|
}
|
|
|
|
void Equalizer::setCorePreset( int i_preset )
|
|
{
|
|
if( i_preset < 1 )
|
|
return;
|
|
|
|
i_preset--;/* 1st in index was an empty entry */
|
|
|
|
preamp->setValue( eqz_preset_10b[i_preset].f_preamp );
|
|
for ( int i=0; i< qMin( eqz_preset_10b[i_preset].i_band,
|
|
sliderDatas.count() ) ; i++ )
|
|
sliderDatas[i]->setValue( eqz_preset_10b[i_preset].f_amp[i] );
|
|
|
|
vlc_object_t *p_aout = (vlc_object_t *)THEMIM->getAout();
|
|
if( p_aout )
|
|
{
|
|
var_SetString( p_aout , "equalizer-preset" , preset_list[i_preset] );
|
|
vlc_object_release( p_aout );
|
|
}
|
|
config_PutPsz( p_intf, "equalizer-preset", preset_list[i_preset] );
|
|
}
|
|
|
|
/* Function called when the set2Pass button is activated */
|
|
void Equalizer::enable2Pass( bool b_enable ) const
|
|
{
|
|
vlc_object_t *p_aout= (vlc_object_t *)THEMIM->getAout();
|
|
|
|
if( p_aout )
|
|
{
|
|
var_SetBool( p_aout, "equalizer-2pass", b_enable );
|
|
vlc_object_release( p_aout );
|
|
}
|
|
config_PutInt( p_intf, "equalizer-2pass", b_enable );
|
|
}
|
|
|
|
/**********************************************************************
|
|
* Audio filters
|
|
**********************************************************************/
|
|
|
|
/**********************************************************************
|
|
* Dynamic range compressor
|
|
**********************************************************************/
|
|
|
|
Compressor::Compressor( intf_thread_t *p_intf, QWidget *parent )
|
|
: AudioFilterControlWidget( p_intf, parent, "compressor" )
|
|
{
|
|
i_smallfont = -2;
|
|
const FilterSliderData::slider_data_t a[7] =
|
|
{
|
|
{ "compressor-rms-peak", qtr("RMS/peak"), "", 0.0f, 1.0f, 0.00f, 0.001f, 1.0 },
|
|
{ "compressor-attack", qtr("Attack"), qtr("ms"), 1.5f, 400.0f, 25.00f, 0.100f, 1.0 },
|
|
{ "compressor-release", qtr("Release"), qtr("ms"), 2.0f, 800.0f, 100.00f, 0.100f, 1.0 },
|
|
{ "compressor-threshold", qtr("Threshold"), qtr("dB"), -30.0f, 0.0f, -11.00f, 0.010f, 1.0 },
|
|
{ "compressor-ratio", qtr("Ratio"), ":1", 1.0f, 20.0f, 8.00f, 0.010f, 1.0 },
|
|
{ "compressor-knee", qtr("Knee\nradius"), qtr("dB"), 1.0f, 10.0f, 2.50f, 0.010f, 1.0 },
|
|
{ "compressor-makeup-gain", qtr("Makeup\ngain"), qtr("dB"), 0.0f, 24.0f, 7.00f, 0.010f, 1.0 },
|
|
};
|
|
for( int i=0; i<7 ;i++ ) controls.append( a[i] );
|
|
build();
|
|
}
|
|
|
|
/**********************************************************************
|
|
* Spatializer
|
|
**********************************************************************/
|
|
|
|
Spatializer::Spatializer( intf_thread_t *p_intf, QWidget *parent )
|
|
: AudioFilterControlWidget( p_intf, parent, "spatializer" )
|
|
{
|
|
i_smallfont = -1;
|
|
const FilterSliderData::slider_data_t a[5] =
|
|
{
|
|
{ "spatializer-roomsize", qtr("Size"), "", 0.0f, 1.1f, 0.85f, 0.1f, 10.0 },
|
|
{ "spatializer-width", qtr("Width"), "", 0.0f, 1.0f, 1.0f, 0.1f, 10.0 },
|
|
{ "spatializer-wet", qtr("Wet"), "", 0.0f, 1.0f, 0.4f, 0.1f, 10.0 },
|
|
{ "spatializer-dry", qtr("Dry"), "", 0.0f, 1.0f, 0.5f, 0.1f, 10.0 },
|
|
{ "spatializer-damp", qtr("Damp"), "", 0.0f, 1.0f, 0.5f, 0.1f, 10.0 },
|
|
};
|
|
for( int i=0; i<5 ;i++ ) controls.append( a[i] );
|
|
build();
|
|
}
|
|
|
|
/**********************************************************************
|
|
* Spatializer
|
|
**********************************************************************/
|
|
|
|
StereoWidener::StereoWidener( intf_thread_t *p_intf, QWidget *parent )
|
|
: AudioFilterControlWidget( p_intf, parent, "stereo_widen" )
|
|
{
|
|
i_smallfont = -1;
|
|
const FilterSliderData::slider_data_t a[4] =
|
|
{
|
|
{ "stereowiden-delay", N_("Delay time"), "ms", 1.0, 100, 20, 1.0, 1.0 },
|
|
{ "stereowiden-feedback", N_("Feedback gain"), "%", 0.0, 0.9, 0.3, 0.1, 1.0 },
|
|
{ "stereowiden-crossfeed", N_("Crossfeed"), "%", 0.0, 0.8, 0.3, 0.1, 1.0 },
|
|
{ "stereowiden-dry-mix", N_("Dry mix"), "%", 0.0, 1.0, 0.8, 0.1, 1.0 },
|
|
};
|
|
for( int i=0; i<4 ;i++ ) controls.append( a[i] );
|
|
build();
|
|
}
|
|
|
|
#include <QToolButton>
|
|
#include <QGridLayout>
|
|
|
|
#define SUBSDELAY_CFG_MODE "subsdelay-mode"
|
|
#define SUBSDELAY_CFG_FACTOR "subsdelay-factor"
|
|
#define SUBSDELAY_MODE_ABSOLUTE 0
|
|
#define SUBSDELAY_MODE_RELATIVE_SOURCE_DELAY 1
|
|
#define SUBSDELAY_MODE_RELATIVE_SOURCE_CONTENT 2
|
|
|
|
SyncWidget::SyncWidget( QWidget *_parent ) : QWidget( _parent )
|
|
{
|
|
QHBoxLayout *layout = new QHBoxLayout;
|
|
spinBox.setAlignment( Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter );
|
|
spinBox.setDecimals( 3 );
|
|
spinBox.setMinimum( -600.0 );
|
|
spinBox.setMaximum( 600.0 );
|
|
spinBox.setSingleStep( 0.1 );
|
|
spinBox.setSuffix( " s" );
|
|
spinBox.setButtonSymbols( QDoubleSpinBox::PlusMinus );
|
|
CONNECT( &spinBox, valueChanged( double ), this, valueChangedHandler( double ) );
|
|
layout->addWidget( &spinBox );
|
|
layout->addWidget( &spinLabel );
|
|
layout->setContentsMargins( 0, 0, 0, 0 );
|
|
setLayout( layout );
|
|
}
|
|
|
|
void SyncWidget::valueChangedHandler( double d )
|
|
{
|
|
if ( d < 0 )
|
|
spinLabel.setText( qtr("(Hastened)") );
|
|
else if ( d > 0 )
|
|
spinLabel.setText( qtr("(Delayed)") );
|
|
else
|
|
spinLabel.setText( "" );
|
|
emit valueChanged( d );
|
|
}
|
|
|
|
void SyncWidget::setValue( double d )
|
|
{
|
|
spinBox.setValue( d );
|
|
}
|
|
|
|
SyncControls::SyncControls( intf_thread_t *_p_intf, QWidget *_parent ) :
|
|
QWidget( _parent ) , p_intf( _p_intf )
|
|
{
|
|
QGroupBox *AVBox, *subsBox;
|
|
QToolButton *updateButton;
|
|
|
|
b_userAction = true;
|
|
|
|
QGridLayout *mainLayout = new QGridLayout( this );
|
|
|
|
/* AV sync */
|
|
AVBox = new QGroupBox( qtr( "Audio/Video" ) );
|
|
QGridLayout *AVLayout = new QGridLayout( AVBox );
|
|
|
|
QLabel *AVLabel = new QLabel;
|
|
AVLabel->setText( qtr( "Audio track synchronization:" ) );
|
|
AVLayout->addWidget( AVLabel, 0, 0, 1, 1 );
|
|
|
|
AVSpin = new SyncWidget( this );
|
|
AVLayout->addWidget( AVSpin, 0, 2, 1, 1 );
|
|
mainLayout->addWidget( AVBox, 1, 0, 1, 5 );
|
|
|
|
/* Subs */
|
|
subsBox = new QGroupBox( qtr( "Subtitles/Video" ) );
|
|
QGridLayout *subsLayout = new QGridLayout( subsBox );
|
|
|
|
QLabel *subsLabel = new QLabel;
|
|
subsLabel->setText( qtr( "Subtitle track synchronization:" ) );
|
|
subsLayout->addWidget( subsLabel, 0, 0, 1, 1 );
|
|
|
|
subsSpin = new SyncWidget( this );
|
|
subsLayout->addWidget( subsSpin, 0, 2, 1, 1 );
|
|
|
|
QLabel *subSpeedLabel = new QLabel;
|
|
subSpeedLabel->setText( qtr( "Subtitle speed:" ) );
|
|
subsLayout->addWidget( subSpeedLabel, 1, 0, 1, 1 );
|
|
|
|
subSpeedSpin = new QDoubleSpinBox;
|
|
subSpeedSpin->setAlignment( Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter );
|
|
subSpeedSpin->setDecimals( 3 );
|
|
subSpeedSpin->setMinimum( 1 );
|
|
subSpeedSpin->setMaximum( 100 );
|
|
subSpeedSpin->setSingleStep( 0.2 );
|
|
subSpeedSpin->setSuffix( " fps" );
|
|
subSpeedSpin->setButtonSymbols( QDoubleSpinBox::PlusMinus );
|
|
subsLayout->addWidget( subSpeedSpin, 1, 2, 1, 1 );
|
|
|
|
QLabel *subDurationLabel = new QLabel;
|
|
subDurationLabel->setText( qtr( "Subtitle duration factor:" ) );
|
|
subsLayout->addWidget( subDurationLabel, 2, 0, 1, 1 );
|
|
|
|
subDurationSpin = new QDoubleSpinBox;
|
|
subDurationSpin->setAlignment( Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter );
|
|
subDurationSpin->setDecimals( 3 );
|
|
subDurationSpin->setMinimum( 0 );
|
|
subDurationSpin->setMaximum( 20 );
|
|
subDurationSpin->setSingleStep( 0.2 );
|
|
subDurationSpin->setButtonSymbols( QDoubleSpinBox::PlusMinus );
|
|
subsLayout->addWidget( subDurationSpin, 2, 2, 1, 1 );
|
|
|
|
mainLayout->addWidget( subsBox, 2, 0, 2, 5 );
|
|
|
|
updateButton = new QToolButton;
|
|
updateButton->setAutoRaise( true );
|
|
mainLayout->addWidget( updateButton, 0, 4, 1, 1 );
|
|
|
|
/* Various Connects */
|
|
CONNECT( AVSpin, valueChanged ( double ), this, advanceAudio( double ) ) ;
|
|
CONNECT( subsSpin, valueChanged ( double ), this, advanceSubs( double ) ) ;
|
|
CONNECT( subSpeedSpin, valueChanged ( double ),
|
|
this, adjustSubsSpeed( double ) );
|
|
CONNECT( subDurationSpin, valueChanged ( double ),
|
|
this, adjustSubsDuration( double ) );
|
|
|
|
CONNECT( THEMIM->getIM(), synchroChanged(), this, update() );
|
|
BUTTON_SET_ACT_I( updateButton, "", update,
|
|
qtr( "Force update of this dialog's values" ), update() );
|
|
|
|
initSubsDuration();
|
|
|
|
/* Set it */
|
|
update();
|
|
}
|
|
|
|
SyncControls::~SyncControls()
|
|
{
|
|
subsdelayClean();
|
|
}
|
|
|
|
void SyncControls::clean()
|
|
{
|
|
b_userAction = false;
|
|
AVSpin->setValue( 0.0 );
|
|
subsSpin->setValue( 0.0 );
|
|
subSpeedSpin->setValue( 1.0 );
|
|
subsdelayClean();
|
|
b_userAction = true;
|
|
}
|
|
|
|
void SyncControls::update()
|
|
{
|
|
b_userAction = false;
|
|
|
|
int64_t i_delay;
|
|
if( THEMIM->getInput() )
|
|
{
|
|
i_delay = var_GetInteger( THEMIM->getInput(), "audio-delay" );
|
|
AVSpin->setValue( ( (double)i_delay ) / CLOCK_FREQ );
|
|
i_delay = var_GetInteger( THEMIM->getInput(), "spu-delay" );
|
|
subsSpin->setValue( ( (double)i_delay ) / CLOCK_FREQ );
|
|
subSpeedSpin->setValue( var_GetFloat( THEMIM->getInput(), "sub-fps" ) );
|
|
subDurationSpin->setValue( var_InheritFloat( p_intf, SUBSDELAY_CFG_FACTOR ) );
|
|
}
|
|
b_userAction = true;
|
|
}
|
|
|
|
void SyncControls::advanceAudio( double f_advance )
|
|
{
|
|
if( THEMIM->getInput() && b_userAction )
|
|
{
|
|
int64_t i_delay = f_advance * CLOCK_FREQ;
|
|
var_SetInteger( THEMIM->getInput(), "audio-delay", i_delay );
|
|
}
|
|
}
|
|
|
|
void SyncControls::advanceSubs( double f_advance )
|
|
{
|
|
if( THEMIM->getInput() && b_userAction )
|
|
{
|
|
int64_t i_delay = f_advance * CLOCK_FREQ;
|
|
var_SetInteger( THEMIM->getInput(), "spu-delay", i_delay );
|
|
}
|
|
}
|
|
|
|
void SyncControls::adjustSubsSpeed( double f_fps )
|
|
{
|
|
if( THEMIM->getInput() && b_userAction )
|
|
{
|
|
var_SetFloat( THEMIM->getInput(), "sub-fps", f_fps );
|
|
}
|
|
}
|
|
|
|
void SyncControls::adjustSubsDuration( double f_factor )
|
|
{
|
|
if( THEMIM->getInput() && b_userAction )
|
|
{
|
|
subsdelaySetFactor( f_factor );
|
|
ChangeVFiltersString( p_intf, "subsdelay", f_factor > 0 );
|
|
}
|
|
}
|
|
|
|
void SyncControls::initSubsDuration()
|
|
{
|
|
int i_mode = var_InheritInteger( p_intf, SUBSDELAY_CFG_MODE );
|
|
|
|
switch (i_mode)
|
|
{
|
|
default:
|
|
case SUBSDELAY_MODE_ABSOLUTE:
|
|
subDurationSpin->setToolTip( qtr( "Extend subtitle duration by this value.\n"
|
|
"Set 0 to disable." ) );
|
|
subDurationSpin->setSuffix( " s" );
|
|
break;
|
|
case SUBSDELAY_MODE_RELATIVE_SOURCE_DELAY:
|
|
subDurationSpin->setToolTip( qtr( "Multiply subtitle duration by this value.\n"
|
|
"Set 0 to disable." ) );
|
|
subDurationSpin->setSuffix( "" );
|
|
break;
|
|
case SUBSDELAY_MODE_RELATIVE_SOURCE_CONTENT:
|
|
subDurationSpin->setToolTip( qtr( "Recalculate subtitle duration according\n"
|
|
"to their content and this value.\n"
|
|
"Set 0 to disable." ) );
|
|
subDurationSpin->setSuffix( "" );
|
|
break;
|
|
}
|
|
}
|
|
|
|
void SyncControls::subsdelayClean()
|
|
{
|
|
/* Remove subsdelay filter */
|
|
ChangeVFiltersString( p_intf, "subsdelay", false );
|
|
}
|
|
|
|
void SyncControls::subsdelaySetFactor( double f_factor )
|
|
{
|
|
/* Set the factor in the preferences */
|
|
config_PutFloat( p_intf, SUBSDELAY_CFG_FACTOR, f_factor );
|
|
|
|
/* Try to find an instance of subsdelay, and set its factor */
|
|
vlc_object_t *p_obj = ( vlc_object_t * ) vlc_object_find_name( p_intf->obj.libvlc, "subsdelay" );
|
|
if( p_obj )
|
|
{
|
|
var_SetFloat( p_obj, SUBSDELAY_CFG_FACTOR, f_factor );
|
|
vlc_object_release( p_obj );
|
|
}
|
|
}
|
|
|
|
|
|
/**********************************************************************
|
|
* Video filters / Adjust
|
|
**********************************************************************/
|
|
|
|
/**********************************************************************
|
|
* Extended playbak controls
|
|
**********************************************************************/
|
|
|