/****************************************************************************** Copyright (C) 2013-2014 by Hugh Bailey Philippe Groarke 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, see . ******************************************************************************/ #include #include #include #include #include #include #include #include #include #include #include #include #include "obs-app.hpp" #include "platform.hpp" #include "properties-view.hpp" #include "qt-wrappers.hpp" #include "window-basic-main.hpp" #include "window-basic-settings.hpp" #include using namespace std; // Used for QVariant in codec comboboxes namespace { struct FormatDesc { const char *name = nullptr; const char *mimeType = nullptr; const ff_format_desc *desc = nullptr; inline FormatDesc() = default; inline FormatDesc(const char *name, const char *mimeType, const ff_format_desc *desc = nullptr) : name(name), mimeType(mimeType), desc(desc) {} bool operator==(const FormatDesc &f) const { if ((name == nullptr) ^ (f.name == nullptr)) return false; if (name != nullptr && strcmp(name, f.name) != 0) return false; if ((mimeType == nullptr) ^ (f.mimeType == nullptr)) return false; if (mimeType != nullptr && strcmp(mimeType, f.mimeType) != 0) return false; return true; } }; struct CodecDesc { const char *name = nullptr; int id = 0; inline CodecDesc() = default; inline CodecDesc(const char *name, int id) : name(name), id(id) {} bool operator==(const CodecDesc &codecDesc) const { if ((name == nullptr) ^ (codecDesc.name == nullptr)) return false; if (id != codecDesc.id) return false; return name == nullptr || strcmp(name, codecDesc.name) == 0; } }; } Q_DECLARE_METATYPE(FormatDesc) Q_DECLARE_METATYPE(CodecDesc) /* parses "[width]x[height]", string, i.e. 1024x768 */ static bool ConvertResText(const char *res, uint32_t &cx, uint32_t &cy) { BaseLexer lex; base_token token; lexer_start(lex, res); /* parse width */ if (!lexer_getbasetoken(lex, &token, IGNORE_WHITESPACE)) return false; if (token.type != BASETOKEN_DIGIT) return false; cx = std::stoul(token.text.array); /* parse 'x' */ if (!lexer_getbasetoken(lex, &token, IGNORE_WHITESPACE)) return false; if (strref_cmpi(&token.text, "x") != 0) return false; /* parse height */ if (!lexer_getbasetoken(lex, &token, IGNORE_WHITESPACE)) return false; if (token.type != BASETOKEN_DIGIT) return false; cy = std::stoul(token.text.array); /* shouldn't be any more tokens after this */ if (lexer_getbasetoken(lex, &token, IGNORE_WHITESPACE)) return false; return true; } static inline bool WidgetChanged(QWidget *widget) { return widget->property("changed").toBool(); } static inline void SetComboByName(QComboBox *combo, const char *name) { int idx = combo->findText(QT_UTF8(name)); if (idx != -1) combo->setCurrentIndex(idx); } static inline void SetComboByValue(QComboBox *combo, const char *name) { int idx = combo->findData(QT_UTF8(name)); if (idx != -1) combo->setCurrentIndex(idx); } static inline QString GetComboData(QComboBox *combo) { int idx = combo->currentIndex(); if (idx == -1) return QString(); return combo->itemData(idx).toString(); } static int FindEncoder(QComboBox *combo, const char *name, int id) { CodecDesc codecDesc(name, id); for(int i = 0; i < combo->count(); i++) { QVariant v = combo->itemData(i); if (!v.isNull()) { if (codecDesc == v.value()) { return i; break; } } } return -1; } static CodecDesc GetDefaultCodecDesc(const ff_format_desc *formatDesc, ff_codec_type codecType) { int id = 0; switch (codecType) { case FF_CODEC_AUDIO: id = ff_format_desc_audio(formatDesc); break; case FF_CODEC_VIDEO: id = ff_format_desc_video(formatDesc); break; default: return CodecDesc(); } return CodecDesc(ff_format_desc_get_default_name(formatDesc, codecType), id); } void OBSBasicSettings::HookWidget(QWidget *widget, const char *signal, const char *slot) { QObject::connect(widget, signal, this, slot); widget->setProperty("changed", QVariant(false)); } #define COMBO_CHANGED SIGNAL(currentIndexChanged(int)) #define EDIT_CHANGED SIGNAL(textChanged(const QString &)) #define CBEDIT_CHANGED SIGNAL(editTextChanged(const QString &)) #define CHECK_CHANGED SIGNAL(clicked(bool)) #define SCROLL_CHANGED SIGNAL(valueChanged(int)) #define GENERAL_CHANGED SLOT(GeneralChanged()) #define STREAM1_CHANGED SLOT(Stream1Changed()) #define OUTPUTS_CHANGED SLOT(OutputsChanged()) #define AUDIO_RESTART SLOT(AudioChangedRestart()) #define AUDIO_CHANGED SLOT(AudioChanged()) #define VIDEO_RESTART SLOT(VideoChangedRestart()) #define VIDEO_RES SLOT(VideoChangedResolution()) #define VIDEO_CHANGED SLOT(VideoChanged()) #define ADV_CHANGED SLOT(AdvancedChanged()) #define ADV_RESTART SLOT(AdvancedChangedRestart()) OBSBasicSettings::OBSBasicSettings(QWidget *parent) : QDialog (parent), main (qobject_cast(parent)), ui (new Ui::OBSBasicSettings) { string path; ui->setupUi(this); HookWidget(ui->language, COMBO_CHANGED, GENERAL_CHANGED); HookWidget(ui->theme, COMBO_CHANGED, GENERAL_CHANGED); HookWidget(ui->outputMode, COMBO_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->streamType, COMBO_CHANGED, STREAM1_CHANGED); HookWidget(ui->simpleOutputPath, EDIT_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->simpleOutputVBitrate, SCROLL_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->simpleOutputABitrate, COMBO_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->simpleOutReconnect, CHECK_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->simpleOutRetryDelay, SCROLL_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->simpleOutMaxRetries, SCROLL_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->simpleOutAdvanced, CHECK_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->simpleOutUseCBR, CHECK_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->simpleOutPreset, COMBO_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->simpleOutUseBufsize, CHECK_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->simpleOutPreset, COMBO_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->simpleOutVBufsize, SCROLL_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->simpleOutCustom, EDIT_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutReconnect, CHECK_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutRetryDelay, SCROLL_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutMaxRetries, SCROLL_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutEncoder, COMBO_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutUseRescale, CHECK_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutRescale, CBEDIT_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutTrack1, CHECK_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutTrack2, CHECK_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutTrack3, CHECK_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutTrack4, CHECK_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutApplyService, CHECK_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutRecType, COMBO_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutRecPath, EDIT_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutRecEncoder, COMBO_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutRecUseRescale, CHECK_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutRecRescale, CBEDIT_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutRecTrack1, CHECK_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutRecTrack2, CHECK_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutRecTrack3, CHECK_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutRecTrack4, CHECK_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutFFURL, EDIT_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutFFFormat, COMBO_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutFFVBitrate, SCROLL_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutFFUseRescale, CHECK_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutFFRescale, CBEDIT_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutFFVEncoder, COMBO_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutFFVCfg, EDIT_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutFFABitrate, SCROLL_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutFFTrack1, CHECK_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutFFTrack2, CHECK_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutFFTrack3, CHECK_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutFFTrack4, CHECK_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutFFAEncoder, COMBO_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutFFACfg, EDIT_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutTrack1Bitrate, COMBO_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutTrack1Name, EDIT_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutTrack2Bitrate, COMBO_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutTrack2Name, EDIT_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutTrack3Bitrate, COMBO_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutTrack3Name, EDIT_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutTrack4Bitrate, COMBO_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->advOutTrack4Name, EDIT_CHANGED, OUTPUTS_CHANGED); HookWidget(ui->channelSetup, COMBO_CHANGED, AUDIO_RESTART); HookWidget(ui->sampleRate, COMBO_CHANGED, AUDIO_RESTART); HookWidget(ui->desktopAudioDevice1, COMBO_CHANGED, AUDIO_CHANGED); HookWidget(ui->desktopAudioDevice2, COMBO_CHANGED, AUDIO_CHANGED); HookWidget(ui->auxAudioDevice1, COMBO_CHANGED, AUDIO_CHANGED); HookWidget(ui->auxAudioDevice2, COMBO_CHANGED, AUDIO_CHANGED); HookWidget(ui->auxAudioDevice3, COMBO_CHANGED, AUDIO_CHANGED); HookWidget(ui->renderer, COMBO_CHANGED, VIDEO_RESTART); HookWidget(ui->adapter, COMBO_CHANGED, VIDEO_RESTART); HookWidget(ui->baseResolution, CBEDIT_CHANGED, VIDEO_RES); HookWidget(ui->outputResolution, CBEDIT_CHANGED, VIDEO_RES); HookWidget(ui->downscaleFilter, COMBO_CHANGED, VIDEO_CHANGED); HookWidget(ui->fpsType, COMBO_CHANGED, VIDEO_CHANGED); HookWidget(ui->fpsCommon, COMBO_CHANGED, VIDEO_CHANGED); HookWidget(ui->fpsInteger, SCROLL_CHANGED, VIDEO_CHANGED); HookWidget(ui->fpsInteger, SCROLL_CHANGED, VIDEO_CHANGED); HookWidget(ui->fpsNumerator, SCROLL_CHANGED, VIDEO_CHANGED); HookWidget(ui->fpsDenominator, SCROLL_CHANGED, VIDEO_CHANGED); HookWidget(ui->audioBufferingTime, SCROLL_CHANGED, ADV_RESTART); HookWidget(ui->colorFormat, COMBO_CHANGED, ADV_CHANGED); HookWidget(ui->colorSpace, COMBO_CHANGED, ADV_CHANGED); HookWidget(ui->colorRange, COMBO_CHANGED, ADV_CHANGED); //Apply button disabled until change. EnableApplyButton(false); // Initialize libff library ff_init(); LoadServiceTypes(); LoadEncoderTypes(); LoadColorRanges(); LoadFormats(); LoadSettings(false); } void OBSBasicSettings::SaveCombo(QComboBox *widget, const char *section, const char *value) { if (WidgetChanged(widget)) config_set_string(main->Config(), section, value, QT_TO_UTF8(widget->currentText())); } void OBSBasicSettings::SaveComboData(QComboBox *widget, const char *section, const char *value) { if (WidgetChanged(widget)) { QString str = GetComboData(widget); config_set_string(main->Config(), section, value, QT_TO_UTF8(str)); } } void OBSBasicSettings::SaveCheckBox(QAbstractButton *widget, const char *section, const char *value, bool invert) { if (WidgetChanged(widget)) { bool checked = widget->isChecked(); if (invert) checked = !checked; config_set_bool(main->Config(), section, value, checked); } } void OBSBasicSettings::SaveEdit(QLineEdit *widget, const char *section, const char *value) { if (WidgetChanged(widget)) config_set_string(main->Config(), section, value, QT_TO_UTF8(widget->text())); } void OBSBasicSettings::SaveSpinBox(QSpinBox *widget, const char *section, const char *value) { if (WidgetChanged(widget)) config_set_int(main->Config(), section, value, widget->value()); } void OBSBasicSettings::LoadServiceTypes() { const char *type; size_t idx = 0; while (obs_enum_service_types(idx++, &type)) { const char *name = obs_service_get_display_name(type); QString qName = QT_UTF8(name); QString qType = QT_UTF8(type); ui->streamType->addItem(qName, qType); } type = obs_service_get_type(main->GetService()); SetComboByValue(ui->streamType, type); } #define TEXT_USE_STREAM_ENC \ QTStr("Basic.Settings.Output.Adv.Recording.UseStreamEncoder") void OBSBasicSettings::LoadEncoderTypes() { const char *type; size_t idx = 0; ui->advOutRecEncoder->addItem(TEXT_USE_STREAM_ENC, "none"); while (obs_enum_encoder_types(idx++, &type)) { const char *name = obs_encoder_get_display_name(type); const char *codec = obs_get_encoder_codec(type); if (strcmp(codec, "h264") != 0) continue; QString qName = QT_UTF8(name); QString qType = QT_UTF8(type); ui->advOutEncoder->addItem(qName, qType); ui->advOutRecEncoder->addItem(qName, qType); } } #define CS_PARTIAL_STR QTStr("Basic.Settings.Advanced.Video.ColorRange.Partial") #define CS_FULL_STR QTStr("Basic.Settings.Advanced.Video.ColorRange.Full") void OBSBasicSettings::LoadColorRanges() { ui->colorRange->addItem(CS_PARTIAL_STR, "Partial"); ui->colorRange->addItem(CS_FULL_STR, "Full"); } #define AV_FORMAT_DEFAULT_STR \ QTStr("Basic.Settings.Output.Adv.FFmpeg.FormatDefault") #define AUDIO_STR \ QTStr("Basic.Settings.Output.Adv.FFmpeg.FormatAudio") #define VIDEO_STR \ QTStr("Basic.Settings.Output.Adv.FFmpeg.FormatVideo") void OBSBasicSettings::LoadFormats() { formats.reset(ff_format_supported()); const ff_format_desc *format = formats.get(); while(format != nullptr) { bool audio = ff_format_desc_has_audio(format); bool video = ff_format_desc_has_video(format); FormatDesc formatDesc(ff_format_desc_name(format), ff_format_desc_mime_type(format), format); if (audio || video) { QString itemText(ff_format_desc_name(format)); if (audio ^ video) itemText += QString(" (%1)").arg( audio ? AUDIO_STR : VIDEO_STR); ui->advOutFFFormat->addItem(itemText, qVariantFromValue(formatDesc)); } format = ff_format_desc_next(format); } ui->advOutFFFormat->model()->sort(0); ui->advOutFFFormat->insertItem(0, AV_FORMAT_DEFAULT_STR); } static void AddCodec(QComboBox *combo, const ff_codec_desc *codec_desc) { QString itemText(ff_codec_desc_name(codec_desc)); if (ff_codec_desc_is_alias(codec_desc)) itemText += QString(" (%1)").arg( ff_codec_desc_base_name(codec_desc)); CodecDesc cd(ff_codec_desc_name(codec_desc), ff_codec_desc_id(codec_desc)); combo->addItem(itemText, qVariantFromValue(cd)); } #define AV_ENCODER_DEFAULT_STR \ QTStr("Basic.Settings.Output.Adv.FFmpeg.AVEncoderDefault") static void AddDefaultCodec(QComboBox *combo, const ff_format_desc *formatDesc, ff_codec_type codecType) { CodecDesc cd = GetDefaultCodecDesc(formatDesc, codecType); int existingIdx = FindEncoder(combo, cd.name, cd.id); if (existingIdx >= 0) combo->removeItem(existingIdx); combo->addItem(QString("%1 (%2)").arg(cd.name, AV_ENCODER_DEFAULT_STR), qVariantFromValue(cd)); } #define AV_ENCODER_DISABLE_STR \ QTStr("Basic.Settings.Output.Adv.FFmpeg.AVEncoderDisable") void OBSBasicSettings::ReloadCodecs(const ff_format_desc *formatDesc) { ui->advOutFFAEncoder->blockSignals(true); ui->advOutFFVEncoder->blockSignals(true); ui->advOutFFAEncoder->clear(); ui->advOutFFVEncoder->clear(); if (formatDesc == nullptr) return; OBSFFCodecDesc codecDescs(ff_codec_supported(formatDesc)); const ff_codec_desc *codec = codecDescs.get(); while(codec != nullptr) { switch (ff_codec_desc_type(codec)) { case FF_CODEC_AUDIO: AddCodec(ui->advOutFFAEncoder, codec); break; case FF_CODEC_VIDEO: AddCodec(ui->advOutFFVEncoder, codec); break; default: break; } codec = ff_codec_desc_next(codec); } if (ff_format_desc_has_audio(formatDesc)) AddDefaultCodec(ui->advOutFFAEncoder, formatDesc, FF_CODEC_AUDIO); if (ff_format_desc_has_video(formatDesc)) AddDefaultCodec(ui->advOutFFVEncoder, formatDesc, FF_CODEC_VIDEO); ui->advOutFFAEncoder->model()->sort(0); ui->advOutFFVEncoder->model()->sort(0); QVariant disable = qVariantFromValue(CodecDesc()); ui->advOutFFAEncoder->insertItem(0, AV_ENCODER_DISABLE_STR, disable); ui->advOutFFVEncoder->insertItem(0, AV_ENCODER_DISABLE_STR, disable); ui->advOutFFAEncoder->blockSignals(false); ui->advOutFFVEncoder->blockSignals(false); } void OBSBasicSettings::LoadLanguageList() { const char *currentLang = App()->GetLocale(); ui->language->clear(); for (const auto &locale : GetLocaleNames()) { int idx = ui->language->count(); ui->language->addItem(QT_UTF8(locale.second.c_str()), QT_UTF8(locale.first.c_str())); if (locale.first == currentLang) ui->language->setCurrentIndex(idx); } ui->language->model()->sort(0); } void OBSBasicSettings::LoadThemeList() { /* Save theme if user presses Cancel */ savedTheme = string(App()->GetTheme()); ui->theme->clear(); QSet uniqueSet; string themeDir; char userThemeDir[512]; int ret = os_get_config_path(userThemeDir, sizeof(userThemeDir), "obs-studio/themes/"); GetDataFilePath("themes/", themeDir); /* Check user dir first. */ if (ret > 0) { QDirIterator it(QString(userThemeDir), QStringList() << "*.qss", QDir::Files); while (it.hasNext()) { it.next(); QString name = it.fileName().section(".",0,0); ui->theme->addItem(name); uniqueSet.insert(name); } } /* Check shipped themes. */ QDirIterator uIt(QString(themeDir.c_str()), QStringList() << "*.qss", QDir::Files); while (uIt.hasNext()) { uIt.next(); QString name = uIt.fileName().section(".",0,0); if (!uniqueSet.contains(name)) ui->theme->addItem(name); } int idx = ui->theme->findText(App()->GetTheme()); if (idx != -1) ui->theme->setCurrentIndex(idx); } void OBSBasicSettings::LoadGeneralSettings() { loading = true; LoadLanguageList(); LoadThemeList(); loading = false; } void OBSBasicSettings::LoadStream1Settings() { QLayout *layout = ui->streamContainer->layout(); obs_service_t *service = main->GetService(); const char *type = obs_service_get_type(service); loading = true; obs_data_t *settings = obs_service_get_settings(service); delete streamProperties; streamProperties = new OBSPropertiesView(settings, type, (PropertiesReloadCallback)obs_get_service_properties, 170); streamProperties->setProperty("changed", QVariant(false)); layout->addWidget(streamProperties); QObject::connect(streamProperties, SIGNAL(Changed()), this, STREAM1_CHANGED); obs_data_release(settings); loading = false; if (main->StreamingActive()) { ui->streamType->setEnabled(false); ui->streamContainer->setEnabled(false); } } void OBSBasicSettings::LoadRendererList() { const char *renderer = config_get_string(GetGlobalConfig(), "Video", "Renderer"); #ifdef _WIN32 ui->renderer->addItem(QT_UTF8("Direct3D 11")); #endif ui->renderer->addItem(QT_UTF8("OpenGL")); int idx = ui->renderer->findText(QT_UTF8(renderer)); if (idx == -1) idx = 0; ui->renderer->setCurrentIndex(idx); } Q_DECLARE_METATYPE(MonitorInfo); static string ResString(uint32_t cx, uint32_t cy) { stringstream res; res << cx << "x" << cy; return res.str(); } /* some nice default output resolution vals */ static const double vals[] = { 1.0, 1.25, (1.0/0.75), 1.5, (1.0/0.6), 1.75, 2.0, 2.25, 2.5, 2.75, 3.0 }; static const size_t numVals = sizeof(vals)/sizeof(double); void OBSBasicSettings::ResetDownscales(uint32_t cx, uint32_t cy, uint32_t out_cx, uint32_t out_cy) { QString advRescale; QString advRecRescale; QString advFFRescale; advRescale = ui->advOutRescale->lineEdit()->text(); advRecRescale = ui->advOutRecRescale->lineEdit()->text(); advFFRescale = ui->advOutFFRescale->lineEdit()->text(); ui->outputResolution->clear(); ui->advOutRescale->clear(); ui->advOutRecRescale->clear(); ui->advOutFFRescale->clear(); for (size_t idx = 0; idx < numVals; idx++) { uint32_t downscaleCX = uint32_t(double(cx) / vals[idx]); uint32_t downscaleCY = uint32_t(double(cy) / vals[idx]); uint32_t outDownscaleCX = uint32_t(double(out_cx) / vals[idx]); uint32_t outDownscaleCY = uint32_t(double(out_cy) / vals[idx]); downscaleCX &= 0xFFFFFFFC; downscaleCY &= 0xFFFFFFFE; outDownscaleCX &= 0xFFFFFFFE; outDownscaleCY &= 0xFFFFFFFE; string res = ResString(downscaleCX, downscaleCY); string outRes = ResString(outDownscaleCX, outDownscaleCY); ui->outputResolution->addItem(res.c_str()); ui->advOutRescale->addItem(outRes.c_str()); ui->advOutRecRescale->addItem(outRes.c_str()); ui->advOutFFRescale->addItem(outRes.c_str()); } string res = ResString(cx, cy); ui->outputResolution->lineEdit()->setText(res.c_str()); if (advRescale.isEmpty()) advRescale = res.c_str(); if (advRecRescale.isEmpty()) advRecRescale = res.c_str(); if (advFFRescale.isEmpty()) advFFRescale = res.c_str(); ui->advOutRescale->lineEdit()->setText(advRescale); ui->advOutRecRescale->lineEdit()->setText(advRecRescale); ui->advOutFFRescale->lineEdit()->setText(advFFRescale); } void OBSBasicSettings::LoadDownscaleFilters() { ui->downscaleFilter->addItem( QTStr("Basic.Settings.Video.DownscaleFilter.Bilinear"), QT_UTF8("bilinear")); ui->downscaleFilter->addItem( QTStr("Basic.Settings.Video.DownscaleFilter.Bicubic"), QT_UTF8("bicubic")); ui->downscaleFilter->addItem( QTStr("Basic.Settings.Video.DownscaleFilter.Lanczos"), QT_UTF8("lanczos")); const char *scaleType = config_get_string(main->Config(), "Video", "ScaleType"); if (astrcmpi(scaleType, "bilinear") == 0) ui->downscaleFilter->setCurrentIndex(0); else if (astrcmpi(scaleType, "lanczos") == 0) ui->downscaleFilter->setCurrentIndex(2); else ui->downscaleFilter->setCurrentIndex(1); } void OBSBasicSettings::LoadResolutionLists() { uint32_t cx = config_get_uint(main->Config(), "Video", "BaseCX"); uint32_t cy = config_get_uint(main->Config(), "Video", "BaseCY"); uint32_t out_cx = config_get_uint(main->Config(), "Video", "OutputCX"); uint32_t out_cy = config_get_uint(main->Config(), "Video", "OutputCY"); vector monitors; ui->baseResolution->clear(); GetMonitors(monitors); for (MonitorInfo &monitor : monitors) { string res = ResString(monitor.cx, monitor.cy); ui->baseResolution->addItem(res.c_str()); } ResetDownscales(cx, cy, out_cx, out_cy); ui->baseResolution->lineEdit()->setText(ResString(cx, cy).c_str()); ui->outputResolution->lineEdit()->setText( ResString(out_cx, out_cy).c_str()); } static inline void LoadFPSCommon(OBSBasic *main, Ui::OBSBasicSettings *ui) { const char *val = config_get_string(main->Config(), "Video", "FPSCommon"); int idx = ui->fpsCommon->findText(val); if (idx == -1) idx = 3; ui->fpsCommon->setCurrentIndex(idx); } static inline void LoadFPSInteger(OBSBasic *main, Ui::OBSBasicSettings *ui) { uint32_t val = config_get_uint(main->Config(), "Video", "FPSInt"); ui->fpsInteger->setValue(val); } static inline void LoadFPSFraction(OBSBasic *main, Ui::OBSBasicSettings *ui) { uint32_t num = config_get_uint(main->Config(), "Video", "FPSNum"); uint32_t den = config_get_uint(main->Config(), "Video", "FPSDen"); ui->fpsNumerator->setValue(num); ui->fpsDenominator->setValue(den); } void OBSBasicSettings::LoadFPSData() { LoadFPSCommon(main, ui.get()); LoadFPSInteger(main, ui.get()); LoadFPSFraction(main, ui.get()); uint32_t fpsType = config_get_uint(main->Config(), "Video", "FPSType"); if (fpsType > 2) fpsType = 0; ui->fpsType->setCurrentIndex(fpsType); ui->fpsTypes->setCurrentIndex(fpsType); } void OBSBasicSettings::LoadVideoSettings() { loading = true; if (video_output_active(obs_get_video())) { ui->videoPage->setEnabled(false); ui->videoMsg->setText( QTStr("Basic.Settings.Video.CurrentlyActive")); } LoadRendererList(); LoadResolutionLists(); LoadFPSData(); LoadDownscaleFilters(); loading = false; } void OBSBasicSettings::LoadSimpleOutputSettings() { const char *path = config_get_string(main->Config(), "SimpleOutput", "FilePath"); int videoBitrate = config_get_uint(main->Config(), "SimpleOutput", "VBitrate"); int videoBufsize = config_get_uint(main->Config(), "SimpleOutput", "VBufsize"); int audioBitrate = config_get_uint(main->Config(), "SimpleOutput", "ABitrate"); bool reconnect = config_get_bool(main->Config(), "SimpleOutput", "Reconnect"); int retryDelay = config_get_uint(main->Config(), "SimpleOutput", "RetryDelay"); int maxRetries = config_get_uint(main->Config(), "SimpleOutput", "MaxRetries"); bool advanced = config_get_bool(main->Config(), "SimpleOutput", "UseAdvanced"); bool useCBR = config_get_bool(main->Config(), "SimpleOutput", "UseCBR"); bool useBufsize = config_get_bool(main->Config(), "SimpleOutput", "UseBufsize"); const char *preset = config_get_string(main->Config(), "SimpleOutput", "Preset"); const char *custom = config_get_string(main->Config(), "SimpleOutput", "x264Settings"); ui->simpleOutputPath->setText(path); ui->simpleOutputVBitrate->setValue(videoBitrate); ui->simpleOutUseBufsize->setChecked(useBufsize); ui->simpleOutVBufsize->setValue( useBufsize ? videoBufsize : videoBitrate); SetComboByName(ui->simpleOutputABitrate, std::to_string(audioBitrate).c_str()); ui->simpleOutReconnect->setChecked(reconnect); ui->simpleOutRetryDelay->setValue(retryDelay); ui->simpleOutMaxRetries->setValue(maxRetries); ui->simpleOutAdvanced->setChecked(advanced); ui->simpleOutUseCBR->setChecked(useCBR); ui->simpleOutPreset->setCurrentText(preset); ui->simpleOutCustom->setText(custom); } void OBSBasicSettings::LoadAdvOutputStreamingSettings() { bool reconnect = config_get_bool(main->Config(), "AdvOut", "Reconnect"); int retryDelay = config_get_int(main->Config(), "AdvOut", "RetryDelay"); int maxRetries = config_get_int(main->Config(), "AdvOut", "MaxRetries"); bool rescale = config_get_bool(main->Config(), "AdvOut", "Rescale"); const char *rescaleRes = config_get_string(main->Config(), "AdvOut", "RescaleRes"); int trackIndex = config_get_int(main->Config(), "AdvOut", "TrackIndex"); bool applyServiceSettings = config_get_bool(main->Config(), "AdvOut", "ApplyServiceSettings"); ui->advOutReconnect->setChecked(reconnect); ui->advOutRetryDelay->setValue(retryDelay); ui->advOutMaxRetries->setValue(maxRetries); ui->advOutApplyService->setChecked(applyServiceSettings); ui->advOutUseRescale->setChecked(rescale); ui->advOutRescale->setEnabled(rescale); ui->advOutRescale->setCurrentText(rescaleRes); switch (trackIndex) { case 1: ui->advOutTrack1->setChecked(true); break; case 2: ui->advOutTrack2->setChecked(true); break; case 3: ui->advOutTrack3->setChecked(true); break; case 4: ui->advOutTrack4->setChecked(true); break; } } OBSPropertiesView *OBSBasicSettings::CreateEncoderPropertyView( const char *encoder, const char *path, bool changed) { obs_data_t *settings = obs_encoder_defaults(encoder); OBSPropertiesView *view; char encoderJsonPath[512]; int ret = os_get_config_path(encoderJsonPath, sizeof(encoderJsonPath), path); if (ret > 0) { BPtr jsonData = os_quick_read_utf8_file(encoderJsonPath); if (!!jsonData) { obs_data_t *data = obs_data_create_from_json(jsonData); obs_data_apply(settings, data); obs_data_release(data); } } view = new OBSPropertiesView(settings, encoder, (PropertiesReloadCallback)obs_get_encoder_properties, 170); view->setFrameShape(QFrame::StyledPanel); view->setProperty("changed", QVariant(changed)); QObject::connect(view, SIGNAL(Changed()), this, SLOT(OutputsChanged())); obs_data_release(settings); return view; } void OBSBasicSettings::LoadAdvOutputStreamingEncoderProperties() { const char *encoder = config_get_string(main->Config(), "AdvOut", "Encoder"); delete streamEncoderProps; streamEncoderProps = CreateEncoderPropertyView(encoder, "obs-studio/basic/streamEncoder.json"); ui->advOutputStreamTab->layout()->addWidget(streamEncoderProps); SetComboByValue(ui->advOutEncoder, encoder); } void OBSBasicSettings::LoadAdvOutputRecordingSettings() { const char *type = config_get_string(main->Config(), "AdvOut", "RecType"); const char *path = config_get_string(main->Config(), "AdvOut", "RecFilePath"); bool rescale = config_get_bool(main->Config(), "AdvOut", "RecRescale"); const char *rescaleRes = config_get_string(main->Config(), "AdvOut", "RecRescaleRes"); int trackIndex = config_get_int(main->Config(), "AdvOut", "RecTrackIndex"); int typeIndex = (astrcmpi(type, "FFmpeg") == 0) ? 1 : 0; ui->advOutRecType->setCurrentIndex(typeIndex); ui->advOutRecPath->setText(path); ui->advOutRecUseRescale->setChecked(rescale); ui->advOutRecRescale->setCurrentText(rescaleRes); switch (trackIndex) { case 1: ui->advOutRecTrack1->setChecked(true); break; case 2: ui->advOutRecTrack2->setChecked(true); break; case 3: ui->advOutRecTrack3->setChecked(true); break; case 4: ui->advOutRecTrack4->setChecked(true); break; } } void OBSBasicSettings::LoadAdvOutputRecordingEncoderProperties() { const char *encoder = config_get_string(main->Config(), "AdvOut", "RecEncoder"); delete recordEncoderProps; recordEncoderProps = nullptr; if (astrcmpi(encoder, "none") != 0) { recordEncoderProps = CreateEncoderPropertyView(encoder, "obs-studio/basic/recordEncoder.json"); ui->advOutRecStandard->layout()->addWidget(recordEncoderProps); } SetComboByValue(ui->advOutRecEncoder, encoder); } static void SelectFormat(QComboBox *combo, const char *name, const char *mimeType) { FormatDesc formatDesc(name, mimeType); for(int i = 0; i < combo->count(); i++) { QVariant v = combo->itemData(i); if (!v.isNull()) { if (formatDesc == v.value()) { combo->setCurrentIndex(i); return; } } } combo->setCurrentIndex(0); } static void SelectEncoder(QComboBox *combo, const char *name, int id) { int idx = FindEncoder(combo, name, id); if (idx >= 0) combo->setCurrentIndex(idx); } void OBSBasicSettings::LoadAdvOutputFFmpegSettings() { const char *url = config_get_string(main->Config(), "AdvOut", "FFURL"); const char *format = config_get_string(main->Config(), "AdvOut", "FFFormat"); const char *mimeType = config_get_string(main->Config(), "AdvOut", "FFFormatMimeType"); int videoBitrate = config_get_int(main->Config(), "AdvOut", "FFVBitrate"); bool rescale = config_get_bool(main->Config(), "AdvOut", "FFRescale"); const char *rescaleRes = config_get_string(main->Config(), "AdvOut", "FFRescaleRes"); const char *vEncoder = config_get_string(main->Config(), "AdvOut", "FFVEncoder"); int vEncoderId = config_get_int(main->Config(), "AdvOut", "FFVEncoderId"); const char *vEncCustom = config_get_string(main->Config(), "AdvOut", "FFVCustom"); int audioBitrate = config_get_int(main->Config(), "AdvOut", "FFABitrate"); int audioTrack = config_get_int(main->Config(), "AdvOut", "FFAudioTrack"); const char *aEncoder = config_get_string(main->Config(), "AdvOut", "FFAEncoder"); int aEncoderId = config_get_int(main->Config(), "AdvOut", "FFAEncoderId"); const char *aEncCustom = config_get_string(main->Config(), "AdvOut", "FFACustom"); ui->advOutFFURL->setText(url); SelectFormat(ui->advOutFFFormat, format, mimeType); ui->advOutFFVBitrate->setValue(videoBitrate); ui->advOutFFUseRescale->setChecked(rescale); ui->advOutFFRescale->setEnabled(rescale); ui->advOutFFRescale->setCurrentText(rescaleRes); SelectEncoder(ui->advOutFFVEncoder, vEncoder, vEncoderId); ui->advOutFFVCfg->setText(vEncCustom); ui->advOutFFABitrate->setValue(audioBitrate); SelectEncoder(ui->advOutFFAEncoder, aEncoder, aEncoderId); ui->advOutFFACfg->setText(aEncCustom); switch (audioTrack) { case 1: ui->advOutFFTrack1->setChecked(true); break; case 2: ui->advOutFFTrack2->setChecked(true); break; case 3: ui->advOutFFTrack3->setChecked(true); break; case 4: ui->advOutFFTrack4->setChecked(true); break; } } void OBSBasicSettings::LoadAdvOutputAudioSettings() { int track1Bitrate = config_get_uint(main->Config(), "AdvOut", "Track1Bitrate"); int track2Bitrate = config_get_uint(main->Config(), "AdvOut", "Track2Bitrate"); int track3Bitrate = config_get_uint(main->Config(), "AdvOut", "Track3Bitrate"); int track4Bitrate = config_get_uint(main->Config(), "AdvOut", "Track4Bitrate"); const char *name1 = config_get_string(main->Config(), "AdvOut", "Track1Name"); const char *name2 = config_get_string(main->Config(), "AdvOut", "Track2Name"); const char *name3 = config_get_string(main->Config(), "AdvOut", "Track3Name"); const char *name4 = config_get_string(main->Config(), "AdvOut", "Track4Name"); SetComboByName(ui->advOutTrack1Bitrate, std::to_string(track1Bitrate).c_str()); SetComboByName(ui->advOutTrack2Bitrate, std::to_string(track2Bitrate).c_str()); SetComboByName(ui->advOutTrack3Bitrate, std::to_string(track3Bitrate).c_str()); SetComboByName(ui->advOutTrack4Bitrate, std::to_string(track4Bitrate).c_str()); ui->advOutTrack1Name->setText(name1); ui->advOutTrack2Name->setText(name2); ui->advOutTrack3Name->setText(name3); ui->advOutTrack4Name->setText(name4); } void OBSBasicSettings::LoadOutputSettings() { loading = true; const char *mode = config_get_string(main->Config(), "Output", "Mode"); int modeIdx = astrcmpi(mode, "Advanced") == 0 ? 1 : 0; ui->outputMode->setCurrentIndex(modeIdx); LoadSimpleOutputSettings(); LoadAdvOutputStreamingSettings(); LoadAdvOutputStreamingEncoderProperties(); LoadAdvOutputRecordingSettings(); LoadAdvOutputRecordingEncoderProperties(); LoadAdvOutputFFmpegSettings(); LoadAdvOutputAudioSettings(); if (video_output_active(obs_get_video())) { ui->outputMode->setEnabled(false); ui->outputModeLabel->setEnabled(false); ui->advOutTopContainer->setEnabled(false); ui->advOutRecTopContainer->setEnabled(false); ui->advOutRecTypeContainer->setEnabled(false); ui->advOutputAudioTracksTab->setEnabled(false); } loading = false; } void OBSBasicSettings::SetAdvOutputFFmpegEnablement( ff_codec_type encoderType, bool enabled, bool enableEncoder) { bool rescale = config_get_bool(main->Config(), "AdvOut", "FFRescale"); switch (encoderType) { case FF_CODEC_VIDEO: ui->advOutFFVBitrate->setEnabled(enabled); ui->advOutFFUseRescale->setEnabled(enabled); ui->advOutFFRescale->setEnabled(enabled && rescale); ui->advOutFFVEncoder->setEnabled(enabled || enableEncoder); ui->advOutFFVCfg->setEnabled(enabled); break; case FF_CODEC_AUDIO: ui->advOutFFABitrate->setEnabled(enabled); ui->advOutFFAEncoder->setEnabled(enabled || enableEncoder); ui->advOutFFACfg->setEnabled(enabled); ui->advOutFFTrack1->setEnabled(enabled); ui->advOutFFTrack2->setEnabled(enabled); ui->advOutFFTrack3->setEnabled(enabled); ui->advOutFFTrack4->setEnabled(enabled); default: break; } } static inline void LoadListValue(QComboBox *widget, const char *text, const char *val) { widget->addItem(QT_UTF8(text), QT_UTF8(val)); } void OBSBasicSettings::LoadListValues(QComboBox *widget, obs_property_t *prop, const char *configName) { size_t count = obs_property_list_item_count(prop); const char *deviceId = config_get_string(main->Config(), "Audio", configName); widget->addItem(QTStr("Disabled"), "disabled"); for (size_t i = 0; i < count; i++) { const char *name = obs_property_list_item_name(prop, i); const char *val = obs_property_list_item_string(prop, i); LoadListValue(widget, name, val); } int idx = widget->findData(QVariant(QT_UTF8(deviceId))); if (idx == -1) { deviceId = config_get_default_string(main->Config(), "Audio", configName); idx = widget->findData(QVariant(QT_UTF8(deviceId))); } if (idx != -1) widget->setCurrentIndex(idx); } void OBSBasicSettings::LoadAudioDevices() { const char *input_id = App()->InputAudioSource(); const char *output_id = App()->OutputAudioSource(); obs_properties_t *input_props = obs_get_source_properties( OBS_SOURCE_TYPE_INPUT, input_id); obs_properties_t *output_props = obs_get_source_properties( OBS_SOURCE_TYPE_INPUT, output_id); if (input_props) { obs_property_t *inputs = obs_properties_get(input_props, "device_id"); LoadListValues(ui->auxAudioDevice1, inputs, "AuxDevice1"); LoadListValues(ui->auxAudioDevice2, inputs, "AuxDevice2"); LoadListValues(ui->auxAudioDevice3, inputs, "AuxDevice3"); obs_properties_destroy(input_props); } if (output_props) { obs_property_t *outputs = obs_properties_get(output_props, "device_id"); LoadListValues(ui->desktopAudioDevice1, outputs, "DesktopDevice1"); LoadListValues(ui->desktopAudioDevice2, outputs, "DesktopDevice2"); obs_properties_destroy(output_props); } } void OBSBasicSettings::LoadAudioSettings() { uint32_t sampleRate = config_get_uint(main->Config(), "Audio", "SampleRate"); const char *speakers = config_get_string(main->Config(), "Audio", "ChannelSetup"); loading = true; const char *str; if (sampleRate == 22050) str = "22.05khz"; else if (sampleRate == 48000) str = "48khz"; else str = "44.1khz"; int sampleRateIdx = ui->sampleRate->findText(str); if (sampleRateIdx != -1) ui->sampleRate->setCurrentIndex(sampleRateIdx); if (strcmp(speakers, "Mono") == 0) ui->channelSetup->setCurrentIndex(0); else ui->channelSetup->setCurrentIndex(1); LoadAudioDevices(); loading = false; } void OBSBasicSettings::LoadAdvancedSettings() { uint32_t audioBufferingTime = config_get_uint(main->Config(), "Audio", "BufferingTime"); const char *videoColorFormat = config_get_string(main->Config(), "Video", "ColorFormat"); const char *videoColorSpace = config_get_string(main->Config(), "Video", "ColorSpace"); const char *videoColorRange = config_get_string(main->Config(), "Video", "ColorRange"); loading = true; ui->audioBufferingTime->setValue(audioBufferingTime); SetComboByName(ui->colorFormat, videoColorFormat); SetComboByName(ui->colorSpace, videoColorSpace); SetComboByValue(ui->colorRange, videoColorRange); if (video_output_active(obs_get_video())) { ui->advancedVideoContainer->setEnabled(false); } loading = false; } void OBSBasicSettings::LoadSettings(bool changedOnly) { if (!changedOnly || generalChanged) LoadGeneralSettings(); if (!changedOnly || stream1Changed) LoadStream1Settings(); if (!changedOnly || outputsChanged) LoadOutputSettings(); if (!changedOnly || audioChanged) LoadAudioSettings(); if (!changedOnly || videoChanged) LoadVideoSettings(); if (!changedOnly || advancedChanged) LoadAdvancedSettings(); } void OBSBasicSettings::SaveGeneralSettings() { int languageIndex = ui->language->currentIndex(); QVariant langData = ui->language->itemData(languageIndex); string language = langData.toString().toStdString(); if (WidgetChanged(ui->language)) config_set_string(GetGlobalConfig(), "General", "Language", language.c_str()); int themeIndex = ui->theme->currentIndex(); QString themeData = ui->theme->itemText(themeIndex); string theme = themeData.toStdString(); if (WidgetChanged(ui->theme)) { config_set_string(GetGlobalConfig(), "General", "Theme", theme.c_str()); App()->SetTheme(theme); } } void OBSBasicSettings::SaveStream1Settings() { QString streamType = GetComboData(ui->streamType); obs_service_t *newService = obs_service_create(QT_TO_UTF8(streamType), "default_service", streamProperties->GetSettings()); if (!newService) return; main->SetService(newService); main->SaveService(); } void OBSBasicSettings::SaveVideoSettings() { QString baseResolution = ui->baseResolution->currentText(); QString outputResolution = ui->outputResolution->currentText(); int fpsType = ui->fpsType->currentIndex(); uint32_t cx = 0, cy = 0; /* ------------------- */ if (WidgetChanged(ui->renderer)) config_set_string(App()->GlobalConfig(), "Video", "Renderer", QT_TO_UTF8(ui->renderer->currentText())); if (WidgetChanged(ui->baseResolution) && ConvertResText(QT_TO_UTF8(baseResolution), cx, cy)) { config_set_uint(main->Config(), "Video", "BaseCX", cx); config_set_uint(main->Config(), "Video", "BaseCY", cy); } if (WidgetChanged(ui->outputResolution) && ConvertResText(QT_TO_UTF8(outputResolution), cx, cy)) { config_set_uint(main->Config(), "Video", "OutputCX", cx); config_set_uint(main->Config(), "Video", "OutputCY", cy); } if (WidgetChanged(ui->fpsType)) config_set_uint(main->Config(), "Video", "FPSType", fpsType); SaveCombo(ui->fpsCommon, "Video", "FPSCommon"); SaveSpinBox(ui->fpsInteger, "Video", "FPSInt"); SaveSpinBox(ui->fpsNumerator, "Video", "FPSNum"); SaveSpinBox(ui->fpsDenominator, "Video", "FPSDen"); SaveComboData(ui->downscaleFilter, "Video", "ScaleType"); } void OBSBasicSettings::SaveAdvancedSettings() { SaveSpinBox(ui->audioBufferingTime, "Audio", "BufferingTime"); SaveCombo(ui->colorFormat, "Video", "ColorFormat"); SaveCombo(ui->colorSpace, "Video", "ColorSpace"); SaveComboData(ui->colorRange, "Video", "ColorRange"); } static inline const char *OutputModeFromIdx(int idx) { if (idx == 1) return "Advanced"; else return "Simple"; } static inline const char *RecTypeFromIdx(int idx) { if (idx == 1) return "FFmpeg"; else return "Standard"; } static void WriteJsonData(OBSPropertiesView *view, const char *path) { char full_path[512]; if (!view || !WidgetChanged(view)) return; int ret = os_get_config_path(full_path, sizeof(full_path), path); if (ret > 0) { obs_data_t *settings = view->GetSettings(); if (settings) { const char *json = obs_data_get_json(settings); if (json && *json) { os_quick_write_utf8_file(full_path, json, strlen(json), false); } } } } static void SaveTrackIndex(config_t *config, const char *section, const char *name, QAbstractButton *check1, QAbstractButton *check2, QAbstractButton *check3, QAbstractButton *check4) { if (check1->isChecked()) config_set_int(config, section, name, 1); else if (check2->isChecked()) config_set_int(config, section, name, 2); else if (check3->isChecked()) config_set_int(config, section, name, 3); else if (check4->isChecked()) config_set_int(config, section, name, 4); } void OBSBasicSettings::SaveFormat(QComboBox *combo) { QVariant v = combo->currentData(); if (!v.isNull()) { FormatDesc desc = v.value(); config_set_string(main->Config(), "AdvOut", "FFFormat", desc.name); config_set_string(main->Config(), "AdvOut", "FFFormatMimeType", desc.mimeType); } else { config_set_string(main->Config(), "AdvOut", "FFFormat", nullptr); config_set_string(main->Config(), "AdvOut", "FFFormatMimeType", nullptr); } } void OBSBasicSettings::SaveEncoder(QComboBox *combo, const char *section, const char *value) { QVariant v = combo->currentData(); CodecDesc cd; if (!v.isNull()) cd = v.value(); config_set_int(main->Config(), section, QT_TO_UTF8(QString("%1Id").arg(value)), cd.id); if (cd.id != 0) config_set_string(main->Config(), section, value, cd.name); else config_set_string(main->Config(), section, value, nullptr); } void OBSBasicSettings::SaveOutputSettings() { config_set_string(main->Config(), "Output", "Mode", OutputModeFromIdx(ui->outputMode->currentIndex())); SaveSpinBox(ui->simpleOutputVBitrate, "SimpleOutput", "VBitrate"); SaveCombo(ui->simpleOutputABitrate, "SimpleOutput", "ABitrate"); SaveEdit(ui->simpleOutputPath, "SimpleOutput", "FilePath"); SaveCheckBox(ui->simpleOutReconnect, "SimpleOutput", "Reconnect"); SaveSpinBox(ui->simpleOutRetryDelay, "SimpleOutput", "RetryDelay"); SaveSpinBox(ui->simpleOutMaxRetries, "SimpleOutput", "MaxRetries"); SaveCheckBox(ui->simpleOutAdvanced, "SimpleOutput", "UseAdvanced"); SaveCheckBox(ui->simpleOutUseCBR, "SimpleOutput", "UseCBR"); SaveCheckBox(ui->simpleOutUseBufsize, "SimpleOutput", "UseBufsize"); SaveCombo(ui->simpleOutPreset, "SimpleOutput", "Preset"); SaveEdit(ui->simpleOutCustom, "SimpleOutput", "x264Settings"); if (ui->simpleOutUseBufsize->isChecked()) SaveSpinBox(ui->simpleOutVBufsize, "SimpleOutput", "VBufsize"); SaveCheckBox(ui->advOutReconnect, "AdvOut", "Reconnect"); SaveSpinBox(ui->advOutRetryDelay, "AdvOut", "RetryDelay"); SaveSpinBox(ui->advOutMaxRetries, "AdvOut", "MaxRetries"); SaveCheckBox(ui->advOutApplyService, "AdvOut", "ApplyServiceSettings"); SaveComboData(ui->advOutEncoder, "AdvOut", "Encoder"); SaveCheckBox(ui->advOutUseRescale, "AdvOut", "Rescale"); SaveCombo(ui->advOutRescale, "AdvOut", "RescaleRes"); SaveTrackIndex(main->Config(), "AdvOut", "TrackIndex", ui->advOutTrack1, ui->advOutTrack2, ui->advOutTrack3, ui->advOutTrack4); config_set_string(main->Config(), "AdvOut", "RecType", RecTypeFromIdx(ui->advOutRecType->currentIndex())); SaveEdit(ui->advOutRecPath, "AdvOut", "RecFilePath"); SaveComboData(ui->advOutRecEncoder, "AdvOut", "RecEncoder"); SaveCheckBox(ui->advOutRecUseRescale, "AdvOut", "RecRescale"); SaveCombo(ui->advOutRecRescale, "AdvOut", "RecRescaleRes"); SaveTrackIndex(main->Config(), "AdvOut", "RecTrackIndex", ui->advOutRecTrack1, ui->advOutRecTrack2, ui->advOutRecTrack3, ui->advOutRecTrack4); SaveEdit(ui->advOutFFURL, "AdvOut", "FFURL"); SaveFormat(ui->advOutFFFormat); SaveSpinBox(ui->advOutFFVBitrate, "AdvOut", "FFVBitrate"); SaveCheckBox(ui->advOutFFUseRescale, "AdvOut", "FFRescale"); SaveCombo(ui->advOutFFRescale, "AdvOut", "FFRescaleRes"); SaveEncoder(ui->advOutFFVEncoder, "AdvOut", "FFVEncoder"); SaveEdit(ui->advOutFFVCfg, "AdvOut", "FFVCustom"); SaveSpinBox(ui->advOutFFABitrate, "AdvOut", "FFABitrate"); SaveEncoder(ui->advOutFFAEncoder, "AdvOut", "FFAEncoder"); SaveEdit(ui->advOutFFACfg, "AdvOut", "FFACustom"); SaveTrackIndex(main->Config(), "AdvOut", "FFAudioTrack", ui->advOutFFTrack1, ui->advOutFFTrack2, ui->advOutFFTrack3, ui->advOutFFTrack4); SaveCombo(ui->advOutTrack1Bitrate, "AdvOut", "Track1Bitrate"); SaveCombo(ui->advOutTrack2Bitrate, "AdvOut", "Track2Bitrate"); SaveCombo(ui->advOutTrack3Bitrate, "AdvOut", "Track3Bitrate"); SaveCombo(ui->advOutTrack4Bitrate, "AdvOut", "Track4Bitrate"); SaveEdit(ui->advOutTrack1Name, "AdvOut", "Track1Name"); SaveEdit(ui->advOutTrack2Name, "AdvOut", "Track2Name"); SaveEdit(ui->advOutTrack3Name, "AdvOut", "Track3Name"); SaveEdit(ui->advOutTrack4Name, "AdvOut", "Track4Name"); WriteJsonData(streamEncoderProps, "obs-studio/basic/streamEncoder.json"); WriteJsonData(recordEncoderProps, "obs-studio/basic/recordEncoder.json"); main->ResetOutputs(); } void OBSBasicSettings::SaveAudioSettings() { QString sampleRateStr = ui->sampleRate->currentText(); int channelSetupIdx = ui->channelSetup->currentIndex(); const char *channelSetup = (channelSetupIdx == 0) ? "Mono" : "Stereo"; int sampleRate = 44100; if (sampleRateStr == "22.05khz") sampleRate = 22050; else if (sampleRateStr == "48khz") sampleRate = 48000; if (WidgetChanged(ui->sampleRate)) config_set_uint(main->Config(), "Audio", "SampleRate", sampleRate); if (WidgetChanged(ui->channelSetup)) config_set_string(main->Config(), "Audio", "ChannelSetup", channelSetup); SaveComboData(ui->desktopAudioDevice1, "Audio", "DesktopDevice1"); SaveComboData(ui->desktopAudioDevice2, "Audio", "DesktopDevice2"); SaveComboData(ui->auxAudioDevice1, "Audio", "AuxDevice1"); SaveComboData(ui->auxAudioDevice2, "Audio", "AuxDevice2"); SaveComboData(ui->auxAudioDevice3, "Audio", "AuxDevice3"); main->ResetAudioDevices(); } void OBSBasicSettings::SaveSettings() { if (generalChanged) SaveGeneralSettings(); if (stream1Changed) SaveStream1Settings(); if (outputsChanged) SaveOutputSettings(); if (audioChanged) SaveAudioSettings(); if (videoChanged) SaveVideoSettings(); if (advancedChanged) SaveAdvancedSettings(); if (videoChanged || advancedChanged) main->ResetVideo(); config_save(main->Config()); config_save(GetGlobalConfig()); } bool OBSBasicSettings::QueryChanges() { QMessageBox::StandardButton button; button = QMessageBox::question(this, QTStr("Basic.Settings.ConfirmTitle"), QTStr("Basic.Settings.Confirm"), QMessageBox::Yes | QMessageBox::No | QMessageBox::Cancel); if (button == QMessageBox::Cancel) return false; else if (button == QMessageBox::Yes) SaveSettings(); else LoadSettings(true); ClearChanged(); return true; } void OBSBasicSettings::closeEvent(QCloseEvent *event) { if (Changed() && !QueryChanges()) event->ignore(); } void OBSBasicSettings::on_theme_activated(int idx) { string currT = ui->theme->itemText(idx).toStdString(); App()->SetTheme(currT); } void OBSBasicSettings::on_simpleOutUseBufsize_toggled(bool checked) { if (!checked) ui->simpleOutVBufsize->setValue( ui->simpleOutputVBitrate->value()); } void OBSBasicSettings::on_simpleOutputVBitrate_valueChanged(int val) { if (!ui->simpleOutUseBufsize->isChecked()) ui->simpleOutVBufsize->setValue(val); } void OBSBasicSettings::on_listWidget_itemSelectionChanged() { int row = ui->listWidget->currentRow(); if (loading || row == pageIndex) return; pageIndex = row; } void OBSBasicSettings::on_buttonBox_clicked(QAbstractButton *button) { QDialogButtonBox::ButtonRole val = ui->buttonBox->buttonRole(button); if (val == QDialogButtonBox::ApplyRole || val == QDialogButtonBox::AcceptRole) { SaveSettings(); ClearChanged(); } if (val == QDialogButtonBox::AcceptRole || val == QDialogButtonBox::RejectRole) { if (val == QDialogButtonBox::RejectRole) App()->SetTheme(savedTheme); ClearChanged(); close(); } } void OBSBasicSettings::on_streamType_currentIndexChanged(int idx) { if (loading) return; QLayout *layout = ui->streamContainer->layout(); QString streamType = ui->streamType->itemData(idx).toString(); obs_data_t *settings = obs_service_defaults(QT_TO_UTF8(streamType)); delete streamProperties; streamProperties = new OBSPropertiesView(settings, QT_TO_UTF8(streamType), (PropertiesReloadCallback)obs_get_service_properties, 170); streamProperties->setProperty("changed", QVariant(true)); layout->addWidget(streamProperties); QObject::connect(streamProperties, SIGNAL(Changed()), this, STREAM1_CHANGED); obs_data_release(settings); } void OBSBasicSettings::on_simpleOutputBrowse_clicked() { QString dir = QFileDialog::getExistingDirectory(this, QTStr("Basic.Settings.Output.SelectDirectory"), ui->simpleOutputPath->text(), QFileDialog::ShowDirsOnly | QFileDialog::DontResolveSymlinks); if (dir.isEmpty()) return; ui->simpleOutputPath->setText(dir); } void OBSBasicSettings::on_advOutRecPathBrowse_clicked() { QString dir = QFileDialog::getExistingDirectory(this, QTStr("Basic.Settings.Output.SelectDirectory"), ui->advOutRecPath->text(), QFileDialog::ShowDirsOnly | QFileDialog::DontResolveSymlinks); if (dir.isEmpty()) return; ui->advOutRecPath->setText(dir); } void OBSBasicSettings::on_advOutFFPathBrowse_clicked() { QString filter; filter += QTStr("Basic.Settings.Output.Adv.FFmpeg.SaveFilter.Common"); filter += " (*.avi *.mp4 *.flv *.ts *.mkv *.wav *.aac);;"; filter += QTStr("Basic.Settings.Output.Adv.FFmpeg.SaveFilter.All"); filter += " (*.*)"; QString file = QFileDialog::getSaveFileName(this, QTStr("Basic.Settings.Output.SelectFile"), ui->simpleOutputPath->text(), filter); if (file.isEmpty()) return; ui->advOutFFURL->setText(file); } void OBSBasicSettings::on_advOutEncoder_currentIndexChanged(int idx) { QString encoder = GetComboData(ui->advOutEncoder); delete streamEncoderProps; streamEncoderProps = CreateEncoderPropertyView(QT_TO_UTF8(encoder), "obs-studio/basic/streamEncoder.json", true); ui->advOutputStreamTab->layout()->addWidget(streamEncoderProps); UNUSED_PARAMETER(idx); } void OBSBasicSettings::on_advOutRecEncoder_currentIndexChanged(int idx) { ui->advOutRecUseRescale->setEnabled(idx > 0); ui->advOutRecRescaleContainer->setEnabled(idx > 0); delete recordEncoderProps; recordEncoderProps = nullptr; if (idx > 0) { QString encoder = GetComboData(ui->advOutRecEncoder); recordEncoderProps = CreateEncoderPropertyView( QT_TO_UTF8(encoder), "obs-studio/basic/recordEncoder.json", true); ui->advOutRecStandard->layout()->addWidget(recordEncoderProps); } } #define DEFAULT_CONTAINER_STR \ QTStr("Basic.Settings.Output.Adv.FFmpeg.FormatDescDef") void OBSBasicSettings::on_advOutFFFormat_currentIndexChanged(int idx) { const QVariant itemDataVariant = ui->advOutFFFormat->itemData(idx); if (!itemDataVariant.isNull()) { FormatDesc desc = itemDataVariant.value(); SetAdvOutputFFmpegEnablement(FF_CODEC_AUDIO, ff_format_desc_has_audio(desc.desc), false); SetAdvOutputFFmpegEnablement(FF_CODEC_VIDEO, ff_format_desc_has_video(desc.desc), false); ReloadCodecs(desc.desc); ui->advOutFFFormatDesc->setText(ff_format_desc_long_name( desc.desc)); CodecDesc defaultAudioCodecDesc = GetDefaultCodecDesc(desc.desc, FF_CODEC_AUDIO); CodecDesc defaultVideoCodecDesc = GetDefaultCodecDesc(desc.desc, FF_CODEC_VIDEO); SelectEncoder(ui->advOutFFAEncoder, defaultAudioCodecDesc.name, defaultAudioCodecDesc.id); SelectEncoder(ui->advOutFFVEncoder, defaultVideoCodecDesc.name, defaultVideoCodecDesc.id); } else { ReloadCodecs(nullptr); ui->advOutFFFormatDesc->setText(DEFAULT_CONTAINER_STR); } } void OBSBasicSettings::on_advOutFFAEncoder_currentIndexChanged(int idx) { const QVariant itemDataVariant = ui->advOutFFAEncoder->itemData(idx); if (!itemDataVariant.isNull()) { CodecDesc desc = itemDataVariant.value(); SetAdvOutputFFmpegEnablement(FF_CODEC_AUDIO, desc.id != 0 || desc.name != nullptr, true); } } void OBSBasicSettings::on_advOutFFVEncoder_currentIndexChanged(int idx) { const QVariant itemDataVariant = ui->advOutFFVEncoder->itemData(idx); if (!itemDataVariant.isNull()) { CodecDesc desc = itemDataVariant.value(); SetAdvOutputFFmpegEnablement(FF_CODEC_VIDEO, desc.id != 0 || desc.name != nullptr, true); } } #define INVALID_RES_STR "Basic.Settings.Video.InvalidResolution" static bool ValidResolutions(Ui::OBSBasicSettings *ui) { QString baseRes = ui->baseResolution->lineEdit()->text(); QString outputRes = ui->outputResolution->lineEdit()->text(); uint32_t cx, cy; if (!ConvertResText(QT_TO_UTF8(baseRes), cx, cy) || !ConvertResText(QT_TO_UTF8(outputRes), cx, cy)) { ui->videoMsg->setText(QTStr(INVALID_RES_STR)); return false; } ui->videoMsg->setText(""); return true; } void OBSBasicSettings::on_baseResolution_editTextChanged(const QString &text) { if (!loading && ValidResolutions(ui.get())) { QString baseResolution = text; uint32_t cx, cy, out_cx, out_cy; ConvertResText(QT_TO_UTF8(baseResolution), cx, cy); QString outRes = ui->outputResolution->lineEdit()->text(); ConvertResText(QT_TO_UTF8(outRes), out_cx, out_cy); ResetDownscales(cx, cy, out_cx, out_cy); } } void OBSBasicSettings::GeneralChanged() { if (!loading) { generalChanged = true; sender()->setProperty("changed", QVariant(true)); EnableApplyButton(true); } } void OBSBasicSettings::Stream1Changed() { if (!loading) { stream1Changed = true; sender()->setProperty("changed", QVariant(true)); EnableApplyButton(true); } } void OBSBasicSettings::OutputsChanged() { if (!loading) { outputsChanged = true; sender()->setProperty("changed", QVariant(true)); EnableApplyButton(true); } } void OBSBasicSettings::AudioChanged() { if (!loading) { audioChanged = true; sender()->setProperty("changed", QVariant(true)); EnableApplyButton(true); } } void OBSBasicSettings::AudioChangedRestart() { if (!loading) { audioChanged = true; ui->audioMsg->setText(QTStr("Basic.Settings.ProgramRestart")); sender()->setProperty("changed", QVariant(true)); EnableApplyButton(true); } } void OBSBasicSettings::VideoChangedRestart() { if (!loading) { videoChanged = true; ui->videoMsg->setText(QTStr("Basic.Settings.ProgramRestart")); sender()->setProperty("changed", QVariant(true)); EnableApplyButton(true); } } void OBSBasicSettings::AdvancedChangedRestart() { if (!loading) { advancedChanged = true; ui->advancedMsg->setText( QTStr("Basic.Settings.ProgramRestart")); sender()->setProperty("changed", QVariant(true)); EnableApplyButton(true); } } void OBSBasicSettings::VideoChangedResolution() { if (!loading && ValidResolutions(ui.get())) { videoChanged = true; sender()->setProperty("changed", QVariant(true)); EnableApplyButton(true); } } void OBSBasicSettings::VideoChanged() { if (!loading) { videoChanged = true; sender()->setProperty("changed", QVariant(true)); EnableApplyButton(true); } } void OBSBasicSettings::AdvancedChanged() { if (!loading) { advancedChanged = true; sender()->setProperty("changed", QVariant(true)); EnableApplyButton(true); } }