// SPDX-FileCopyrightText: 2002-2023 PCSX2 Dev Team // SPDX-License-Identifier: LGPL-3.0+ #include #include #include "pcsx2/SPU2/Global.h" #include "pcsx2/SPU2/spu2.h" #include "pcsx2/VMManager.h" #include "AudioSettingsWidget.h" #include "QtHost.h" #include "QtUtils.h" #include "SettingWidgetBinder.h" #include "SettingsWindow.h" static constexpr s32 DEFAULT_SYNCHRONIZATION_MODE = 0; static constexpr s32 DEFAULT_EXPANSION_MODE = 0; static constexpr s32 DEFAULT_DPL_DECODING_LEVEL = 0; static const char* DEFAULT_OUTPUT_MODULE = "cubeb"; static constexpr s32 DEFAULT_TARGET_LATENCY = 60; static constexpr s32 DEFAULT_OUTPUT_LATENCY = 20; static constexpr s32 DEFAULT_VOLUME = 100; static constexpr s32 DEFAULT_SOUNDTOUCH_SEQUENCE_LENGTH = 30; static constexpr s32 DEFAULT_SOUNDTOUCH_SEEK_WINDOW = 20; static constexpr s32 DEFAULT_SOUNDTOUCH_OVERLAP = 10; AudioSettingsWidget::AudioSettingsWidget(SettingsWindow* dialog, QWidget* parent) : QWidget(parent) , m_dialog(dialog) { SettingsInterface* sif = dialog->getSettingsInterface(); m_ui.setupUi(this); populateOutputModules(); SettingWidgetBinder::BindWidgetToIntSetting(sif, m_ui.syncMode, "SPU2/Output", "SynchMode", DEFAULT_SYNCHRONIZATION_MODE); SettingWidgetBinder::BindWidgetToIntSetting(sif, m_ui.expansionMode, "SPU2/Output", "SpeakerConfiguration", DEFAULT_EXPANSION_MODE); SettingWidgetBinder::BindWidgetToIntSetting(sif, m_ui.dplLevel, "SPU2/Output", "DplDecodingLevel", DEFAULT_DPL_DECODING_LEVEL); connect(m_ui.syncMode, QOverload::of(&QComboBox::currentIndexChanged), this, &AudioSettingsWidget::updateTargetLatencyRange); connect(m_ui.expansionMode, QOverload::of(&QComboBox::currentIndexChanged), this, &AudioSettingsWidget::expansionModeChanged); updateTargetLatencyRange(); expansionModeChanged(); SettingWidgetBinder::BindWidgetToStringSetting(sif, m_ui.outputModule, "SPU2/Output", "OutputModule", DEFAULT_OUTPUT_MODULE); SettingWidgetBinder::BindSliderToIntSetting( //: Measuring unit that will appear after the number selected in its option. Adapt the space depending on your language's rules. sif, m_ui.targetLatency, m_ui.targetLatencyLabel, tr(" ms"), "SPU2/Output", "Latency", DEFAULT_TARGET_LATENCY); SettingWidgetBinder::BindSliderToIntSetting( sif, m_ui.outputLatency, m_ui.outputLatencyLabel, tr(" ms"), "SPU2/Output", "OutputLatency", DEFAULT_OUTPUT_LATENCY); SettingWidgetBinder::BindWidgetToBoolSetting(sif, m_ui.outputLatencyMinimal, "SPU2/Output", "OutputLatencyMinimal", false); connect(m_ui.outputModule, &QComboBox::currentIndexChanged, this, &AudioSettingsWidget::outputModuleChanged); connect(m_ui.backend, &QComboBox::currentIndexChanged, this, &AudioSettingsWidget::outputBackendChanged); connect(m_ui.targetLatency, &QSlider::valueChanged, this, &AudioSettingsWidget::updateLatencyLabels); connect(m_ui.outputLatency, &QSlider::valueChanged, this, &AudioSettingsWidget::updateLatencyLabels); connect(m_ui.outputLatencyMinimal, &QCheckBox::stateChanged, this, &AudioSettingsWidget::updateLatencyLabels); connect(m_ui.outputLatencyMinimal, &QCheckBox::stateChanged, this, &AudioSettingsWidget::onMinimalOutputLatencyStateChanged); outputModuleChanged(); m_ui.volume->setValue(m_dialog->getEffectiveIntValue("SPU2/Mixing", "FinalVolume", DEFAULT_VOLUME)); connect(m_ui.volume, &QSlider::valueChanged, this, &AudioSettingsWidget::volumeChanged); updateVolumeLabel(); if (dialog->isPerGameSettings()) { connect(m_ui.volume, &QSlider::customContextMenuRequested, this, &AudioSettingsWidget::volumeContextMenuRequested); m_ui.volume->setContextMenuPolicy(Qt::CustomContextMenu); if (sif->ContainsValue("SPU2/Mixing", "FinalVolume")) { QFont bold_font(m_ui.volume->font()); bold_font.setBold(true); m_ui.volumeLabel->setFont(bold_font); } } SettingWidgetBinder::BindSliderToIntSetting(sif, m_ui.sequenceLength, m_ui.sequenceLengthLabel, tr(" ms"), "Soundtouch", "SequenceLengthMS", DEFAULT_SOUNDTOUCH_SEQUENCE_LENGTH); SettingWidgetBinder::BindSliderToIntSetting( sif, m_ui.seekWindowSize, m_ui.seekWindowSizeLabel, tr(" ms"), "Soundtouch", "SeekWindowMS", DEFAULT_SOUNDTOUCH_SEEK_WINDOW); SettingWidgetBinder::BindSliderToIntSetting( sif, m_ui.overlap, m_ui.overlapLabel, tr(" ms"), "Soundtouch", "OverlapMS", DEFAULT_SOUNDTOUCH_OVERLAP); connect(m_ui.resetTimestretchDefaults, &QPushButton::clicked, this, &AudioSettingsWidget::resetTimestretchDefaults); m_ui.label_3b->setVisible(false); m_ui.dplLevel->setVisible(false); onMinimalOutputLatencyStateChanged(); updateLatencyLabels(); dialog->registerWidgetHelp(m_ui.syncMode, tr("Synchronization"), tr("TimeStretch (Recommended)"), tr("When running outside of 100% speed, adjusts the tempo on audio instead of dropping frames. Produces much nicer fast-forward/slowdown audio.")); dialog->registerWidgetHelp(m_ui.expansionMode, tr("Expansion"), tr("Stereo (None, Default)"), tr("Determines how the stereo output from the emulated system is upmixed into a greater number of the output speakers.")); //: Cubeb is an audio engine name. Leave as-is. dialog->registerWidgetHelp(m_ui.outputModule, tr("Output Module"), tr("Cubeb (Cross-platform)"), tr("Selects the library to be used for audio output.")); dialog->registerWidgetHelp(m_ui.backend, tr("Output Backend"), tr("Default"), tr("When the sound output module supports multiple audio backends, determines the API to be used for audio output to the system.")); dialog->registerWidgetHelp(m_ui.outputDevice, tr("Output Device"), tr("Default"), tr("Determines which audio device to output the sound to.")); dialog->registerWidgetHelp(m_ui.targetLatency, tr("Target Latency"), tr("60 ms"), tr("Determines the buffer size which the time stretcher will try to keep filled. It effectively selects the average latency, as " "audio will be stretched/shrunk to keep the buffer size within check.")); dialog->registerWidgetHelp(m_ui.outputLatency, tr("Output Latency"), tr("20 ms"), tr("Determines the latency from the buffer to the host audio output. This can be set lower than the target latency to reduce audio " "delay.")); dialog->registerWidgetHelp(m_ui.sequenceLength, tr("Sequence Length"), tr("30 ms"), tr("This is the default length of a single processing sequence which determines how the original sound is chopped in the time-stretch algorithm. " "Larger values mean fewer sequences are used in processing. In principle a larger value sounds better when slowing down the tempo, but worse when increasing the tempo.")); //: Seek Window: the region of samples (window) the audio stretching algorithm is allowed to search. dialog->registerWidgetHelp(m_ui.seekWindowSize, tr("Seek Window Size"), tr("20 ms"), tr("The seeking window is for the algorithm that seeks the best possible overlapping location. " "This determines from how wide a sample window the algorithm can use to find an optimal mixing location when the sound sequences are to be linked back together.")); dialog->registerWidgetHelp(m_ui.overlap, tr("Overlap"), tr("10 ms"), tr("When the sound sequences are mixed back together to form again a continuous sound stream, this parameter defines how much the ends of the consecutive sequences will overlap with each other.")); dialog->registerWidgetHelp(m_ui.volume, tr("Volume"), tr("100%"), tr("Pre-applies a volume modifier to the game's audio output before forwarding it to your computer.")); } AudioSettingsWidget::~AudioSettingsWidget() = default; void AudioSettingsWidget::expansionModeChanged() { const bool expansion51 = m_dialog->getEffectiveIntValue("SPU2/Output", "SpeakerConfiguration", 0) == 2; m_ui.dplLevel->setDisabled(!expansion51); } void AudioSettingsWidget::populateOutputModules() { for (const SndOutModule* mod : GetSndOutModules()) m_ui.outputModule->addItem(qApp->translate("SPU2", mod->GetDisplayName()), QString::fromUtf8(mod->GetIdent())); } void AudioSettingsWidget::outputModuleChanged() { const std::string module_name(m_dialog->getEffectiveStringValue("SPU2/Output", "OutputModule", DEFAULT_OUTPUT_MODULE)); const char* const* backend_names = GetOutputModuleBackends(module_name.c_str()); const std::string backend_name(m_dialog->getEffectiveStringValue("SPU2/Output", "BackendName", "")); QSignalBlocker sb(m_ui.backend); m_ui.backend->clear(); if (m_dialog->isPerGameSettings()) { const QString global_backend(QString::fromStdString(Host::GetStringSettingValue("SPU2/Output", "BackendName", ""))); m_ui.backend->addItem(tr("Use Global Setting [%1]").arg(global_backend.isEmpty() ? tr("Default") : global_backend)); } m_ui.backend->setEnabled(backend_names != nullptr); m_ui.backend->addItem(tr("Default")); if (!backend_names || backend_name.empty()) m_ui.backend->setCurrentIndex(0); if (backend_names) { for (u32 i = 0; backend_names[i] != nullptr; i++) { const int index = m_ui.backend->count(); m_ui.backend->addItem(QString::fromUtf8(backend_names[i])); if (backend_name == backend_names[i]) m_ui.backend->setCurrentIndex(index); } } updateDevices(); } void AudioSettingsWidget::outputBackendChanged() { int index = m_ui.backend->currentIndex(); if (m_dialog->isPerGameSettings()) { if (index == 0) { m_dialog->setStringSettingValue("SPU2/Output", "BackendName", std::nullopt); return; } index--; } if (index == 0) m_dialog->setStringSettingValue("SPU2/Output", "BackendName", ""); else m_dialog->setStringSettingValue("SPU2/Output", "BackendName", m_ui.backend->currentText().toUtf8().constData()); updateDevices(); } void AudioSettingsWidget::updateDevices() { const std::string module_name(m_dialog->getEffectiveStringValue("SPU2/Output", "OutputModule", DEFAULT_OUTPUT_MODULE)); const std::string backend_name(m_dialog->getEffectiveStringValue("SPU2/Output", "BackendName", "")); m_ui.outputDevice->disconnect(); m_ui.outputDevice->clear(); m_output_device_latency = 0; std::vector devices(GetOutputDeviceList(module_name.c_str(), backend_name.c_str())); if (devices.empty()) { m_ui.outputDevice->addItem(tr("Default")); m_ui.outputDevice->setEnabled(false); } else { const std::string current_device(m_dialog->getEffectiveStringValue("SPU2/Output", "DeviceName", "")); m_ui.outputDevice->setEnabled(true); for (const SndOutDeviceInfo& devi : devices) { m_ui.outputDevice->addItem(QString::fromStdString(devi.display_name), QString::fromStdString(devi.name)); if (devi.name == current_device) m_output_device_latency = devi.minimum_latency_frames; } SettingWidgetBinder::BindWidgetToStringSetting( m_dialog->getSettingsInterface(), m_ui.outputDevice, "SPU2/Output", "DeviceName", std::move(devices.front().name)); } } void AudioSettingsWidget::volumeChanged(int value) { // Nasty, but needed so we don't do a full settings apply and lag while dragging. if (SettingsInterface* sif = m_dialog->getSettingsInterface()) { if (!m_ui.volumeLabel->font().bold()) { QFont bold_font(m_ui.volumeLabel->font()); bold_font.setBold(true); m_ui.volumeLabel->setFont(bold_font); } sif->SetIntValue("SPU2/Mixing", "FinalVolume", value); sif->Save(); // There's two separate interfaces - one we're editing, and the active one. // We need to reload the latter. g_emu_thread->reloadGameSettings(); } else { Host::SetBaseIntSettingValue("SPU2/Mixing", "FinalVolume", value); Host::CommitBaseSettingChanges(); // Push through to emu thread since we're not applying. if (QtHost::IsVMValid()) { Host::RunOnCPUThread([]() { if (!VMManager::HasValidVM()) return; EmuConfig.SPU2.FinalVolume = Host::GetIntSettingValue("SPU2/Mixing", "FinalVolume", DEFAULT_VOLUME); SPU2::SetOutputVolume(EmuConfig.SPU2.FinalVolume); }); } } updateVolumeLabel(); } void AudioSettingsWidget::volumeContextMenuRequested(const QPoint& pt) { QMenu menu(m_ui.volume); m_ui.volume->connect(menu.addAction(qApp->translate("SettingWidgetBinder", "Reset")), &QAction::triggered, this, [this]() { const s32 global_value = Host::GetBaseIntSettingValue("SPU2/Mixing", "FinalVolume", DEFAULT_VOLUME); { QSignalBlocker sb(m_ui.volume); m_ui.volume->setValue(global_value); updateVolumeLabel(); } if (m_ui.volumeLabel->font().bold()) { QFont orig_font(m_ui.volumeLabel->font()); orig_font.setBold(false); m_ui.volumeLabel->setFont(orig_font); } SettingsInterface* sif = m_dialog->getSettingsInterface(); if (sif->ContainsValue("SPU2/Mixing", "FinalVolume")) { sif->DeleteValue("SPU2/Mixing", "FinalVolume"); sif->Save(); g_emu_thread->reloadGameSettings(); } }); menu.exec(m_ui.volume->mapToGlobal(pt)); } void AudioSettingsWidget::updateVolumeLabel() { //: Variable value that indicates a percentage. Preserve the %1 variable, adapt the latter % (and/or any possible spaces) to your language's ruleset. m_ui.volumeLabel->setText(tr("%1%").arg(m_ui.volume->value())); } void AudioSettingsWidget::updateTargetLatencyRange() { const Pcsx2Config::SPU2Options::SynchronizationMode sync_mode = static_cast( m_dialog->getIntValue("SPU2/Output", "SynchMode", DEFAULT_SYNCHRONIZATION_MODE).value_or(DEFAULT_SYNCHRONIZATION_MODE)); m_ui.targetLatency->setMinimum((sync_mode == Pcsx2Config::SPU2Options::SynchronizationMode::TimeStretch) ? Pcsx2Config::SPU2Options::MIN_LATENCY_TIMESTRETCH : Pcsx2Config::SPU2Options::MIN_LATENCY); m_ui.targetLatency->setMaximum(Pcsx2Config::SPU2Options::MAX_LATENCY); } void AudioSettingsWidget::updateLatencyLabels() { const bool minimal_output = m_dialog->getEffectiveBoolValue("SPU2/Output", "OutputLatencyMinimal", false); //: Preserve the %1 variable, adapt the latter ms (and/or any possible spaces in between) to your language's ruleset. m_ui.outputLatencyLabel->setText(minimal_output ? tr("N/A") : tr("%1 ms").arg(m_ui.outputLatency->value())); const u32 output_latency_ms = minimal_output ? (((m_output_device_latency * 1000u) + 47999u) / 48000u) : static_cast(m_ui.outputLatency->value()); const u32 buffer_ms = static_cast(m_ui.targetLatency->value()); if (output_latency_ms > 0) { m_ui.latencySummary->setText(tr("Average Latency: %1 ms (%2 ms buffer + %3 ms output)") .arg(buffer_ms + output_latency_ms) .arg(buffer_ms) .arg(output_latency_ms)); } else { m_ui.latencySummary->setText(tr("Average Latency: %1 ms (minimum output latency unknown)").arg(buffer_ms)); } } void AudioSettingsWidget::onMinimalOutputLatencyStateChanged() { m_ui.outputLatency->setEnabled(!m_dialog->getEffectiveBoolValue("SPU2/Output", "OutputLatencyMinimal", false)); } void AudioSettingsWidget::resetTimestretchDefaults() { m_ui.sequenceLength->setValue(DEFAULT_SOUNDTOUCH_SEQUENCE_LENGTH); m_ui.seekWindowSize->setValue(DEFAULT_SOUNDTOUCH_SEEK_WINDOW); m_ui.overlap->setValue(DEFAULT_SOUNDTOUCH_OVERLAP); }