extended_panels.cpp 55.4 KB
Newer Older
Clément Stenac's avatar
Clément Stenac committed
1
/*****************************************************************************
Clément Stenac's avatar
Clément Stenac committed
2
 * extended_panels.cpp : Extended controls panels
Clément Stenac's avatar
Clément Stenac committed
3
 ****************************************************************************
Ludovic Fauvet's avatar
Ludovic Fauvet committed
4
 * Copyright (C) 2006-2013 the VideoLAN team
5
 * $Id$
Clément Stenac's avatar
Clément Stenac committed
6 7
 *
 * Authors: Clément Stenac <zorglub@videolan.org>
8
 *          Antoine Cellerier <dionoea .t videolan d@t org>
9
 *          Jean-Baptiste Kempf <jb@videolan.org>
Clément Stenac's avatar
Clément Stenac committed
10 11 12 13
 *
 * 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
14
 * ( at your option ) any later version.
Clément Stenac's avatar
Clément Stenac committed
15 16 17 18 19 20 21 22 23 24
 *
 * 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.
 *****************************************************************************/
25

26 27 28
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
Clément Stenac's avatar
Clément Stenac committed
29

30 31
#include <math.h>

Clément Stenac's avatar
Clément Stenac committed
32 33 34 35 36
#include <QLabel>
#include <QVariant>
#include <QString>
#include <QFont>
#include <QGridLayout>
37
#include <QComboBox>
38
#include <QTimer>
39
#include <QFileDialog>
40 41
#include <QGraphicsScene>
#include <QPainter>
42
#include <QRegExp>
Clément Stenac's avatar
Clément Stenac committed
43

Clément Stenac's avatar
Clément Stenac committed
44
#include "components/extended_panels.hpp"
45
#include "dialogs/preferences.hpp"
Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
46
#include "qt.hpp"
47
#include "input_manager.hpp"
48
#include "util/qt_dirs.hpp"
49
#include "util/customwidgets.hpp"
Clément Stenac's avatar
Clément Stenac committed
50

Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
51
#include "../../audio_filter/equalizer_presets.h"
Clément Stenac's avatar
Clément Stenac committed
52
#include <vlc_vout.h>
53
#include <vlc_modules.h>
54
#include <vlc_plugin.h>
Clément Stenac's avatar
Clément Stenac committed
55

56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78
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;
    }
}
79

80
static const QString ModuleFromWidgetName( QObject *obj )
81
{
82
    return obj->objectName().replace( "Enable","" );
83 84
}

85
static QString OptionFromWidgetName( QObject *obj )
86 87
{
    /* Gruik ? ... nah */
88 89 90 91
    return obj->objectName()
        .remove( QRegExp( "Slider|Combo|Dial|Check|Spin|Text" ) )
        .replace( QRegExp( "([A-Z])" ), "-\\1" )
        .toLower();
92
}
Clément Stenac's avatar
Clément Stenac committed
93

94
static inline void setup_vfilter( intf_thread_t *p_intf, const char* psz_name, QWidget *widget )
Clément Stenac's avatar
Clément Stenac committed
95
{
96
    vlc_object_t *p_obj = ( vlc_object_t * )
97
        vlc_object_find_name( p_intf->obj.libvlc, psz_name );
98 99 100 101 102 103 104 105 106 107 108 109 110 111
    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 );
    }
}
Clément Stenac's avatar
Clément Stenac committed
112

113
#define SETUP_VFILTER( widget ) \
114
    setup_vfilter( p_intf, #widget, ui.widget##Enable ); \
115
    CONNECT( ui.widget##Enable, clicked(), this, updateFilters() );
116

117
#define SETUP_VFILTER_OPTION( widget, signal ) \
118
    initComboBoxItems( ui.widget ); \
119 120
    setWidgetValue( ui.widget ); \
    CONNECT( ui.widget, signal, this, updateFilterOptions() );
Antoine Cellerier's avatar
Antoine Cellerier committed
121

122 123 124 125 126
ExtVideo::ExtVideo( intf_thread_t *_p_intf, QTabWidget *_parent ) :
            QObject( _parent ), p_intf( _p_intf )
{
    ui.setupUi( _parent );

127
    SETUP_VFILTER( adjust )
128 129 130 131 132 133
    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 ) )
134 135

    SETUP_VFILTER( extract )
136
    SETUP_VFILTER_OPTION( extractComponentText, textChanged( const QString& ) )
137

138 139
    SETUP_VFILTER( posterize )

140
    SETUP_VFILTER( colorthres )
141
    SETUP_VFILTER_OPTION( colorthresColorText, textChanged( const QString& ) )
142 143
    SETUP_VFILTER_OPTION( colorthresSaturationthresSlider, valueChanged( int ) )
    SETUP_VFILTER_OPTION( colorthresSimilaritythresSlider, valueChanged( int ) )
144

145 146 147
    SETUP_VFILTER( sepia )
    SETUP_VFILTER_OPTION( sepiaIntensitySpin, valueChanged( int ) )

148 149 150
    SETUP_VFILTER( invert )

    SETUP_VFILTER( gradient )
151 152 153
    SETUP_VFILTER_OPTION( gradientModeCombo, currentIndexChanged( QString ) )
    SETUP_VFILTER_OPTION( gradientTypeCheck, stateChanged( int ) )
    SETUP_VFILTER_OPTION( gradientCartoonCheck, stateChanged( int ) )
154

155
    SETUP_VFILTER( motionblur )
156
    SETUP_VFILTER_OPTION( blurFactorSlider, valueChanged( int ) )
157 158 159 160 161 162

    SETUP_VFILTER( motiondetect )

    SETUP_VFILTER( psychedelic )

    SETUP_VFILTER( sharpen )
163
    SETUP_VFILTER_OPTION( sharpenSigmaSlider, valueChanged( int ) )
164 165 166 167 168 169

    SETUP_VFILTER( ripple )

    SETUP_VFILTER( wave )

    SETUP_VFILTER( transform )
170
    SETUP_VFILTER_OPTION( transformTypeCombo, currentIndexChanged( QString ) )
171 172

    SETUP_VFILTER( rotate )
173
    SETUP_VFILTER_OPTION( rotateAngleDial, valueChanged( int ) )
174 175
    ui.rotateAngleDial->setWrapping( true );
    ui.rotateAngleDial->setNotchesVisible( true );
176 177

    SETUP_VFILTER( puzzle )
178 179
    SETUP_VFILTER_OPTION( puzzleRowsSpin, valueChanged( int ) )
    SETUP_VFILTER_OPTION( puzzleColsSpin, valueChanged( int ) )
180 181 182 183

    SETUP_VFILTER( magnify )

    SETUP_VFILTER( clone )
184
    SETUP_VFILTER_OPTION( cloneCountSpin, valueChanged( int ) )
185 186

    SETUP_VFILTER( wall )
187 188
    SETUP_VFILTER_OPTION( wallRowsSpin, valueChanged( int ) )
    SETUP_VFILTER_OPTION( wallColsSpin, valueChanged( int ) )
189

Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
190

191 192
    SETUP_VFILTER( erase )
    SETUP_VFILTER_OPTION( eraseMaskText, editingFinished() )
193 194
    SETUP_VFILTER_OPTION( eraseYSpin, valueChanged( int ) )
    SETUP_VFILTER_OPTION( eraseXSpin, valueChanged( int ) )
195
    BUTTONACT( ui.eraseBrowseBtn, browseEraseFile() );
196 197

    SETUP_VFILTER( marq )
198
    SETUP_VFILTER_OPTION( marqMarqueeText, textChanged( const QString& ) )
199
    SETUP_VFILTER_OPTION( marqPositionCombo, currentIndexChanged( QString ) )
200 201 202

    SETUP_VFILTER( logo )
    SETUP_VFILTER_OPTION( logoFileText, editingFinished() )
203 204
    SETUP_VFILTER_OPTION( logoYSpin, valueChanged( int ) )
    SETUP_VFILTER_OPTION( logoXSpin, valueChanged( int ) )
205
    SETUP_VFILTER_OPTION( logoOpacitySlider, valueChanged( int ) )
206
    BUTTONACT( ui.logoBrowseBtn, browseLogo() );
207

208 209 210
    SETUP_VFILTER( gradfun )
    SETUP_VFILTER_OPTION( gradfunRadiusSlider, valueChanged( int ) )

Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
211 212 213
    SETUP_VFILTER( grain )
    SETUP_VFILTER_OPTION( grainVarianceSlider, valueChanged( int ) )

214 215 216
    SETUP_VFILTER( mirror )

    SETUP_VFILTER( gaussianblur )
217
    SETUP_VFILTER_OPTION( gaussianblurSigmaSlider, valueChanged( int ) )
218 219 220 221

    SETUP_VFILTER( antiflicker )
    SETUP_VFILTER_OPTION( antiflickerSofteningSizeSlider, valueChanged( int ) )

222 223 224 225 226 227
    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 ) )

228

229 230
    SETUP_VFILTER( anaglyph )

231 232
#undef SETUP_VFILTER
#undef SETUP_VFILTER_OPTION
233 234 235 236 237

    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() );
Dylan Yudaken's avatar
Dylan Yudaken committed
238 239
    CONNECT( ui.leftRightCropSync, toggled ( bool ), this, cropChange() );
    CONNECT( ui.topBotCropSync, toggled ( bool ), this, cropChange() );
240 241 242 243
    CONNECT( ui.topBotCropSync, toggled( bool ),
             ui.cropBotPx, setDisabled( bool ) );
    CONNECT( ui.leftRightCropSync, toggled( bool ),
             ui.cropRightPx, setDisabled( bool ) );
Clément Stenac's avatar
Clément Stenac committed
244 245
}

246 247
void ExtVideo::cropChange()
{
Dylan Yudaken's avatar
Dylan Yudaken committed
248 249 250 251 252
    if( ui.topBotCropSync->isChecked() )
        ui.cropBotPx->setValue( ui.cropTopPx->value() );
    if( ui.leftRightCropSync->isChecked() )
        ui.cropRightPx->setValue( ui.cropLeftPx->value() );

253
    vout_thread_t *p_vout = THEMIM->getVout();
254
    if( p_vout )
255
    {
256 257 258 259
        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() );
260
        vlc_object_release( p_vout );
261 262 263
    }
}

264 265 266 267 268 269 270 271
void ExtVideo::clean()
{
    ui.cropTopPx->setValue( 0 );
    ui.cropBotPx->setValue( 0 );
    ui.cropLeftPx->setValue( 0 );
    ui.cropRightPx->setValue( 0 );
}

272
static QString ChangeFiltersString( struct intf_thread_t *p_intf, const char *psz_filter_type, const char *psz_name, bool b_add )
Clément Stenac's avatar
Clément Stenac committed
273
{
274
    char* psz_chain = config_GetPsz( p_intf, psz_filter_type );
275

276 277
    QString const chain = QString( psz_chain ? psz_chain : "" );
    QStringList list = chain.split( ':', QString::SplitBehavior::SkipEmptyParts );
Clément Stenac's avatar
Clément Stenac committed
278

279 280
    if( b_add ) list << psz_name;
    else        list.removeAll( psz_name );
Clément Stenac's avatar
Clément Stenac committed
281

282
    free( psz_chain );
Clément Stenac's avatar
Clément Stenac committed
283

KO Myung-Hun's avatar
KO Myung-Hun committed
284
    return list.join( ":" );
285 286
}

287 288 289 290 291 292 293 294 295
static void ChangeAFiltersString( struct intf_thread_t *p_intf, const char *psz_name, bool b_add )
{
    module_t *p_obj = module_find( psz_name );
    if( !p_obj )
    {
        msg_Err( p_intf, "Unable to find filter module \"%s\".", psz_name );
        return;
    }

296
    QString result = ChangeFiltersString( p_intf, "audio-filter", psz_name, b_add );
297
    config_PutPsz( p_intf, "audio-filter", qtu( result ) );
298 299
}

300 301 302
static void ChangeVFiltersString( struct intf_thread_t *p_intf, const char *psz_name, bool b_add )
{
    const char *psz_filter_type = GetVFilterType( p_intf, psz_name );
303

304 305 306
    if( psz_filter_type == NULL )
        return;

307
    QString result = ChangeFiltersString( p_intf, psz_filter_type, psz_name, b_add );
308

Clément Stenac's avatar
Clément Stenac committed
309
    /* Vout is not kept, so put that in the config */
310
    config_PutPsz( p_intf, psz_filter_type, qtu( result ) );
Clément Stenac's avatar
Clément Stenac committed
311 312

    /* Try to set on the fly */
313
    if( !strcmp( psz_filter_type, "video-splitter" ) )
Clément Stenac's avatar
Clément Stenac committed
314
    {
315
        playlist_t *p_playlist = THEPL;
316
        var_SetString( p_playlist, psz_filter_type, qtu( result ) );
317 318 319
    }
    else
    {
320
        vout_thread_t *p_vout = THEMIM->getVout();
321 322
        if( p_vout )
        {
323
            var_SetString( p_vout, psz_filter_type, qtu( result ) );
324 325
            vlc_object_release( p_vout );
        }
Clément Stenac's avatar
Clément Stenac committed
326 327 328 329 330
    }
}

void ExtVideo::updateFilters()
{
331
    QString module = ModuleFromWidgetName( sender() );
Clément Stenac's avatar
Clément Stenac committed
332

333 334
    QCheckBox *checkbox = qobject_cast<QCheckBox*>( sender() );
    QGroupBox *groupbox = qobject_cast<QGroupBox*>( sender() );
335

336
    ChangeVFiltersString( p_intf, qtu( module ),
337 338
                          checkbox ? checkbox->isChecked()
                                   : groupbox->isChecked() );
Clément Stenac's avatar
Clément Stenac committed
339 340
}

341 342 343 344 345 346
#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& ) ) );

347 348 349 350
void ExtVideo::browseLogo()
{
    QString file = QFileDialog::getOpenFileName( NULL, qtr( "Logo filenames" ),
                   p_intf->p_sys->filepath, "Images (*.png *.jpg);;All (*)" );
351 352

    UPDATE_AND_APPLY_TEXT( logoFileText, file );
353 354 355 356 357 358
}

void ExtVideo::browseEraseFile()
{
    QString file = QFileDialog::getOpenFileName( NULL, qtr( "Image mask" ),
                   p_intf->p_sys->filepath, "Images (*.png *.jpg);;All (*)" );
359 360

    UPDATE_AND_APPLY_TEXT( eraseMaskText, file );
361 362
}

363 364
#undef UPDATE_AND_APPLY_TEXT

365 366
void ExtVideo::initComboBoxItems( QObject *widget )
{
367
    QComboBox *combobox = qobject_cast<QComboBox*>( widget );
368
    if( !combobox ) return;
369

370
    QString option = OptionFromWidgetName( widget );
371
    module_config_t *p_item = config_FindConfig( VLC_OBJECT( p_intf ),
372
                                                 qtu( option ) );
373
    if( p_item == NULL )
374
    {
375 376 377 378 379 380 381 382 383 384 385 386
        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++ )
387
        {
388
            combobox->addItem( qtr( texts[i] ), qlonglong(values[i]) );
389
            free( texts[i] );
390
        }
391 392
        free( texts );
        free( values );
393
    }
394
    else if( p_item->i_type == CONFIG_ITEM_STRING )
395
    {
396 397 398 399 400 401
        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++ )
        {
402
            combobox->addItem( qtr( texts[i] ), qfu(values[i]) );
403 404 405 406 407
            free( texts[i] );
            free( values[i] );
        }
        free( texts );
        free( values );
408 409 410
    }
}

411 412 413 414 415 416 417
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;

418
    vlc_object_t *p_obj = ( vlc_object_t * )
419
        vlc_object_find_name( p_intf->obj.libvlc, qtu( module ) );
420 421 422 423 424
    int i_type;
    vlc_value_t val;

    if( !p_obj )
    {
425
#if 0
426 427
        msg_Dbg( p_intf,
                 "Module instance %s not found, looking in config values.",
428
                 qtu( module ) );
429
#endif
430
        i_type = config_GetType( p_intf, qtu( option ) ) & VLC_VAR_CLASS;
431 432 433 434
        switch( i_type )
        {
            case VLC_VAR_INTEGER:
            case VLC_VAR_BOOL:
435
                val.i_int = config_GetInt( p_intf, qtu( option ) );
436 437
                break;
            case VLC_VAR_FLOAT:
438
                val.f_float = config_GetFloat( p_intf, qtu( option ) );
439 440
                break;
            case VLC_VAR_STRING:
441
                val.psz_string = config_GetPsz( p_intf, qtu( option ) );
442 443 444 445 446
                break;
        }
    }
    else
    {
447
        i_type = var_Type( p_obj, qtu( option ) ) & VLC_VAR_CLASS;
448
        var_Get( p_obj, qtu( option ), &val );
449 450 451 452 453
        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. */
454 455 456 457
    QSlider        *slider        = qobject_cast<QSlider*>       ( widget );
    QCheckBox      *checkbox      = qobject_cast<QCheckBox*>     ( widget );
    QSpinBox       *spinbox       = qobject_cast<QSpinBox*>      ( widget );
    QDoubleSpinBox *doublespinbox = qobject_cast<QDoubleSpinBox*>( widget );
458
    VLCQDial       *dial          = qobject_cast<VLCQDial*>      ( widget );
459 460
    QLineEdit      *lineedit      = qobject_cast<QLineEdit*>     ( widget );
    QComboBox      *combobox      = qobject_cast<QComboBox*>     ( widget );
461 462 463 464 465 466 467

    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 );
468
        else if( dial )     dial->setValue( (360 - val.i_int) % 360 );
469 470 471
        else if( lineedit )
        {
            char str[30];
Tristan Matthews's avatar
Tristan Matthews committed
472
            snprintf( str, sizeof(str), "%06" PRIX64, val.i_int );
473 474
            lineedit->setText( str );
        }
475
        else if( combobox ) combobox->setCurrentIndex(
476
                            combobox->findData( qlonglong(val.i_int) ) );
477
        else msg_Warn( p_intf, "Could not find the correct Integer widget" );
478 479 480
    }
    else if( i_type == VLC_VAR_FLOAT )
    {
481 482
        if( slider ) slider->setValue( ( int )( val.f_float*( double )slider->tickInterval() ) ); /* hack alert! */
        else if( doublespinbox ) doublespinbox->setValue( val.f_float );
483
        else if( dial ) dial->setValue( (360 - lroundf(val.f_float)) % 360 );
484
        else msg_Warn( p_intf, "Could not find the correct Float widget" );
485 486 487
    }
    else if( i_type == VLC_VAR_STRING )
    {
488
        if( lineedit ) lineedit->setText( qfu( val.psz_string ) );
489
        else if( combobox ) combobox->setCurrentIndex(
490
                            combobox->findData( qfu( val.psz_string ) ) );
491
        else msg_Warn( p_intf, "Could not find the correct String widget" );
492 493 494
        free( val.psz_string );
    }
    else
495 496 497 498 499 500
        if( p_obj )
            msg_Err( p_intf,
                     "Module %s's %s variable is of an unsupported type ( %d )",
                     qtu( module ),
                     qtu( option ),
                     i_type );
501 502
}

503 504
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 )
505
{
506
    vlc_object_t *p_obj = ( vlc_object_t * )vlc_object_find_name( p_intf->obj.libvlc, psz_module );
507 508
    int i_type;
    bool b_is_command;
509

510 511
    if( !p_obj )
    {
512 513
        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 );
514 515 516 517
        b_is_command = false;
    }
    else
    {
518
        i_type = var_Type( p_obj, psz_option );
519
        if( i_type == 0 )
520
            i_type = config_GetType( p_intf, psz_option );
521
        b_is_command = ( i_type & VLC_VAR_ISCOMMAND );
522 523
    }

524
    i_type &= VLC_VAR_CLASS;
525 526
    if( i_type == VLC_VAR_INTEGER || i_type == VLC_VAR_BOOL )
    {
527 528 529
        if( i_int == -1 )
            msg_Warn( p_intf, "Could not find the correct Integer widget" );
        config_PutInt( p_intf, psz_option, i_int );
530 531 532
        if( b_is_command )
        {
            if( i_type == VLC_VAR_INTEGER )
533
                var_SetInteger( p_obj, psz_option, i_int );
534
            else
535
                var_SetBool( p_obj, psz_option, i_int );
536
        }
537 538 539
    }
    else if( i_type == VLC_VAR_FLOAT )
    {
540 541 542
        if( f_float == -1 )
            msg_Warn( p_intf, "Could not find the correct Float widget" );
        config_PutFloat( p_intf, psz_option, f_float );
543
        if( b_is_command )
544
            var_SetFloat( p_obj, psz_option, f_float );
545 546 547
    }
    else if( i_type == VLC_VAR_STRING )
    {
548
        if( val.isNull() )
549
            msg_Warn( p_intf, "Could not find the correct String widget" );
550
        config_PutPsz( p_intf, psz_option, qtu( val ) );
551
        if( b_is_command )
552
            var_SetString( p_obj, psz_option, qtu( val ) );
553 554 555
    }
    else
        msg_Err( p_intf,
556
                 "Module %s's %s variable is of an unsupported type ( %d )",
557 558
                 psz_module,
                 psz_option,
559 560
                 i_type );

561 562 563
    if( !b_is_command )
    {
        msg_Warn( p_intf, "Module %s's %s variable isn't a command. Brute-restarting the filter.",
564 565 566 567
                 psz_module,
                 psz_option );
        ChangeVFiltersString( p_intf, psz_module, false );
        ChangeVFiltersString( p_intf, psz_module, true );
568 569
    }

570
    if( p_obj ) vlc_object_release( p_obj );
571 572
}

573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601
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 ) {
602 603
        i_int = (360 - dial->value()) % 360;
        f_float = i_int;
604 605 606 607 608 609 610 611 612 613 614 615 616 617
    }
    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);
}

618 619 620 621 622
/**********************************************************************
 * v4l2 controls
 **********************************************************************/

ExtV4l2::ExtV4l2( intf_thread_t *_p_intf, QWidget *_parent )
623
    : QWidget( _parent ), p_intf( _p_intf ), box( NULL )
624
{
625 626 627 628 629 630
    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 );
631
    help->setWordWrap( true );
632 633
    layout->addWidget( help );
    setLayout( layout );
634 635
}

636 637 638 639 640 641
void ExtV4l2::showEvent( QShowEvent *event )
{
    QWidget::showEvent( event );
    Refresh();
}

642 643
void ExtV4l2::Refresh( void )
{
644
    vlc_object_t *p_obj = (vlc_object_t*)vlc_object_find_name( THEPL, "v4l2" );
645
    help->hide();
646 647
    if( box )
    {
648
        layout()->removeWidget( box );
649 650 651
        delete box;
        box = NULL;
    }
652 653
    if( p_obj )
    {
654
        vlc_value_t val, text;
655 656 657 658
        int i_ret = var_Change( p_obj, "controls", VLC_VAR_GETCHOICES,
                                &val, &text );
        if( i_ret < 0 )
        {
659
            msg_Err( p_intf, "Oops, v4l2 object doesn't have a 'controls' variable." );
660
            help->show();
661 662 663
            vlc_object_release( p_obj );
            return;
        }
664 665

        box = new QGroupBox( this );
666
        layout()->addWidget( box );
667 668 669
        QVBoxLayout *layout = new QVBoxLayout( box );
        box->setLayout( layout );

670 671
        for( int i = 0; i < val.p_list->i_count; i++ )
        {
672
            vlc_value_t vartext;
673
            const char *psz_var = text.p_list->p_values[i].psz_string;
674 675 676 677 678 679

            if( var_Change( p_obj, psz_var, VLC_VAR_GETTEXT, &vartext, NULL ) )
                continue;

            QString name = qtr( vartext.psz_string );
            free( vartext.psz_string );
Tristan Matthews's avatar
Tristan Matthews committed
680
            msg_Dbg( p_intf, "v4l2 control \"%" PRIx64 "\": %s (%s)",
681
                     val.p_list->p_values[i].i_int, psz_var, qtu( name ) );
682 683 684 685 686 687

            int i_type = var_Type( p_obj, psz_var );
            switch( i_type & VLC_VAR_TYPE )
            {
                case VLC_VAR_INTEGER:
                {
688
                    QLabel *label = new QLabel( name, box );
689
                    QHBoxLayout *hlayout = new QHBoxLayout();
690 691 692 693
                    hlayout->addWidget( label );
                    int i_val = var_GetInteger( p_obj, psz_var );
                    if( i_type & VLC_VAR_HASCHOICE )
                    {
694
                        QComboBox *combobox = new QComboBox( box );
695
                        combobox->setObjectName( qfu( psz_var ) );
696 697 698 699 700 701 702 703

                        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,
Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
704
                                       qlonglong( val2.p_list->p_values[j].i_int) );
705 706 707
                            if( i_val == val2.p_list->p_values[j].i_int )
                                combobox->setCurrentIndex( j );
                        }
708
                        var_FreeList( &val2, &text2 );
709 710 711 712 713 714 715

                        CONNECT( combobox, currentIndexChanged( int ), this,
                                 ValueChange( int ) );
                        hlayout->addWidget( combobox );
                    }
                    else
                    {
716
                        QSlider *slider = new QSlider( box );
717
                        slider->setObjectName( qfu( psz_var ) );
718 719 720 721
                        slider->setOrientation( Qt::Horizontal );
                        vlc_value_t val2;
                        var_Change( p_obj, psz_var, VLC_VAR_GETMIN,
                                    &val2, NULL );
722 723
                        if( val2.i_int < INT_MIN )
                            val2.i_int = INT_MIN; /* FIXME */
724 725 726
                        slider->setMinimum( val2.i_int );
                        var_Change( p_obj, psz_var, VLC_VAR_GETMAX,
                                    &val2, NULL );
727 728
                        if( val2.i_int > INT_MAX )
                            val2.i_int = INT_MAX; /* FIXME */
729
                        slider->setMaximum( val2.i_int );
730 731 732
                        if( !var_Change( p_obj, psz_var, VLC_VAR_GETSTEP,
                                         &val2, NULL ) )
                            slider->setSingleStep( val2.i_int );
733 734 735 736 737 738 739 740 741 742
                        slider->setValue( i_val );
                        CONNECT( slider, valueChanged( int ), this,
                                 ValueChange( int ) );
                        hlayout->addWidget( slider );
                    }
                    layout->addLayout( hlayout );
                    break;
                }
                case VLC_VAR_BOOL:
                {
743
                    QCheckBox *button = new QCheckBox( name, box );
744
                    button->setObjectName( qfu( psz_var ) );
745 746 747 748 749 750 751 752 753
                    button->setChecked( var_GetBool( p_obj, psz_var ) );

                    CONNECT( button, clicked( bool ), this,
                             ValueChange( bool ) );
                    layout->addWidget( button );
                    break;
                }
                case VLC_VAR_VOID:
                {
754 755
                    if( i_type & VLC_VAR_ISCOMMAND )
                    {
756
                        QPushButton *button = new QPushButton( name, box );
757
                        button->setObjectName( qfu( psz_var ) );
758

759 760 761 762 763 764
                        CONNECT( button, clicked( bool ), this,
                                 ValueChange( bool ) );
                        layout->addWidget( button );
                    }
                    else
                    {
765
                        QLabel *label = new QLabel( name, box );
766 767
                        layout->addWidget( label );
                    }
768 769 770 771 772 773 774
                    break;
                }
                default:
                    msg_Warn( p_intf, "Unhandled var type for %s", psz_var );
                    break;
            }
        }
775
        var_FreeList( &val, &text );
776 777 778 779 780
        vlc_object_release( p_obj );
    }
    else
    {
        msg_Dbg( p_intf, "Couldn't find v4l2 instance" );
781 782 783
        help->show();
        if ( isVisible() )
            QTimer::singleShot( 2000, this, SLOT(Refresh()) );
784 785 786 787 788 789 790 791 792 793 794
    }
}

void ExtV4l2::ValueChange( bool value )
{
    ValueChange( (int)value );
}

void ExtV4l2::ValueChange( int value )
{
    QObject *s = sender();
795
    vlc_object_t *p_obj = (vlc_object_t*)vlc_object_find_name( THEPL, "v4l2" );
796 797
    if( p_obj )
    {
798 799
        QString var = s->objectName();
        int i_type = var_Type( p_obj, qtu( var ) );
800 801 802 803 804 805 806 807
        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();
                }
808
                var_SetInteger( p_obj, qtu( var ), value );
809 810
                break;
            case VLC_VAR_BOOL:
811
                var_SetBool( p_obj, qtu( var ), value );
812 813
                break;
            case VLC_VAR_VOID:
814
                var_TriggerCallback( p_obj, qtu( var ) );
815 816 817 818 819 820 821 822 823 824 825
                break;
        }
        vlc_object_release( p_obj );
    }
    else
    {
        msg_Warn( p_intf, "Oops, v4l2 object isn't available anymore" );
        Refresh();
    }
}

826 827 828 829
/**********************************************************************
 * Sliders
 **********************************************************************/

830 831
FilterSliderData::FilterSliderData( QObject *parent, QSlider *_slider ) :
    QObject( parent ), slider( _slider )
832 833 834
{
    b_save_to_config = false;
}
835

836 837 838 839 840
FilterSliderData::FilterSliderData( QObject *parent,
                                    intf_thread_t *_p_intf,
                                    QSlider *_slider,
                                    QLabel *_label, QLabel *_nameLabel,
                                    const slider_data_t *_p_data ):
841 842
    QObject( parent ), slider( _slider ), valueLabel( _label ),
    nameLabel( _nameLabel ), p_data( _p_data ), p_intf( _p_intf )
843
{
844
    b_save_to_config = false;
845 846
    slider->setMinimum( p_data->f_min / p_data->f_resolution );
    slider->setMaximum( p_data->f_max / p_data->f_resolution );
847
    nameLabel->setText( p_data->descs );
848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863
    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 )
{
864
    float f = ((float) i) * p_data->f_resolution * p_data->f_visual_multiplier;
865
    valueLabel->setText( QString( p_data->units )
866
                    .prepend( "%1 " )
867 868 869 870 871 872 873 874 875
                    .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 )
    {
876
        if ( var_Type( p_aout, qtu(p_data->name) ) == 0 )
877 878 879 880 881 882
        {
            vlc_object_release( p_aout );
            /* Not found, will try in config */
        }
        else
        {
883
            f = var_GetFloat( p_aout, qtu(p_data->name) );
884 885 886 887 888
            vlc_object_release( p_aout );
            return f;
        }
    }

889
    if ( ! config_FindConfig( VLC_OBJECT(p_intf), qtu(p_data->name) ) )
890 891
        return f;

892
    f = config_GetFloat( p_intf, qtu(p_data->name) );
893 894 895
    return f;
}

896
void FilterSliderData::onValueChanged( int i ) const
897 898 899 900 901
{
    float f = ((float) i) * p_data->f_resolution;
    vlc_object_t *p_aout = (vlc_object_t *) THEMIM->getAout();
    if ( p_aout )
    {
902
        var_SetFloat( p_aout, qtu(p_data->name), f );
903 904 905 906 907
        vlc_object_release( p_aout );
    }
    writeToConfig();
}

908
void FilterSliderData::writeToConfig() const
909
{
910
    if ( !b_save_to_config ) return;
911
    float f = ((float) slider->value()) * p_data->f_resolution;
912 913 914
    config_PutFloat( p_intf, qtu(p_data->name), f );
}

915 916 917 918 919
void FilterSliderData::setSaveToConfig( bool b )
{
    b_save_to_config = b;
}

920 921
AudioFilterControlWidget::AudioFilterControlWidget
( intf_thread_t *_p_intf, QWidget *parent, const char *_name ) :
922
    QWidget( parent ), p_intf( _p_intf ), name( _name ), i_smallfont(0)
923 924 925 926 927
{}

void AudioFilterControlWidget::build()
{
    QFont smallFont = QApplication::font();
928
    smallFont.setPointSize( smallFont.pointSize() + i_smallfont );