/***************************************************************************** * simple_preferences.cpp : "Simple preferences" **************************************************************************** * Copyright (C) 2006-2010 the VideoLAN team * * Authors: Clément Stenac * Antoine Cellerier * Jean-Baptiste Kempf * * 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 "simple_preferences.hpp" #include "preferences_widgets.hpp" #include "maininterface/mainctx.hpp" #include "util/color_scheme_model.hpp" #include "util/qvlcapp.hpp" #include "util/proxycolumnmodel.hpp" #include "medialibrary/mlrecentsmodel.hpp" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #define ICON_HEIGHT 48 #define ICON_WIDTH 48 #ifdef _WIN32 # include # include #endif #include static struct { const char iso[6]; const char name[34]; } const language_map[] = { { "auto", N_("Auto") }, { "en", "American English" }, { "ar", "عربي" }, { "bn", "বাংলা" }, { "pt_BR", "Português Brasileiro" }, { "en_GB", "British English" }, { "el", "Νέα Ελληνικά" }, { "bg", "български език" }, { "ca", "Català" }, { "zh_TW", "正體中文" }, { "cs", "Čeština" }, { "cy", "Cymraeg" }, { "da", "Dansk" }, { "nl", "Nederlands" }, { "fi", "Suomi" }, { "et", "eesti keel" }, { "eu", "Euskara" }, { "fr", "Français" }, { "ga", "Gaeilge" }, { "gd", "Gàidhlig" }, { "gl", "Galego" }, { "ka", "ქართული" }, { "de", "Deutsch" }, { "he", "עברית" }, { "hr", "hrvatski" }, { "hu", "Magyar" }, { "hy", "հայերեն" }, { "is", "íslenska" }, { "id", "Bahasa Indonesia" }, { "it", "Italiano" }, { "ja", "日本語" }, { "ko", "한국어" }, { "lt", "lietuvių" }, { "mn", "Монгол хэл" }, { "ms", "Melayu" }, { "nb", "Bokmål" }, { "nn", "Nynorsk" }, { "kk", "Қазақ тілі" }, { "km", "ភាសាខ្មែរ" }, { "ne", "नेपाली" }, { "oc", "Occitan" }, { "fa", "فارسی" }, { "pl", "Polski" }, { "pt_PT", "Português" }, { "pa", "ਪੰਜਾਬੀ" }, { "ro", "Română" }, { "ru", "Русский" }, { "zh_CN", "简体中文" }, { "si", "සිංහල" }, { "sr", "српски" }, { "sk", "Slovensky" }, { "sl", "slovenščina" }, { "ckb", "کوردیی سۆرانی" }, { "es", "Español" }, { "sv", "Svenska" }, { "te", "తెలుగు" }, { "tr", "Türkçe" }, { "uk", "украї́нська мо́ва" }, { "vi", "tiếng Việt" }, { "wa", "Walon" } }; static int getDefaultAudioVolume(const char *aout) { if (!strcmp(aout, "") || !strcmp(aout, "any")) #ifdef _WIN32 /* All Windows aouts, that can be selected automatically, handle volume * saving. In case of automatic mode, we'll save the last volume for * every modules. Therefore, all volumes variable we be the same and we * can use the first one (mmdevice). */ return config_GetFloat("mmdevice-volume") * 100.f + .5f; #else return -1; #endif /* Note: For hysterical raisins, this is sorted by decreasing priority * order (then alphabetical order). */ if (!strcmp(aout, "pulse")) return -1; #ifdef __linux__ if (!strcmp(aout, "alsa") && module_exists("alsa")) return cbrtf(config_GetFloat("alsa-gain")) * 100.f + .5f; #endif #ifdef _WIN32 if (!strcmp(aout, "mmdevice")) return config_GetFloat("mmdevice-volume") * 100.f + .5f; #endif #ifdef __APPLE__ if (!strcmp(aout, "auhal") && module_exists("auhal")) return (config_GetFloat("auhal-volume") * 100.f + .5f) / AOUT_VOLUME_DEFAULT; #endif #ifdef _WIN32 if (!strcmp(aout, "directsound") && module_exists("directsound")) return config_GetFloat("directx-volume") * 100.f + .5f; #endif if (!strcmp(aout, "jack")) return cbrtf(config_GetFloat("jack-gain")) * 100.f + 0.5f; #ifdef __OS2__ if (!strcmp(aout, "kai")) return cbrtf(config_GetFloat("kai-gain")) * 100.f + .5f; #endif #ifdef _WIN32 if (!strcmp(aout, "waveout")) return config_GetFloat("waveout-volume") * 100.f + .5f; #endif return -1; } namespace { void fillStylesCombo( QComboBox *stylesCombo, const QString &initialStyle) { stylesCombo->addItem( qtr("System's default") ); stylesCombo->addItems( QStyleFactory::keys() ); stylesCombo->setCurrentIndex( stylesCombo->findText( initialStyle ) ); stylesCombo->insertSeparator( 1 ); if ( stylesCombo->currentIndex() < 0 ) stylesCombo->setCurrentIndex( 0 ); /* default */ } QString getQStyleKey(const QComboBox *stylesCombo, const QString &defaultStyleName) { vlc_assert( stylesCombo ); const int index = stylesCombo->currentIndex(); if (stylesCombo->currentIndex() == 0) return defaultStyleName; return QStyleFactory::keys().at( index - 2 ); } } class PropertyResetter { public: PropertyResetter(QWidget *control, const char * property) : m_control {control} , m_property {property} , m_initialValue {m_control->property(property)} { } void reset() { bool success = m_control->setProperty(m_property.data(), m_initialValue); vlc_assert(success); } private: QWidget *m_control; const QByteArray m_property; const QVariant m_initialValue; }; /********************************************************************* * The List of categories *********************************************************************/ SPrefsCatList::SPrefsCatList( qt_intf_t *_p_intf, QWidget *_parent ) : QWidget( _parent ), p_intf( _p_intf ) { QHBoxLayout *layout = new QHBoxLayout(); /* Use autoExclusive buttons and a mapper as QButtonGroup can't set focus (keys) when it manages the buttons's exclusivity. See QT bugs 131 & 816 and QAbstractButton's source code. */ QSignalMapper *mapper = new QSignalMapper( layout ); connect( mapper, QSIGNALMAPPER_MAPPEDINT_SIGNAL, this, &SPrefsCatList::switchPanel ); QPixmap scaled; qreal dpr = devicePixelRatioF(); #define ADD_CATEGORY( button, label, ltooltip, icon, numb ) \ QToolButton * button = new QToolButton( this ); \ /* Scale icon to non native size outside of toolbutton to avoid widget size */\ /* computation using native size */\ scaled = QPixmap( ":/prefsmenu/" #icon ".png" )\ .scaledToHeight( ICON_HEIGHT * dpr, Qt::SmoothTransformation );\ scaled.setDevicePixelRatio( dpr ); \ button->setIcon( scaled ); \ button->setText( label ); \ button->setToolTip( ltooltip ); \ button->setToolButtonStyle( Qt::ToolButtonTextUnderIcon ); \ button->setIconSize( QSize( ICON_WIDTH, ICON_HEIGHT ) ); \ button->setMinimumWidth( 40 + ICON_WIDTH );\ button->setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::Minimum); \ button->setAutoRaise( true ); \ button->setCheckable( true ); \ button->setAutoExclusive( true ); \ connect( button, &QToolButton::clicked, mapper, QOverload<>::of(&QSignalMapper::map) ); \ mapper->setMapping( button, numb ); \ layout->addWidget( button ); ADD_CATEGORY( SPrefsInterface, qfut(INTF_TITLE), qfut(INTF_TOOLTIP), cone_interface_64, 0 ); ADD_CATEGORY( SPrefsAudio, qfut(AUDIO_TITLE), qfut(AUDIO_TOOLTIP), cone_audio_64, 1 ); ADD_CATEGORY( SPrefsVideo, qfut(VIDEO_TITLE), qfut(VIDEO_TOOLTIP), cone_video_64, 2 ); ADD_CATEGORY( SPrefsSubtitles, qfut(SUBPIC_TITLE), qfut(SUBPIC_TOOLTIP), cone_subtitles_64, 3 ); ADD_CATEGORY( SPrefsInputAndCodecs, qfut(INPUT_TITLE), qfut(INPUT_TOOLTIP), cone_input_64, 4 ); ADD_CATEGORY( SPrefsHotkeys, qfut(HOTKEYS_TITLE), qfut(HOTKEYS_TOOLTIP), cone_hotkeys_64, 5 ); ADD_CATEGORY( SPrefsMediaLibrary, qfut(ML_TITLE), qfut(ML_TOOLTIP), cone_medialibrary_64, 6 ); #undef ADD_CATEGORY SPrefsInterface->setChecked( true ); layout->setMargin( 0 ); layout->setSpacing( 1 ); setSizePolicy(QSizePolicy::MinimumExpanding,QSizePolicy::Preferred); setMinimumWidth( ICON_HEIGHT * 6 + 10 ); setLayout( layout ); } void SPrefsCatList::switchPanel( int i ) { emit currentItemChanged( i ); } /********************************************************************* * The Panels *********************************************************************/ SPrefsPanel::SPrefsPanel( qt_intf_t *_p_intf, QWidget *_parent, int _number ) : QWidget( _parent ), p_intf( _p_intf ) { module_config_t *p_config; ConfigControl *control; number = _number; lang = NULL; radioGroup = NULL; #define CONFIG_GENERIC( option, type, label, qcontrol ) \ p_config = config_FindConfig( option ); \ if( p_config ) \ { \ control = new type ## ConfigControl( \ p_config, label, ui.qcontrol ); \ controls.append( control ); \ } \ else { \ QWidget *label_ = label; \ ui.qcontrol->setEnabled( false ); \ if( label_ ) label_->setEnabled( false ); \ } #define CONFIG_BOOL( option, qcontrol ) \ p_config = config_FindConfig( option ); \ if( p_config ) \ { \ control = new BoolConfigControl( \ p_config, NULL, ui.qcontrol ); \ controls.append( control ); \ } \ else { ui.qcontrol->setEnabled( false ); } #define CONFIG_GENERIC_NO_UI( option, type, label, qcontrol ) \ p_config = config_FindConfig( option ); \ if( p_config ) \ { \ control = new type ## ConfigControl( \ p_config, label, qcontrol ); \ controls.append( control ); \ } \ else { \ QWidget *widget = label; \ qcontrol->setVisible( false ); \ if( widget ) widget->setEnabled( false ); \ } #define CONFIG_GENERIC_FILE( option, type, label, qcontrol, qbutton ) \ p_config = config_FindConfig( option ); \ if( p_config ) \ { \ control = new type ## ConfigControl( \ p_config, label, qcontrol, qbutton ); \ controls.append( control ); \ } \ else { \ qcontrol->setEnabled( false ); \ if( label ) label->setEnabled( false ); \ if( qbutton ) qbutton->setEnabled( false ); \ } #define START_SPREFS_CAT( name , label ) \ case SPrefs ## name: \ { \ Ui::SPrefs ## name ui; \ ui.setupUi( panel ); \ panel_label->setText( label ); #define END_SPREFS_CAT \ break; \ } QVBoxLayout *panel_layout = new QVBoxLayout(); QWidget *panel = new QWidget(); panel_layout->setMargin( 3 ); // Title Label QLabel *panel_label = new QLabel; QFont labelFont = QApplication::font(); labelFont.setPointSize( labelFont.pointSize() + 6 ); panel_label->setFont( labelFont ); // Title
QFrame *title_line = new QFrame; title_line->setFrameShape(QFrame::HLine); title_line->setFrameShadow(QFrame::Sunken); QFont italicFont = QApplication::font(); italicFont.setItalic( true ); switch( number ) { /****************************** * VIDEO Panel Implementation * ******************************/ START_SPREFS_CAT( Video, qtr("Video Settings") ); CONFIG_BOOL( "video", enableVideo ); ui.videoZone->setEnabled( ui.enableVideo->isChecked() ); connect( ui.enableVideo, &QCheckBox::toggled, ui.videoZone, &QWidget::setEnabled ); CONFIG_BOOL( "fullscreen", fullscreen ); CONFIG_BOOL( "video-deco", windowDecorations ); CONFIG_GENERIC( "vout", StringList, ui.voutLabel, outputModule ); optionWidgets["videoOutCoB"] = ui.outputModule; optionWidgets["fullscreenScreenB"] = ui.fullscreenScreenBox; ui.fullscreenScreenBox->addItem( qtr("Automatic"), -1 ); int i_screenCount = 0; foreach( QScreen* screen, QGuiApplication::screens() ) { ui.fullscreenScreenBox->addItem( screen->name(), i_screenCount ); i_screenCount++; } p_config = config_FindConfig( "qt-fullscreen-screennumber" ); if( p_config ) { int i_defaultScreen = p_config->value.i + 1; if ( i_defaultScreen < 0 || i_defaultScreen > ( ui.fullscreenScreenBox->count() - 1 ) ) ui.fullscreenScreenBox->setCurrentIndex( 0 ); else ui.fullscreenScreenBox->setCurrentIndex(p_config->value.i + 1); } #ifdef _WIN32 CONFIG_BOOL( "directx-hw-yuv", hwYUVBox ); #else ui.directXBox->setVisible( false ); #endif #ifdef __OS2__ CONFIG_BOOL( "kva-fixt23", kvaFixT23 ); CONFIG_GENERIC( "kva-video-mode", StringList, ui.kvaVideoModeLabel, kvaVideoMode ); #else ui.kvaBox->setVisible( false ); #endif CONFIG_GENERIC( "deinterlace", IntegerList, ui.deinterLabel, deinterlaceBox ); CONFIG_GENERIC( "deinterlace-mode", StringList, ui.deinterModeLabel, deinterlaceModeBox ); CONFIG_GENERIC( "aspect-ratio", String, ui.arLabel, arLine ); CONFIG_GENERIC_FILE( "snapshot-path", Directory, ui.dirLabel, ui.snapshotsDirectory, ui.snapshotsDirectoryBrowse ); CONFIG_GENERIC( "snapshot-prefix", String, ui.prefixLabel, snapshotsPrefix ); CONFIG_BOOL( "snapshot-sequential", snapshotsSequentialNumbering ); CONFIG_GENERIC( "snapshot-format", StringList, ui.arLabel, snapshotsFormat ); END_SPREFS_CAT; /****************************** * AUDIO Panel Implementation * ******************************/ START_SPREFS_CAT( Audio, qtr("Audio Settings") ); CONFIG_BOOL( "audio", enableAudio ); ui.audioZone->setEnabled( ui.enableAudio->isChecked() ); connect( ui.enableAudio, &QCheckBox::toggled, ui.audioZone, &QWidget::setEnabled ); #define audioCommon( name ) \ QLabel * name ## Label = new QLabel( qtr( "Device:" ) ); \ name ## Label->setMinimumSize(QSize(250, 0)); \ outputAudioLayout->addWidget( name ## Label, outputAudioLayout->rowCount(), 0, 1, 1 ); \ #define audioControl( name) \ audioCommon( name ) \ QComboBox * name ## Device = new QComboBox; \ name ## Label->setBuddy( name ## Device ); \ name ## Device->setSizePolicy( QSizePolicy::Ignored, QSizePolicy::Preferred );\ outputAudioLayout->addWidget( name ## Device, outputAudioLayout->rowCount() - 1, 1, 1, -1 ); #define audioControl2( name) \ audioCommon( name ) \ QHBoxLayout * name ## hboxLayout = new QHBoxLayout; \ QLineEdit * name ## Device = new QLineEdit; \ name ## Label->setBuddy( name ## Device ); \ name ## hboxLayout->addWidget( name ## Device ); \ QPushButton * name ## Browse = new QPushButton( qtr( "Browse..." ) ); \ name ## hboxLayout->addWidget( name ## Browse ); \ outputAudioLayout->addLayout( name ## hboxLayout, outputAudioLayout->rowCount() - 1, 1, 1, 1, Qt::AlignLeft ); /* Build if necessary */ QGridLayout * outputAudioLayout = qobject_cast(ui.outputAudioBox->layout()); #ifdef _WIN32 audioControl( DirectX ); optionWidgets["directxL" ] = DirectXLabel; optionWidgets["directxW" ] = DirectXDevice; CONFIG_GENERIC_NO_UI( "directx-audio-device", StringList, DirectXLabel, DirectXDevice ); audioControl( Waveout ); optionWidgets["waveoutL" ] = WaveoutLabel; optionWidgets["waveoutW" ] = WaveoutDevice; CONFIG_GENERIC_NO_UI( "waveout-audio-device", StringList, WaveoutLabel, WaveoutDevice ); #elif defined( __OS2__ ) audioControl( kai ); optionWidgets["kaiL"] = kaiLabel; optionWidgets["kaiW"] = kaiDevice; CONFIG_GENERIC_NO_UI( "kai-audio-device", StringList, kaiLabel, kaiDevice ); #else if( module_exists( "alsa" ) ) { audioControl( alsa ); optionWidgets["alsaL"] = alsaLabel; optionWidgets["alsaW"] = alsaDevice; CONFIG_GENERIC_NO_UI( "alsa-audio-device" , StringList, alsaLabel, alsaDevice ); } if( module_exists( "oss" ) ) { audioControl2( OSS ); optionWidgets["ossL"] = OSSLabel; optionWidgets["ossW"] = OSSDevice; optionWidgets["ossB"] = OSSBrowse; CONFIG_GENERIC_FILE( "oss-audio-device" , File, OSSLabel, OSSDevice, OSSBrowse ); } #endif #ifdef _WIN32 audioControl( MMDevice ); optionWidgets["mmdeviceL" ] = MMDeviceLabel; optionWidgets["mmdeviceW" ] = MMDeviceDevice; CONFIG_GENERIC_NO_UI( "mmdevice-audio-device", StringList, MMDeviceLabel, MMDeviceDevice ); CONFIG_GENERIC( "mmdevice-passthrough", IntegerList, ui.mmdevicePassthroughLabel, mmdevicePassthroughBox ); optionWidgets["mmdevicePassthroughL"] = ui.mmdevicePassthroughLabel; optionWidgets["mmdevicePassthroughB"] = ui.mmdevicePassthroughBox; #else ui.mmdevicePassthroughLabel->setVisible( false ); ui.mmdevicePassthroughBox->setVisible( false ); #endif #undef audioControl2 #undef audioControl #undef audioCommon int i_max_volume = config_GetInt( "qt-max-volume" ); /* Audio Options */ ui.volumeValue->setMaximum( i_max_volume ); ui.defaultVolume->setMaximum( i_max_volume ); connect( ui.defaultVolume, &QSlider::valueChanged, this, &SPrefsPanel::updateAudioVolume ); ui.defaultVolume_zone->setEnabled( ui.resetVolumeCheckbox->isChecked() ); connect( ui.resetVolumeCheckbox, &QCheckBox::toggled, ui.defaultVolume_zone, &QWidget::setEnabled ); CONFIG_GENERIC( "audio-language" , String , ui.langLabel, preferredAudioLanguage ); CONFIG_BOOL( "spdif", spdifBox ); if( !module_exists( "normvol" ) ) ui.volNormBox->setEnabled( false ); else { CONFIG_GENERIC( "norm-max-level" , Float, nullptr, volNormSpin ); } CONFIG_GENERIC( "audio-replay-gain-mode", StringList, ui.replayLabel, replayCombo ); CONFIG_GENERIC( "audio-visual" , StringList, ui.visuLabel, visualisation); CONFIG_BOOL( "audio-time-stretch", autoscaleBox ); /* Audio Output Specifics */ CONFIG_GENERIC( "aout", StringList, ui.outputLabel, outputModule ); connect( ui.outputModule, QOverload::of(&QComboBox::currentIndexChanged), this, &SPrefsPanel::updateAudioOptions ); /* File output exists on all platforms */ CONFIG_GENERIC_FILE( "audiofile-file", File, ui.fileLabel, ui.fileName, ui.fileBrowseButton ); optionWidgets["fileW"] = ui.fileControl; optionWidgets["audioOutCoB"] = ui.outputModule; optionWidgets["normalizerChB"] = ui.volNormBox; /*Little modification of ui.volumeValue to compile with Qt < 4.3 */ ui.volumeValue->setButtonSymbols(QAbstractSpinBox::NoButtons); optionWidgets["volLW"] = ui.volumeValue; optionWidgets["spdifChB"] = ui.spdifBox; optionWidgets["defaultVolume"] = ui.defaultVolume; optionWidgets["resetVolumeCheckbox"] = ui.resetVolumeCheckbox; updateAudioOptions( ui.outputModule->currentIndex() ); /* LastFM */ if( module_exists( "audioscrobbler" ) ) { CONFIG_GENERIC( "lastfm-username", String, ui.lastfm_user_label, lastfm_user_edit ); CONFIG_GENERIC( "lastfm-password", String, ui.lastfm_pass_label, lastfm_pass_edit ); if( config_ExistIntf( "audioscrobbler" ) ) ui.lastfm->setChecked( true ); else ui.lastfm->setChecked( false ); ui.lastfm_zone->setVisible( ui.lastfm->isChecked() ); connect( ui.lastfm, &QCheckBox::toggled, ui.lastfm_zone, &QWidget::setVisible ); connect( ui.lastfm, &QCheckBox::stateChanged, this, &SPrefsPanel::lastfm_Changed ); } else { ui.lastfm->hide(); ui.lastfm_zone->hide(); } /* Normalizer */ connect( ui.volNormBox, &QCheckBox::toggled, ui.volNormSpin, &QDoubleSpinBox::setEnabled ); char* psz = config_GetPsz( "audio-filter" ); qs_filter = qfu( psz ).split( ':', #if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0) Qt::SkipEmptyParts #else QString::SkipEmptyParts #endif ); free( psz ); bool b_enabled = ( qs_filter.contains( "normvol" ) ); ui.volNormBox->setChecked( b_enabled ); ui.volNormSpin->setEnabled( b_enabled && ui.volNormBox->isEnabled() ); /* Volume Label */ updateAudioVolume( ui.defaultVolume->value() ); // First time init END_SPREFS_CAT; /***************************************** * INPUT AND CODECS Panel Implementation * *****************************************/ START_SPREFS_CAT( InputAndCodecs, qtr("Input & Codecs Settings") ); /* Disk Devices */ { ui.DVDDeviceComboBox->setToolTip( qtr( "If this property is blank, different values\n" "for DVD, VCD, and CDDA are set.\n" "You can define a unique one or configure them \n" "individually in the advanced preferences." ) ); bool have_cdda = module_exists( "cdda" ); char *dvd_discpath = config_GetPsz( "dvd" ); char *vcd_discpath = config_GetPsz( "vcd" ); char *cdda_discpath = have_cdda ? config_GetPsz( "cd-audio" ) : nullptr; if( dvd_discpath && vcd_discpath && ( !have_cdda || cdda_discpath ) ) { if( !strcmp( dvd_discpath, vcd_discpath ) && ( !have_cdda || !strcmp( cdda_discpath, dvd_discpath ) ) ) { ui.DVDDeviceComboBox->setEditText( qfu( dvd_discpath ) ); } } free( cdda_discpath ); free( dvd_discpath ); free( vcd_discpath ); } #ifndef _WIN32 QStringList DVDDeviceComboBoxStringList = QStringList(); DVDDeviceComboBoxStringList << "dvd*" << "scd*" << "sr*" << "sg*" << "cd*"; ui.DVDDeviceComboBox->addItems( QDir( "/dev/" ) .entryList( DVDDeviceComboBoxStringList, QDir::System ) .replaceInStrings( QRegExp("^"), "/dev/" ) ); #endif CONFIG_GENERIC( "dvd", String, ui.DVDLabel, DVDDeviceComboBox->lineEdit() ); CONFIG_GENERIC_FILE( "input-record-path", Directory, ui.recordLabel, ui.recordPath, ui.recordBrowse ); CONFIG_GENERIC( "http-proxy", String , ui.httpProxyLabel, proxy ); CONFIG_GENERIC( "postproc-q", Integer, ui.ppLabel, PostProcLevel ); CONFIG_GENERIC( "avi-index", IntegerList, ui.aviLabel, AviRepair ); /* live555 module prefs */ CONFIG_BOOL( "rtsp-tcp", live555TransportRTSP_TCPRadio ); if ( !module_exists( "live555" ) ) { ui.live555TransportRTSP_TCPRadio->hide(); ui.live555TransportHTTPRadio->hide(); ui.live555TransportLabel->hide(); } CONFIG_GENERIC( "dec-dev", StringList, ui.hwAccelLabel, hwAccelModule ); CONFIG_BOOL( "input-fast-seek", fastSeekBox ); optionWidgets["inputLE"] = ui.DVDDeviceComboBox; optionWidgets["cachingCoB"] = ui.cachingCombo; CONFIG_GENERIC( "avcodec-skiploopfilter", IntegerList, ui.filterLabel, loopFilterBox ); CONFIG_GENERIC( "sout-x264-tune", StringList, ui.x264Label, tuneBox ); CONFIG_GENERIC( "sout-x264-preset", StringList, ui.x264Label, presetBox ); CONFIG_GENERIC( "sout-x264-profile", StringList, ui.x264profileLabel, profileBox ); CONFIG_GENERIC( "sout-x264-level", String, ui.x264profileLabel, levelBox ); CONFIG_BOOL( "mkv-preload-local-dir", mkvPreloadBox ); /* Caching */ /* Add the things to the ComboBox */ #define addToCachingBox( str, cachingNumber ) \ ui.cachingCombo->addItem( qfut(str), QVariant( cachingNumber ) ); addToCachingBox( N_("Custom"), CachingCustom ); addToCachingBox( N_("Lowest latency"), CachingLowest ); addToCachingBox( N_("Low latency"), CachingLow ); addToCachingBox( N_("Normal"), CachingNormal ); addToCachingBox( N_("High latency"), CachingHigh ); addToCachingBox( N_("Higher latency"), CachingHigher ); #undef addToCachingBox #define TestCaC( name, factor ) \ b_cache_equal = b_cache_equal && \ ( i_cache * factor == config_GetInt( name ) ); /* Select the accurate value of the ComboBox */ bool b_cache_equal = true; int i_cache = config_GetInt( "file-caching" ); TestCaC( "network-caching", 10/3 ); TestCaC( "disc-caching", 1); TestCaC( "live-caching", 1 ); if( b_cache_equal == 1 ) ui.cachingCombo->setCurrentIndex( ui.cachingCombo->findData( QVariant( i_cache ) ) ); #undef TestCaC END_SPREFS_CAT; /********************************** * INTERFACE Panel Implementation * **********************************/ START_SPREFS_CAT( Interface, qtr("Interface Settings") ); #ifndef _WIN32 ui.langBox->hide(); #else for( size_t i = 0; i < ARRAY_SIZE( language_map ); i++) ui.langCombo->addItem( qfu( language_map[i].name ), language_map[i].iso ); connect( ui.langCombo, QOverload::of(&QComboBox::currentIndexChanged), this, &SPrefsPanel::langChanged ); HKEY h_key; char *langReg = NULL; if( RegOpenKeyEx( HKEY_CURRENT_USER, TEXT("Software\\VideoLAN\\VLC\\"), 0, KEY_READ, &h_key ) == ERROR_SUCCESS ) { WCHAR szData[256]; DWORD len = 256; if( RegQueryValueEx( h_key, TEXT("Lang"), NULL, NULL, (LPBYTE) &szData, &len ) == ERROR_SUCCESS ) { langReg = FromWide( szData ); ui.langCombo->setCurrentIndex( ui.langCombo->findData(langReg) ); } } free( langReg); #endif // ui.defaultLabel->setFont( italicFont ); ui.skinsLabel->setText( qtr( "This is VLC's skinnable interface. You can download other skins at" ) + QString( " " ) + qtr( "VLC skins website" ) + QString( "." ) ); ui.skinsLabel->setFont( italicFont ); #ifdef _WIN32 BUTTONACT( ui.assoButton, &SPrefsPanel::assoDialog ); #else ui.osGroupBox->hide(); #endif /* interface */ char *psz_intf = config_GetPsz( "intf" ); if( psz_intf ) { if( strstr( psz_intf, "skin" ) ) ui.skins->setChecked( true ); } else { /* defaults to qt */ ui.qt->setChecked( true ); } free( psz_intf ); optionWidgets["skinRB"] = ui.skins; optionWidgets["qtRB"] = ui.qt; #if !defined( _WIN32) fillStylesCombo( ui.stylesCombo, getSettings()->value( "MainWindow/QtStyle", "" ).toString() ); m_resetters.push_back( std::make_unique( ui.stylesCombo, "currentIndex" ) ); connect( ui.stylesCombo, QOverload::of(&QComboBox::currentIndexChanged), this, &SPrefsPanel::changeStyle ); optionWidgets["styleCB"] = ui.stylesCombo; #else ui.stylesCombo->hide(); ui.stylesLabel->hide(); #endif radioGroup = new QButtonGroup(this); radioGroup->addButton( ui.qt, 0 ); radioGroup->addButton( ui.skins, 1 ); #if QT_VERSION >= QT_VERSION_CHECK(5,15,0) connect( radioGroup, &QButtonGroup::idClicked, ui.styleStackedWidget, &QStackedWidget::setCurrentIndex ); #else connect( radioGroup, QOverload::of(&QButtonGroup::buttonClicked), ui.styleStackedWidget, &QStackedWidget::setCurrentIndex ); #endif ui.styleStackedWidget->setCurrentIndex( radioGroup->checkedId() ); connect( ui.minimalviewBox, &QCheckBox::toggled, ui.mainPreview, &InterfacePreviewWidget::setNormalPreview ); CONFIG_BOOL( "qt-minimal-view", minimalviewBox ); ui.mainPreview->setNormalPreview( ui.minimalviewBox->isChecked() ); ui.skinsPreview->setPreview( InterfacePreviewWidget::SKINS ); CONFIG_BOOL( "embedded-video", embedVideo ); CONFIG_BOOL( "qt-video-autoresize", resizingBox ); connect( ui.embedVideo, &QCheckBox::toggled, ui.resizingBox, &QCheckBox::setEnabled ); ui.resizingBox->setEnabled( ui.embedVideo->isChecked() ); CONFIG_BOOL( "qt-fs-controller", fsController ); CONFIG_BOOL( "qt-system-tray", systrayBox ); CONFIG_GENERIC( "qt-notification", IntegerList, ui.notificationComboLabel, notificationCombo ); connect( ui.systrayBox, &QCheckBox::toggled, [=]( bool checked ) { ui.notificationCombo->setEnabled( checked ); ui.notificationComboLabel->setEnabled( checked ); } ); ui.notificationCombo->setEnabled( ui.systrayBox->isChecked() ); CONFIG_BOOL( "qt-pause-minimized", pauseMinimizedBox ); CONFIG_BOOL( "playlist-tree", treePlaylist ); CONFIG_BOOL( "play-and-pause", playPauseBox ); CONFIG_GENERIC_FILE( "skins2-last", File, ui.skinFileLabel, ui.fileSkin, ui.skinBrowse ); CONFIG_BOOL( "metadata-network-access", MetadataNetworkAccessMode ); CONFIG_BOOL( "qt-menubar", menuBarCheck ); ui.pinVideoControlsCheckbox->setChecked( p_intf->p_mi->pinVideoControls() ); m_resetters.push_back(std::make_unique(ui.pinVideoControlsCheckbox, "checked")); QObject::connect( ui.pinVideoControlsCheckbox, &QCheckBox::stateChanged, p_intf->p_mi, &MainCtx::setPinVideoControls ); ui.colorSchemeComboBox->setModel( p_intf->p_mi->getColorScheme() ); ui.colorSchemeComboBox->setCurrentText( p_intf->p_mi->getColorScheme()->currentText() ); m_resetters.push_back(std::make_unique( ui.colorSchemeComboBox, "currentIndex" )); QObject::connect( ui.colorSchemeComboBox, QOverload::of(&QComboBox::currentIndexChanged) , p_intf->p_mi->getColorScheme(), &ColorSchemeModel::setCurrentIndex ); const float intfScaleFloatFactor = 100.f; const auto updateIntfUserScaleFactorFromControls = [this, slider = ui.intfScaleFactorSlider, spinBox = ui.intfScaleFactorSpinBox, intfScaleFloatFactor](const int value) { if (slider->value() != value) { QSignalBlocker s( slider ); slider->setValue( value ); } if (spinBox->value() != value) { QSignalBlocker s( spinBox ); spinBox->setValue( value ); } p_intf->p_mi->setIntfUserScaleFactor( value / intfScaleFloatFactor ); }; ui.intfScaleFactorSlider->setRange( p_intf->p_mi->getMinIntfUserScaleFactor() * intfScaleFloatFactor , p_intf->p_mi->getMaxIntfUserScaleFactor() * intfScaleFloatFactor); ui.intfScaleFactorSpinBox->setRange( p_intf->p_mi->getMinIntfUserScaleFactor() * intfScaleFloatFactor , p_intf->p_mi->getMaxIntfUserScaleFactor() * intfScaleFloatFactor); updateIntfUserScaleFactorFromControls( p_intf->p_mi->getIntfUserScaleFactor() * intfScaleFloatFactor ); m_resetters.push_back( std::make_unique( ui.intfScaleFactorSlider, "value" ) ); QObject::connect( ui.intfScaleFactorSlider, QOverload::of(&QSlider::valueChanged) , p_intf->p_mi , updateIntfUserScaleFactorFromControls ); QObject::connect( ui.intfScaleFactorSpinBox, QOverload::of(&QSpinBox::valueChanged) , p_intf->p_mi , updateIntfUserScaleFactorFromControls ); #if QT_CLIENT_SIDE_DECORATION_AVAILABLE CONFIG_BOOL( "qt-titlebar", titleBarCheckBox ); #else ui.titleBarCheckBox->hide(); #endif /* UPDATE options */ #ifdef UPDATE_CHECK CONFIG_BOOL( "qt-updates-notif", updatesBox ); CONFIG_GENERIC( "qt-updates-days", Integer, nullptr, updatesDays ); ui.updatesDays->setEnabled( ui.updatesBox->isChecked() ); connect( ui.updatesBox, &QCheckBox::toggled, ui.updatesDays, &QSpinBox::setEnabled ); #else ui.updatesBox->hide(); ui.updatesDays->hide(); #endif /* ONE INSTANCE options */ #if !defined( _WIN32 ) && !defined(__APPLE__) && !defined(__OS2__) if( !module_exists( "dbus" ) ) { ui.OneInterfaceMode->hide(); ui.EnqueueOneInterfaceMode->hide(); ui.oneInstanceFromFile->hide(); } else #endif { CONFIG_BOOL( "one-instance", OneInterfaceMode ); CONFIG_BOOL( "playlist-enqueue", EnqueueOneInterfaceMode ); ui.EnqueueOneInterfaceMode->setEnabled( ui.OneInterfaceMode->isChecked() ); connect( ui.OneInterfaceMode, &QCheckBox::toggled, ui.EnqueueOneInterfaceMode, &QCheckBox::setEnabled ); CONFIG_BOOL( "one-instance-when-started-from-file", oneInstanceFromFile ); } CONFIG_GENERIC( "qt-auto-raise", IntegerList, ui.autoRaiseLabel, autoRaiseComboBox ); /* RECENTLY PLAYED options */ const auto hasMedialibrary = p_intf->p_mi->hasMediaLibrary(); ui.continuePlaybackLabel->setVisible( hasMedialibrary ); ui.continuePlaybackComboBox->setVisible( hasMedialibrary ); ui.saveRecentlyPlayed->setVisible( hasMedialibrary ); ui.clearRecent->setVisible( hasMedialibrary ); ui.clearRecentSpacer->changeSize( 0, 0 ); if ( hasMedialibrary ) { CONFIG_GENERIC( "restore-playback-pos", IntegerList, ui.continuePlaybackLabel, continuePlaybackComboBox ); CONFIG_BOOL( "save-recentplay", saveRecentlyPlayed ); ui.clearRecentSpacer->changeSize( 1, 1, QSizePolicy::Expanding, QSizePolicy::Minimum ); MLRecentsModel *recentsModel = new MLRecentsModel( ui.clearRecent ); recentsModel->setMl( p_intf->p_mi->getMediaLibrary() ); connect( ui.clearRecent, &QPushButton::clicked, recentsModel, &MLRecentsModel::clearHistory ); } END_SPREFS_CAT; /********************************** * SUBTITLES Panel Implementation * **********************************/ START_SPREFS_CAT( Subtitles, qtr("Subtitle & On Screen Display Settings") ); CONFIG_BOOL( "osd", OSDBox); CONFIG_BOOL( "video-title-show", OSDTitleBox); CONFIG_GENERIC( "video-title-position", IntegerList, ui.OSDTitlePosLabel, OSDTitlePos ); CONFIG_BOOL( "spu", spuActiveBox); ui.spuZone->setEnabled( ui.spuActiveBox->isChecked() ); connect( ui.spuActiveBox, &QCheckBox::toggled, ui.spuZone, &QWidget::setEnabled ); CONFIG_GENERIC( "subsdec-encoding", StringList, ui.encodLabel, encoding ); CONFIG_GENERIC( "sub-language", String, ui.subLangLabel, preferredLanguage ); CONFIG_GENERIC( "freetype-rel-fontsize", IntegerList, ui.fontSizeLabel, fontSize ); CONFIG_GENERIC( "freetype-font", Font, ui.fontLabel, font ); CONFIG_GENERIC( "freetype-color", Color, ui.fontColorLabel, fontColor ); CONFIG_GENERIC( "freetype-outline-thickness", IntegerList, ui.fontEffectLabel, effect ); CONFIG_GENERIC( "freetype-outline-color", Color, ui.outlineColorLabel, outlineColor ); CONFIG_GENERIC( "sub-margin", Integer, ui.subsPosLabel, subsPosition ); if( module_exists( "freetype" ) ) { ui.shadowCheck->setChecked( config_GetInt( "freetype-shadow-opacity" ) > 0 ); ui.backgroundCheck->setChecked( config_GetInt( "freetype-background-opacity" ) > 0 ); } else { ui.shadowCheck->setEnabled( false ); ui.backgroundCheck->setEnabled( false ); } optionWidgets["shadowCB"] = ui.shadowCheck; optionWidgets["backgroundCB"] = ui.backgroundCheck; CONFIG_GENERIC( "secondary-sub-alignment", IntegerList, ui.secondarySubsAlignmentLabel, secondarySubsAlignment ); CONFIG_GENERIC( "secondary-sub-margin", Integer, ui.secondarySubsPosLabel, secondarySubsPosition ); END_SPREFS_CAT; /******************************** * HOTKEYS Panel Implementation * ********************************/ case SPrefsHotkeys: { QGridLayout *gLayout = new QGridLayout; panel->setLayout( gLayout ); int line = 0; panel_label->setText( qtr( "Configure Hotkeys" ) ); control = new KeySelectorControl( this ); control->insertInto( gLayout, line ); controls.append( control ); line++; p_config = config_FindConfig( "hotkeys-y-wheel-mode" ); control = new IntegerListConfigControl( p_config, this ); control->insertInto( gLayout, line ); controls.append( control ); line++; p_config = config_FindConfig( "hotkeys-x-wheel-mode" ); control = new IntegerListConfigControl( p_config, this ); control->insertInto( gLayout, line ); controls.append( control ); #ifdef _WIN32 line++; p_config = config_FindConfig( "qt-disable-volume-keys" ); control = new BoolConfigControl( p_config, this ); control->insertInto( gLayout, line ); controls.append( control ); #endif break; } /************************************** * MEDIA LIBRARY Panel Implementation * **************************************/ START_SPREFS_CAT( MediaLibrary , qtr("Media Library Settings") ); if ( vlc_ml_instance_get( p_intf ) != NULL ) { auto foldersModel = new MLFoldersModel( this ); foldersModel->setCtx( p_intf->p_mi ); ui.entryPoints->setMLFoldersModel( foldersModel ); mlFoldersEditor = ui.entryPoints; auto bannedFoldersModel = new MLBannedFoldersModel( this ); bannedFoldersModel->setCtx( p_intf->p_mi ); ui.bannedEntryPoints->setMLFoldersModel( bannedFoldersModel ); mlBannedFoldersEditor = ui.bannedEntryPoints; BUTTONACT( ui.addButton, &SPrefsPanel::MLaddNewFolder ); BUTTONACT( ui.banButton, &SPrefsPanel::MLBanFolder ); } else { ui.mlGroupBox->hide( ); } END_SPREFS_CAT; } panel_layout->addWidget( panel_label ); panel_layout->addWidget( title_line ); QScrollArea *scroller= new QScrollArea; scroller->setWidget( panel ); scroller->setWidgetResizable( true ); scroller->setFrameStyle( QFrame::NoFrame ); panel_layout->addWidget( scroller ); setLayout( panel_layout ); #undef END_SPREFS_CAT #undef START_SPREFS_CAT #undef CONFIG_GENERIC_FILE #undef CONFIG_GENERIC_NO_UI #undef CONFIG_GENERIC #undef CONFIG_BOOL } void SPrefsPanel::updateAudioOptions( int number ) { QString value = qobject_cast(optionWidgets["audioOutCoB"]) ->itemData( number ).toString(); #ifdef _WIN32 /* Since MMDevice is most likely to be used by default, we show MMDevice * options by default */ const bool mmDeviceEnabled = value == "mmdevice" || value == "any"; optionWidgets["mmdevicePassthroughL"]->setVisible( mmDeviceEnabled ); optionWidgets["mmdevicePassthroughB"]->setVisible( mmDeviceEnabled ); optionWidgets["mmdeviceW"]->setVisible( mmDeviceEnabled ); optionWidgets["mmdeviceL"]->setVisible( mmDeviceEnabled ); optionWidgets["directxW"]->setVisible( ( value == "directsound" ) ); optionWidgets["directxL"]->setVisible( ( value == "directsound" ) ); optionWidgets["waveoutW"]->setVisible( ( value == "waveout" ) ); optionWidgets["waveoutL"]->setVisible( ( value == "waveout" ) ); #elif defined( __OS2__ ) optionWidgets["kaiL"]->setVisible( ( value == "kai" ) ); optionWidgets["kaiW"]->setVisible( ( value == "kai" ) ); #else /* optionWidgets["ossW] can be NULL */ if( optionWidgets["ossW"] ) { optionWidgets["ossW"]->setVisible( ( value == "oss" ) ); optionWidgets["ossL"]->setVisible( ( value == "oss" ) ); optionWidgets["ossB"]->setVisible( ( value == "oss" ) ); } /* optionWidgets["alsaW] can be NULL */ if( optionWidgets["alsaW"] ) { optionWidgets["alsaW"]->setVisible( ( value == "alsa" ) ); optionWidgets["alsaL"]->setVisible( ( value == "alsa" ) ); } #endif optionWidgets["fileW"]->setVisible( ( value == "afile" ) ); optionWidgets["spdifChB"]->setVisible( ( value == "alsa" || value == "oss" || value == "auhal" || value == "directsound" || value == "waveout" ) ); int volume = getDefaultAudioVolume(qtu(value)); bool save = true; if (volume >= 0) save = config_GetInt("volume-save"); QCheckBox *resetVolumeCheckBox = qobject_cast(optionWidgets["resetVolumeCheckbox"]); resetVolumeCheckBox->setChecked(!save); resetVolumeCheckBox->setEnabled(volume >= 0); QSlider *defaultVolume = qobject_cast(optionWidgets["defaultVolume"]); defaultVolume->setValue((volume >= 0) ? volume : 100); defaultVolume->setEnabled(volume >= 0); } SPrefsPanel::~SPrefsPanel() { if (!m_isApplied) clean(); qDeleteAll( controls ); controls.clear(); free( lang ); } void SPrefsPanel::updateAudioVolume( int volume ) { qobject_cast(optionWidgets["volLW"]) ->setValue( volume ); } /* Function called from the main Preferences dialog on each SPrefs Panel */ void SPrefsPanel::apply() { m_isApplied = true; /* Generic save for ever panel */ QList::const_iterator i; for( i = controls.begin() ; i != controls.end() ; ++i ) { ConfigControl *c = qobject_cast(*i); c->doApply(); } switch( number ) { case SPrefsInputAndCodecs: { /* Device default selection */ QByteArray devicepath = qobject_cast(optionWidgets["inputLE"])->currentText().toUtf8(); if( devicepath.size() > 0 ) { config_PutPsz( "dvd", devicepath ); config_PutPsz( "vcd", devicepath ); if( module_exists( "cdda" ) ) config_PutPsz( "cd-audio", devicepath ); } #define CaC( name, factor ) config_PutInt( name, i_comboValue * factor ) /* Caching */ QComboBox *cachingCombo = qobject_cast(optionWidgets["cachingCoB"]); int i_comboValue = cachingCombo->itemData( cachingCombo->currentIndex() ).toInt(); if( i_comboValue ) { CaC( "file-caching", 1 ); CaC( "network-caching", 10/3 ); CaC( "disc-caching", 1 ); CaC( "live-caching", 1 ); } break; #undef CaC } /* Interfaces */ case SPrefsInterface: { if( qobject_cast(optionWidgets["skinRB"])->isChecked() ) config_PutPsz( "intf", "skins2,any" ); else //if( qobject_cast(optionWidgets[qtRB])->isChecked() ) config_PutPsz( "intf", "" ); if( auto stylesCombo = qobject_cast(optionWidgets["styleCB"]) ) getSettings()->setValue( "MainWindow/QtStyle", getQStyleKey( stylesCombo , "" ) ); #ifdef _WIN32 saveLang(); #endif break; } case SPrefsVideo: { int i_fullscreenScreen = qobject_cast(optionWidgets["fullscreenScreenB"])->currentData().toInt(); config_PutInt( "qt-fullscreen-screennumber", i_fullscreenScreen ); break; } case SPrefsAudio: { bool b_checked = qobject_cast(optionWidgets["normalizerChB"])->isChecked(); if( b_checked && !qs_filter.contains( "normvol" ) ) qs_filter.append( "normvol" ); if( !b_checked && qs_filter.contains( "normvol" ) ) qs_filter.removeAll( "normvol" ); config_PutPsz( "audio-filter", qtu( qs_filter.join( ":" ) ) ); /* Default volume */ int i_volume = qobject_cast(optionWidgets["defaultVolume"])->value(); bool b_reset_volume = qobject_cast(optionWidgets["resetVolumeCheckbox"])->isChecked(); char *psz_aout = config_GetPsz( "aout" ); float f_gain = powf( i_volume / 100.f, 3 ); #define save_vol_aout( name ) \ module_exists( name ) && ( !psz_aout || !strcmp( psz_aout, name ) || !strcmp( psz_aout, "any" ) ) //FIXME this is moot #if defined( _WIN32 ) VLC_UNUSED( f_gain ); if( save_vol_aout( "mmdevice" ) ) config_PutFloat( "mmdevice-volume", i_volume / 100.f ); if( save_vol_aout( "directsound" ) ) config_PutFloat( "directx-volume", i_volume / 100.f ); if( save_vol_aout( "waveout" ) ) config_PutFloat( "waveout-volume", i_volume / 100.f ); #elif defined( Q_OS_MAC ) VLC_UNUSED( f_gain ); if( save_vol_aout( "auhal" ) ) config_PutFloat( "auhal-volume", i_volume / 100.f * AOUT_VOLUME_DEFAULT ); #elif defined( __OS2__ ) if( save_vol_aout( "kai" ) ) config_PutFloat( "kai-gain", f_gain ); #else if( save_vol_aout( "alsa" ) ) config_PutFloat( "alsa-gain", f_gain ); if( save_vol_aout( "jack" ) ) config_PutFloat( "jack-gain", f_gain ); #endif #undef save_vol_aout free( psz_aout ); config_PutInt( "volume-save", !b_reset_volume ); break; } case SPrefsSubtitles: { bool b_checked = qobject_cast(optionWidgets["shadowCB"])->isChecked(); if( b_checked && config_GetInt( "freetype-shadow-opacity" ) == 0 ) { config_PutInt( "freetype-shadow-opacity", 128 ); } else if (!b_checked ) { config_PutInt( "freetype-shadow-opacity", 0 ); } b_checked = qobject_cast(optionWidgets["backgroundCB"])->isChecked(); if( b_checked && config_GetInt( "freetype-background-opacity" ) == 0 ) { config_PutInt( "freetype-background-opacity", 128 ); } else if (!b_checked ) { config_PutInt( "freetype-background-opacity", 0 ); } break; } case SPrefsMediaLibrary: { mlFoldersEditor->commit(); mlBannedFoldersEditor->commit(); } } } void SPrefsPanel::clean() { for ( auto &resetter : m_resetters ) resetter->reset(); } void SPrefsPanel::lastfm_Changed( int i_state ) { if( i_state == Qt::Checked ) config_AddIntf( "audioscrobbler" ); else if( i_state == Qt::Unchecked ) config_RemoveIntf( "audioscrobbler" ); } void SPrefsPanel::changeStyle() { QApplication::setStyle( getQStyleKey( qobject_cast( optionWidgets["styleCB"] ) , p_intf->p_app->defaultStyle() ) ); /* force refresh on all widgets */ QWidgetList widgets = QApplication::allWidgets(); QWidgetList::iterator it = widgets.begin(); while( it != widgets.end() ) { (*it)->update(); ++it; }; } void SPrefsPanel::langChanged( int i ) { free( lang ); lang = strdup( language_map[i].iso ); } void SPrefsPanel::configML() { #ifdef SQL_MEDIA_LIBRARY MLConfDialog *mld = new MLConfDialog( this, p_intf ); mld->exec(); delete mld; #endif } #ifdef _WIN32 #include #include "util/registry.hpp" void SPrefsPanel::cleanLang() { QVLCRegistry qvReg( HKEY_CURRENT_USER ); qvReg.DeleteValue( "Software\\VideoLAN\\VLC\\", "Lang" ); qvReg.DeleteKey( "Software\\VideoLAN\\", "VLC" ); qvReg.DeleteKey( "Software\\", "VideoLAN" ); } void SPrefsPanel::saveLang() { if( !lang ) return; if( !strncmp( lang, "auto", 4 ) ) { cleanLang(); } else { QVLCRegistry qvReg( HKEY_CURRENT_USER ); qvReg.WriteRegistry( "Software\\VideoLAN\\VLC\\", "Lang", lang ); } } bool SPrefsPanel::addType( const char * psz_ext, QTreeWidgetItem* current, QTreeWidgetItem* parent, QVLCRegistry *qvReg ) { bool b_temp; const char* psz_VLC = "VLC"; current = new QTreeWidgetItem( parent, QStringList( psz_ext ) ); char* psz_reg = qvReg->ReadRegistry( psz_ext, "", "" ); if( psz_reg == NULL ) return false; if( strstr( psz_reg, psz_VLC ) ) { current->setCheckState( 0, Qt::Checked ); b_temp = true; } else { current->setCheckState( 0, Qt::Unchecked ); b_temp = false; } free( psz_reg ); listAsso.append( current ); return b_temp; } #if !defined(__IApplicationAssociationRegistrationUI_INTERFACE_DEFINED__) #define __IApplicationAssociationRegistrationUI_INTERFACE_DEFINED__ const GUID IID_IApplicationAssociationRegistrationUI = {0x1f76a169,0xf994,0x40ac, {0x8f,0xc8,0x09,0x59,0xe8,0x87,0x47,0x10}}; extern const GUID CLSID_ApplicationAssociationRegistrationUI; interface IApplicationAssociationRegistrationUI : public IUnknown { virtual HRESULT STDMETHODCALLTYPE LaunchAdvancedAssociationUI( LPCWSTR pszAppRegName) = 0; }; #endif /* __IApplicationAssociationRegistrationUI_INTERFACE_DEFINED__ */ void SPrefsPanel::assoDialog() { HRESULT hr; hr = CoInitializeEx( NULL, COINIT_APARTMENTTHREADED ); if( SUCCEEDED(hr) ) { void *p; hr = CoCreateInstance(CLSID_ApplicationAssociationRegistrationUI, NULL, CLSCTX_INPROC_SERVER, IID_IApplicationAssociationRegistrationUI, &p); if( SUCCEEDED(hr) ) { IApplicationAssociationRegistrationUI *p_regui = (IApplicationAssociationRegistrationUI *)p; hr = p_regui->LaunchAdvancedAssociationUI(L"VLC" ); p_regui->Release(); } CoUninitialize(); } if( SUCCEEDED(hr) ) return; QDialog *d = new QDialog( this ); d->setWindowTitle( qtr( "File associations" ) ); QGridLayout *assoLayout = new QGridLayout( d ); QTreeWidget *filetypeList = new QTreeWidget; assoLayout->addWidget( filetypeList, 0, 0, 1, 4 ); filetypeList->header()->hide(); QVLCRegistry qvReg( HKEY_CLASSES_ROOT ); QTreeWidgetItem *audioType = new QTreeWidgetItem( QStringList( qtr( "Audio Files" ) ) ); QTreeWidgetItem *videoType = new QTreeWidgetItem( QStringList( qtr( "Video Files" ) ) ); QTreeWidgetItem *otherType = new QTreeWidgetItem( QStringList( qtr( "Playlist Files" ) ) ); filetypeList->addTopLevelItem( audioType ); filetypeList->addTopLevelItem( videoType ); filetypeList->addTopLevelItem( otherType ); audioType->setExpanded( true ); audioType->setCheckState( 0, Qt::Unchecked ); videoType->setExpanded( true ); videoType->setCheckState( 0, Qt::Unchecked ); otherType->setExpanded( true ); otherType->setCheckState( 0, Qt::Unchecked ); QTreeWidgetItem *currentItem = NULL; int i_temp = 0; #define aTa( name ) i_temp += addType( name, currentItem, audioType, &qvReg ) #define aTv( name ) i_temp += addType( name, currentItem, videoType, &qvReg ) #define aTo( name ) i_temp += addType( name, currentItem, otherType, &qvReg ) aTa( ".3ga" ); aTa( ".669" ); aTa( ".a52" ); aTa( ".aac" ); aTa( ".ac3" ); aTa( ".adt" ); aTa( ".adts" ); aTa( ".aif" ); aTa( ".aifc" ); aTa( ".aiff" ); aTa( ".au" ); aTa( ".amr" ); aTa( ".aob" ); aTa( ".ape" ); aTa( ".caf" ); aTa( ".cda" ); aTa( ".dts" ); aTa( ".flac" ); aTa( ".it" ); aTa( ".m4a" ); aTa( ".m4p" ); aTa( ".mid" ); aTa( ".mka" ); aTa( ".mlp" ); aTa( ".mod" ); aTa( ".mp1" ); aTa( ".mp2" ); aTa( ".mp3" ); aTa( ".mpc" ); aTa( ".mpga" ); aTa( ".oga" ); aTa( ".oma" ); aTa( ".opus" ); aTa( ".qcp" ); aTa( ".ra" ); aTa( ".rmi" ); aTa( ".snd" ); aTa( ".s3m" ); aTa( ".spx" ); aTa( ".tta" ); aTa( ".voc" ); aTa( ".vqf" ); aTa( ".w64" ); aTa( ".wav" ); aTa( ".wma" ); aTa( ".wv" ); aTa( ".xa" ); aTa( ".xm" ); audioType->setCheckState( 0, ( i_temp > 0 ) ? ( ( i_temp == audioType->childCount() ) ? Qt::Checked : Qt::PartiallyChecked ) : Qt::Unchecked ); i_temp = 0; aTv( ".3g2" ); aTv( ".3gp" ); aTv( ".3gp2" ); aTv( ".3gpp" ); aTv( ".amv" ); aTv( ".asf" ); aTv( ".avi" ); aTv( ".bik" ); aTv( ".divx" ); aTv( ".drc" ); aTv( ".dv" ); aTv( ".f4v" ); aTv( ".flv" ); aTv( ".gvi" ); aTv( ".gxf" ); aTv( ".m1v" ); aTv( ".m2t" ); aTv( ".m2v" ); aTv( ".m2ts" ); aTv( ".m4v" ); aTv( ".mkv" ); aTv( ".mov" ); aTv( ".mp2v" ); aTv( ".mp4" ); aTv( ".mp4v" ); aTv( ".mpa" ); aTv( ".mpe" ); aTv( ".mpeg" ); aTv( ".mpeg1" ); aTv( ".mpeg2" ); aTv( ".mpeg4" ); aTv( ".mpg" ); aTv( ".mpv2" ); aTv( ".mts" ); aTv( ".mtv" ); aTv( ".mxf" ); aTv( ".nsv" ); aTv( ".nuv" ); aTv( ".ogg" ); aTv( ".ogm" ); aTv( ".ogx" ); aTv( ".ogv" ); aTv( ".rec" ); aTv( ".rm" ); aTv( ".rmvb" ); aTv( ".rpl" ); aTv( ".thp" ); aTv( ".tod" ); aTv( ".ts" ); aTv( ".tts" ); aTv( ".vob" ); aTv( ".vro" ); aTv( ".webm" ); aTv( ".wmv" ); aTv( ".xesc" ); videoType->setCheckState( 0, ( i_temp > 0 ) ? ( ( i_temp == videoType->childCount() ) ? Qt::Checked : Qt::PartiallyChecked ) : Qt::Unchecked ); i_temp = 0; aTo( ".asx" ); aTo( ".b4s" ); aTo( ".cue" ); aTo( ".ifo" ); aTo( ".m3u" ); aTo( ".m3u8" ); aTo( ".pls" ); aTo( ".ram" ); aTo( ".sdp" ); aTo( ".vlc" ); aTo( ".wvx" ); aTo( ".xspf" ); otherType->setCheckState( 0, ( i_temp > 0 ) ? ( ( i_temp == otherType->childCount() ) ? Qt::Checked : Qt::PartiallyChecked ) : Qt::Unchecked ); #undef aTo #undef aTv #undef aTa connect( filetypeList, &QTreeWidget::itemChanged, this, &SPrefsPanel::updateCheckBoxes ); QDialogButtonBox *buttonBox = new QDialogButtonBox( d ); QPushButton *closeButton = new QPushButton( qtr( "&Apply" ) ); QPushButton *clearButton = new QPushButton( qtr( "&Cancel" ) ); buttonBox->addButton( closeButton, QDialogButtonBox::AcceptRole ); buttonBox->addButton( clearButton, QDialogButtonBox::ActionRole ); assoLayout->addWidget( buttonBox, 1, 2, 1, 2 ); connect( closeButton, &QPushButton::clicked, [=]() { this->saveAsso(); d->reject(); } ); d->resize( 300, 400 ); d->exec(); listAsso.clear(); } void SPrefsPanel::updateCheckBoxes(QTreeWidgetItem* item, int column) { if( column != 0 ) return; /* temporarily block signals to avoid signal loops */ bool b_signalsBlocked = item->treeWidget()->blockSignals(true); /* A parent checkbox was changed */ if( item->parent() == 0 ) { Qt::CheckState checkState = item->checkState(0); for( int i = 0; i < item->childCount(); i++ ) { item->child(i)->setCheckState(0, checkState); } } /* A child checkbox was changed */ else { bool b_diff = false; for( int i = 0; i < item->parent()->childCount(); i++ ) { if( i != item->parent()->indexOfChild(item) && item->checkState(0) != item->parent()->child(i)->checkState(0) ) { b_diff = true; break; } } if( b_diff ) item->parent()->setCheckState(0, Qt::PartiallyChecked); else item->parent()->setCheckState(0, item->checkState(0)); } /* Stop signal blocking */ item->treeWidget()->blockSignals(b_signalsBlocked); } void addAsso( QVLCRegistry *qvReg, const char *psz_ext ) { QString s_path( "VLC" ); s_path += psz_ext; QString s_path2 = s_path; /* Save a backup if already assigned */ char *psz_value = qvReg->ReadRegistry( psz_ext, "", "" ); if( !EMPTY_STR(psz_value) && strcmp( qtu(s_path), psz_value ) ) qvReg->WriteRegistry( psz_ext, "VLC.backup", psz_value ); free( psz_value ); /* Put a "link" to VLC.EXT as default */ qvReg->WriteRegistry( psz_ext, "", qtu( s_path ) ); /* Create the needed Key if they weren't done in the installer */ if( !qvReg->RegistryKeyExists( qtu( s_path ) ) ) { qvReg->WriteRegistry( psz_ext, "", qtu( s_path ) ); qvReg->WriteRegistry( qtu( s_path ), "", "Media file" ); qvReg->WriteRegistry( qtu( s_path.append( "\\shell" ) ), "", "Play" ); /* Get the installer path */ QVLCRegistry qvReg2( HKEY_LOCAL_MACHINE ); QString str_temp = qvReg2.ReadRegistry( "Software\\VideoLAN\\VLC", "", "" ); if( str_temp.size() ) { qvReg->WriteRegistry( qtu( s_path.append( "\\Play\\command" ) ), "", qtu( str_temp.append(" --started-from-file \"%1\"" ) ) ); qvReg->WriteRegistry( qtu( s_path2.append( "\\DefaultIcon" ) ), "", qtu( str_temp.append(",0") ) ); } } } void delAsso( QVLCRegistry *qvReg, const char *psz_ext ) { QString s_path( "VLC"); s_path += psz_ext; char *psz_value = qvReg->ReadRegistry( psz_ext, "", "" ); if( psz_value && !strcmp( qtu(s_path), psz_value ) ) { free( psz_value ); psz_value = qvReg->ReadRegistry( psz_ext, "VLC.backup", "" ); if( psz_value ) qvReg->WriteRegistry( psz_ext, "", psz_value ); qvReg->DeleteValue( psz_ext, "VLC.backup" ); } free( psz_value ); } void SPrefsPanel::saveAsso() { QVLCRegistry qvReg( HKEY_CLASSES_ROOT ); for( int i = 0; i < listAsso.size(); i ++ ) { if( listAsso[i]->checkState( 0 ) > 0 ) { addAsso( &qvReg, qtu( listAsso[i]->text( 0 ) ) ); } else { delAsso( &qvReg, qtu( listAsso[i]->text( 0 ) ) ); } } /* Gruik ? Naaah */ qobject_cast(listAsso[0]->treeWidget()->parent())->accept(); } #endif /* _WIN32 */ void SPrefsPanel::MLaddNewFolder() { QUrl newEntryPoint = QFileDialog::getExistingDirectoryUrl( this , qtr("Please choose an entry point folder") , QUrl( QDir::homePath( ) ) ); if(! newEntryPoint.isEmpty() ) mlFoldersEditor->add( newEntryPoint ); } void SPrefsPanel::MLBanFolder( ) { QUrl newEntryPoint = QFileDialog::getExistingDirectoryUrl( this , qtr("Please choose an entry point folder") , QUrl( QDir::homePath( ) ) ); if(! newEntryPoint.isEmpty() ) mlBannedFoldersEditor->add( newEntryPoint ); }