123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473 |
- // Copyright 2017 Dolphin Emulator Project
- // SPDX-License-Identifier: GPL-2.0-or-later
- #include "DolphinQt/Settings/GeneralPane.h"
- #include <map>
- #include <QCheckBox>
- #include <QComboBox>
- #include <QFormLayout>
- #include <QGroupBox>
- #include <QLabel>
- #include <QPushButton>
- #include <QSlider>
- #include <QVBoxLayout>
- #include <QWidget>
- #include "Core/AchievementManager.h"
- #include "Core/Config/MainSettings.h"
- #include "Core/Config/UISettings.h"
- #include "Core/ConfigManager.h"
- #include "Core/Core.h"
- #include "Core/DolphinAnalytics.h"
- #include "Core/PowerPC/PowerPC.h"
- #include "Core/System.h"
- #include "DolphinQt/Config/ConfigControls/ConfigBool.h"
- #include "DolphinQt/Config/ToolTipControls/ToolTipCheckBox.h"
- #include "DolphinQt/Config/ToolTipControls/ToolTipComboBox.h"
- #include "DolphinQt/Config/ToolTipControls/ToolTipPushButton.h"
- #include "DolphinQt/QtUtils/ModalMessageBox.h"
- #include "DolphinQt/QtUtils/SetWindowDecorations.h"
- #include "DolphinQt/QtUtils/SignalBlocking.h"
- #include "DolphinQt/Settings.h"
- #include "UICommon/AutoUpdate.h"
- #ifdef USE_DISCORD_PRESENCE
- #include "UICommon/DiscordPresence.h"
- #endif
- constexpr int AUTO_UPDATE_DISABLE_INDEX = 0;
- constexpr int AUTO_UPDATE_BETA_INDEX = 1;
- constexpr int AUTO_UPDATE_DEV_INDEX = 2;
- constexpr const char* AUTO_UPDATE_DISABLE_STRING = "";
- constexpr const char* AUTO_UPDATE_BETA_STRING = "beta";
- constexpr const char* AUTO_UPDATE_DEV_STRING = "dev";
- constexpr int FALLBACK_REGION_NTSCJ_INDEX = 0;
- constexpr int FALLBACK_REGION_NTSCU_INDEX = 1;
- constexpr int FALLBACK_REGION_PAL_INDEX = 2;
- constexpr int FALLBACK_REGION_NTSCK_INDEX = 3;
- GeneralPane::GeneralPane(QWidget* parent) : QWidget(parent)
- {
- CreateLayout();
- LoadConfig();
- AddDescriptions();
- ConnectLayout();
- connect(&Settings::Instance(), &Settings::EmulationStateChanged, this,
- &GeneralPane::OnEmulationStateChanged);
- connect(&Settings::Instance(), &Settings::ConfigChanged, this, &GeneralPane::LoadConfig);
- OnEmulationStateChanged(Core::GetState(Core::System::GetInstance()));
- }
- void GeneralPane::CreateLayout()
- {
- m_main_layout = new QVBoxLayout;
- // Create layout here
- CreateBasic();
- if (AutoUpdateChecker::SystemSupportsAutoUpdates())
- CreateAutoUpdate();
- CreateFallbackRegion();
- #if defined(USE_ANALYTICS) && USE_ANALYTICS
- CreateAnalytics();
- #endif
- m_main_layout->addStretch(1);
- setLayout(m_main_layout);
- }
- void GeneralPane::OnEmulationStateChanged(Core::State state)
- {
- const bool running = state != Core::State::Uninitialized;
- m_checkbox_dualcore->setEnabled(!running);
- m_checkbox_cheats->setEnabled(!running);
- m_checkbox_override_region_settings->setEnabled(!running);
- #ifdef USE_DISCORD_PRESENCE
- m_checkbox_discord_presence->setEnabled(!running);
- #endif
- m_combobox_fallback_region->setEnabled(!running);
- }
- void GeneralPane::ConnectLayout()
- {
- connect(m_checkbox_cheats, &QCheckBox::toggled, &Settings::Instance(),
- &Settings::EnableCheatsChanged);
- #ifdef USE_DISCORD_PRESENCE
- connect(m_checkbox_discord_presence, &QCheckBox::toggled, this, &GeneralPane::OnSaveConfig);
- #endif
- if (AutoUpdateChecker::SystemSupportsAutoUpdates())
- {
- connect(m_combobox_update_track, &QComboBox::currentIndexChanged, this,
- &GeneralPane::OnSaveConfig);
- connect(&Settings::Instance(), &Settings::AutoUpdateTrackChanged, this,
- &GeneralPane::LoadConfig);
- }
- // Advanced
- connect(m_combobox_speedlimit, &QComboBox::currentIndexChanged, [this]() {
- Config::SetBaseOrCurrent(Config::MAIN_EMULATION_SPEED,
- m_combobox_speedlimit->currentIndex() * 0.1f);
- Config::Save();
- });
- connect(m_combobox_fallback_region, &QComboBox::currentIndexChanged, this,
- &GeneralPane::OnSaveConfig);
- connect(&Settings::Instance(), &Settings::FallbackRegionChanged, this, &GeneralPane::LoadConfig);
- #if defined(USE_ANALYTICS) && USE_ANALYTICS
- connect(&Settings::Instance(), &Settings::AnalyticsToggled, this, &GeneralPane::LoadConfig);
- connect(m_checkbox_enable_analytics, &QCheckBox::toggled, this, &GeneralPane::OnSaveConfig);
- connect(m_button_generate_new_identity, &QPushButton::clicked, this,
- &GeneralPane::GenerateNewIdentity);
- #endif
- }
- void GeneralPane::CreateBasic()
- {
- auto* basic_group = new QGroupBox(tr("Basic Settings"));
- auto* basic_group_layout = new QVBoxLayout;
- basic_group->setLayout(basic_group_layout);
- m_main_layout->addWidget(basic_group);
- m_checkbox_dualcore = new ConfigBool(tr("Enable Dual Core (speedhack)"), Config::MAIN_CPU_THREAD);
- basic_group_layout->addWidget(m_checkbox_dualcore);
- m_checkbox_cheats = new ConfigBool(tr("Enable Cheats"), Config::MAIN_ENABLE_CHEATS);
- basic_group_layout->addWidget(m_checkbox_cheats);
- m_checkbox_override_region_settings =
- new ConfigBool(tr("Allow Mismatched Region Settings"), Config::MAIN_OVERRIDE_REGION_SETTINGS);
- basic_group_layout->addWidget(m_checkbox_override_region_settings);
- m_checkbox_auto_disc_change =
- new ConfigBool(tr("Change Discs Automatically"), Config::MAIN_AUTO_DISC_CHANGE);
- basic_group_layout->addWidget(m_checkbox_auto_disc_change);
- #ifdef USE_DISCORD_PRESENCE
- m_checkbox_discord_presence = new ToolTipCheckBox(tr("Show Current Game on Discord"));
- basic_group_layout->addWidget(m_checkbox_discord_presence);
- #endif
- auto* speed_limit_layout = new QFormLayout;
- speed_limit_layout->setFormAlignment(Qt::AlignLeft | Qt::AlignTop);
- speed_limit_layout->setFieldGrowthPolicy(QFormLayout::AllNonFixedFieldsGrow);
- basic_group_layout->addLayout(speed_limit_layout);
- m_combobox_speedlimit = new ToolTipComboBox();
- m_combobox_speedlimit->addItem(tr("Unlimited"));
- for (int i = 10; i <= 200; i += 10) // from 10% to 200%
- {
- QString str;
- if (i != 100)
- str = QStringLiteral("%1%").arg(i);
- else
- str = tr("%1% (Normal Speed)").arg(i);
- m_combobox_speedlimit->addItem(str);
- }
- speed_limit_layout->addRow(tr("&Speed Limit:"), m_combobox_speedlimit);
- }
- void GeneralPane::CreateAutoUpdate()
- {
- auto* auto_update_group = new QGroupBox(tr("Auto Update Settings"));
- auto* auto_update_group_layout = new QFormLayout;
- auto_update_group->setLayout(auto_update_group_layout);
- m_main_layout->addWidget(auto_update_group);
- auto_update_group_layout->setFormAlignment(Qt::AlignLeft | Qt::AlignTop);
- auto_update_group_layout->setFieldGrowthPolicy(QFormLayout::AllNonFixedFieldsGrow);
- m_combobox_update_track = new ToolTipComboBox();
- auto_update_group_layout->addRow(tr("&Auto Update:"), m_combobox_update_track);
- for (const QString& option :
- {tr("Don't Update"),
- // i18n: Releases is a noun.
- tr("Releases (every few months)"), tr("Dev (multiple times a day)")})
- {
- m_combobox_update_track->addItem(option);
- }
- }
- void GeneralPane::CreateFallbackRegion()
- {
- auto* fallback_region_group = new QGroupBox(tr("Fallback Region"));
- auto* fallback_region_group_layout = new QVBoxLayout;
- fallback_region_group->setLayout(fallback_region_group_layout);
- m_main_layout->addWidget(fallback_region_group);
- auto* fallback_region_dropdown_layout = new QFormLayout;
- fallback_region_dropdown_layout->setFormAlignment(Qt::AlignLeft | Qt::AlignTop);
- fallback_region_dropdown_layout->setFieldGrowthPolicy(QFormLayout::AllNonFixedFieldsGrow);
- fallback_region_group_layout->addLayout(fallback_region_dropdown_layout);
- m_combobox_fallback_region = new ToolTipComboBox();
- fallback_region_dropdown_layout->addRow(tr("Fallback Region:"), m_combobox_fallback_region);
- for (const QString& option : {tr("NTSC-J"), tr("NTSC-U"), tr("PAL"), tr("NTSC-K")})
- m_combobox_fallback_region->addItem(option);
- }
- #if defined(USE_ANALYTICS) && USE_ANALYTICS
- void GeneralPane::CreateAnalytics()
- {
- auto* analytics_group = new QGroupBox(tr("Usage Statistics Reporting Settings"));
- auto* analytics_group_layout = new QVBoxLayout;
- analytics_group->setLayout(analytics_group_layout);
- m_main_layout->addWidget(analytics_group);
- m_checkbox_enable_analytics = new ToolTipCheckBox(tr("Enable Usage Statistics Reporting"));
- m_button_generate_new_identity = new ToolTipPushButton(tr("Generate a New Statistics Identity"));
- analytics_group_layout->addWidget(m_checkbox_enable_analytics);
- analytics_group_layout->addWidget(m_button_generate_new_identity);
- }
- #endif
- void GeneralPane::LoadConfig()
- {
- const QSignalBlocker blocker(this);
- if (AutoUpdateChecker::SystemSupportsAutoUpdates())
- {
- const auto track = Settings::Instance().GetAutoUpdateTrack().toStdString();
- // If the track doesn't match any known value, set to "beta" which is the
- // default config value on Dolphin release builds.
- if (track == AUTO_UPDATE_DISABLE_STRING)
- SignalBlocking(m_combobox_update_track)->setCurrentIndex(AUTO_UPDATE_DISABLE_INDEX);
- else if (track == AUTO_UPDATE_DEV_STRING)
- SignalBlocking(m_combobox_update_track)->setCurrentIndex(AUTO_UPDATE_DEV_INDEX);
- else
- SignalBlocking(m_combobox_update_track)->setCurrentIndex(AUTO_UPDATE_BETA_INDEX);
- }
- #if defined(USE_ANALYTICS) && USE_ANALYTICS
- SignalBlocking(m_checkbox_enable_analytics)
- ->setChecked(Settings::Instance().IsAnalyticsEnabled());
- #endif
- #ifdef USE_DISCORD_PRESENCE
- SignalBlocking(m_checkbox_discord_presence)
- ->setChecked(Config::Get(Config::MAIN_USE_DISCORD_PRESENCE));
- #endif
- int selection = qRound(Config::Get(Config::MAIN_EMULATION_SPEED) * 10);
- if (selection < m_combobox_speedlimit->count())
- SignalBlocking(m_combobox_speedlimit)->setCurrentIndex(selection);
- const auto fallback = Settings::Instance().GetFallbackRegion();
- if (fallback == DiscIO::Region::NTSC_J)
- SignalBlocking(m_combobox_fallback_region)->setCurrentIndex(FALLBACK_REGION_NTSCJ_INDEX);
- else if (fallback == DiscIO::Region::NTSC_U)
- SignalBlocking(m_combobox_fallback_region)->setCurrentIndex(FALLBACK_REGION_NTSCU_INDEX);
- else if (fallback == DiscIO::Region::PAL)
- SignalBlocking(m_combobox_fallback_region)->setCurrentIndex(FALLBACK_REGION_PAL_INDEX);
- else if (fallback == DiscIO::Region::NTSC_K)
- SignalBlocking(m_combobox_fallback_region)->setCurrentIndex(FALLBACK_REGION_NTSCK_INDEX);
- else
- SignalBlocking(m_combobox_fallback_region)->setCurrentIndex(FALLBACK_REGION_NTSCJ_INDEX);
- }
- static QString UpdateTrackFromIndex(int index)
- {
- QString value;
- switch (index)
- {
- case AUTO_UPDATE_DISABLE_INDEX:
- value = QString::fromStdString(AUTO_UPDATE_DISABLE_STRING);
- break;
- case AUTO_UPDATE_BETA_INDEX:
- value = QString::fromStdString(AUTO_UPDATE_BETA_STRING);
- break;
- case AUTO_UPDATE_DEV_INDEX:
- value = QString::fromStdString(AUTO_UPDATE_DEV_STRING);
- break;
- }
- return value;
- }
- static DiscIO::Region UpdateFallbackRegionFromIndex(int index)
- {
- DiscIO::Region value = DiscIO::Region::Unknown;
- switch (index)
- {
- case FALLBACK_REGION_NTSCJ_INDEX:
- value = DiscIO::Region::NTSC_J;
- break;
- case FALLBACK_REGION_NTSCU_INDEX:
- value = DiscIO::Region::NTSC_U;
- break;
- case FALLBACK_REGION_PAL_INDEX:
- value = DiscIO::Region::PAL;
- break;
- case FALLBACK_REGION_NTSCK_INDEX:
- value = DiscIO::Region::NTSC_K;
- break;
- default:
- value = DiscIO::Region::NTSC_J;
- }
- return value;
- }
- void GeneralPane::OnSaveConfig()
- {
- Config::ConfigChangeCallbackGuard config_guard;
- auto& settings = SConfig::GetInstance();
- if (AutoUpdateChecker::SystemSupportsAutoUpdates())
- {
- Settings::Instance().SetAutoUpdateTrack(
- UpdateTrackFromIndex(m_combobox_update_track->currentIndex()));
- }
- #ifdef USE_DISCORD_PRESENCE
- Discord::SetDiscordPresenceEnabled(m_checkbox_discord_presence->isChecked());
- #endif
- #if defined(USE_ANALYTICS) && USE_ANALYTICS
- Settings::Instance().SetAnalyticsEnabled(m_checkbox_enable_analytics->isChecked());
- DolphinAnalytics::Instance().ReloadConfig();
- #endif
- Settings::Instance().SetFallbackRegion(
- UpdateFallbackRegionFromIndex(m_combobox_fallback_region->currentIndex()));
- settings.SaveSettings();
- }
- #if defined(USE_ANALYTICS) && USE_ANALYTICS
- void GeneralPane::GenerateNewIdentity()
- {
- DolphinAnalytics::Instance().GenerateNewIdentity();
- DolphinAnalytics::Instance().ReloadConfig();
- ModalMessageBox message_box(this);
- message_box.setIcon(QMessageBox::Information);
- message_box.setWindowTitle(tr("Identity Generation"));
- message_box.setText(tr("New identity generated."));
- SetQWidgetWindowDecorations(&message_box);
- message_box.exec();
- }
- #endif
- void GeneralPane::AddDescriptions()
- {
- static constexpr char TR_DUALCORE_DESCRIPTION[] =
- QT_TR_NOOP("Separates CPU and GPU emulation work to separate threads. Reduces single-thread "
- "burden by spreading Dolphin's heaviest load across two cores, which usually "
- "improves performance. However, it can result in glitches and crashes."
- "<br><br>This setting cannot be changed while emulation is active."
- "<br><br><dolphin_emphasis>If unsure, leave this checked.</dolphin_emphasis>");
- static constexpr char TR_CHEATS_DESCRIPTION[] = QT_TR_NOOP(
- "Enables the use of AR and Gecko cheat codes which can be used to modify games' behavior. "
- "These codes can be configured with the Cheats Manager in the Tools menu."
- "<br><br>This setting cannot be changed while emulation is active."
- "<br><br><dolphin_emphasis>If unsure, leave this unchecked.</dolphin_emphasis>");
- static constexpr char TR_OVERRIDE_REGION_SETTINGS_DESCRIPTION[] =
- QT_TR_NOOP("Lets you use languages and other region-related settings that the game may not "
- "be designed for. May cause various crashes and bugs."
- "<br><br>This setting cannot be changed while emulation is active."
- "<br><br><dolphin_emphasis>If unsure, leave this unchecked.</dolphin_emphasis>");
- static constexpr char TR_AUTO_DISC_CHANGE_DESCRIPTION[] = QT_TR_NOOP(
- "Automatically changes the game disc when requested by games with two discs. This feature "
- "requires the game to be launched in one of the following ways:"
- "<br>- From the game list, with both discs being present in the game list."
- "<br>- With File > Open or the command line interface, with the paths to both discs being "
- "provided."
- "<br>- By launching an M3U file with File > Open or the command line interface."
- "<br><br><dolphin_emphasis>If unsure, leave this unchecked.</dolphin_emphasis>");
- #ifdef USE_DISCORD_PRESENCE
- static constexpr char TR_DISCORD_PRESENCE_DESCRIPTION[] =
- QT_TR_NOOP("Shows which game is active and the duration of your current play session in your "
- "Discord status."
- "<br><br>This setting cannot be changed while emulation is active."
- "<br><br><dolphin_emphasis>If unsure, leave this checked.</dolphin_emphasis>");
- #endif
- static constexpr char TR_SPEEDLIMIT_DESCRIPTION[] =
- QT_TR_NOOP("Controls how fast emulation runs relative to the original hardware."
- "<br><br>Values higher than 100% will emulate faster than the original hardware "
- "can run, if your hardware is able to keep up. Values lower than 100% will slow "
- "emulation instead. Unlimited will emulate as fast as your hardware is able to."
- "<br><br><dolphin_emphasis>If unsure, select 100%.</dolphin_emphasis>");
- static constexpr char TR_UPDATE_TRACK_DESCRIPTION[] = QT_TR_NOOP(
- "Selects which update track Dolphin uses when checking for updates at startup. If a new "
- "update is available, Dolphin will show a list of changes made since your current version "
- "and ask you if you want to update."
- "<br><br>The Dev track has the latest version of Dolphin which often updates multiple times "
- "per day. Select this track if you want the newest features and fixes."
- "<br><br>The Releases track has an update every few months. Some reasons you might prefer to "
- "use this track:"
- "<br>- You prefer using versions that have had additional testing."
- "<br>- NetPlay requires players to have the same Dolphin version, and the latest Release "
- "version will have the most players to match with."
- "<br>- You frequently use Dolphin's savestate system, which doesn't guarantee backward "
- "compatibility of savestates between Dolphin versions. If this applies to you, make sure you "
- "make an in-game save before updating (i.e. save your game in the same way you would on a "
- "physical GameCube or Wii), then load the in-game save after updating Dolphin and before "
- "making any new savestates."
- "<br><br>Selecting \"Don't Update\" will prevent Dolphin from automatically checking for "
- "updates."
- "<br><br><dolphin_emphasis>If unsure, select Releases.</dolphin_emphasis>");
- static constexpr char TR_FALLBACK_REGION_DESCRIPTION[] =
- QT_TR_NOOP("Sets the region used for titles whose region cannot be determined automatically."
- "<br><br>This setting cannot be changed while emulation is active.");
- #if defined(USE_ANALYTICS) && USE_ANALYTICS
- static constexpr char TR_ENABLE_ANALYTICS_DESCRIPTION[] = QT_TR_NOOP(
- "If selected, Dolphin can collect data on its performance, feature usage, emulated games, "
- "and configuration, as well as data on your system's hardware and operating system."
- "<br><br>No private data is ever collected. This data helps us understand how people and "
- "emulated games use Dolphin and prioritize our efforts. It also helps us identify rare "
- "configurations that are causing bugs, performance and stability issues.");
- static constexpr char TR_GENERATE_NEW_IDENTITY_DESCRIPTION[] =
- QT_TR_NOOP("Generate a new anonymous ID for your usage statistics. This will cause any "
- "future statistics to be unassociated with your previous statistics.");
- #endif
- m_checkbox_dualcore->SetDescription(tr(TR_DUALCORE_DESCRIPTION));
- m_checkbox_cheats->SetDescription(tr(TR_CHEATS_DESCRIPTION));
- m_checkbox_override_region_settings->SetDescription(tr(TR_OVERRIDE_REGION_SETTINGS_DESCRIPTION));
- m_checkbox_auto_disc_change->SetDescription(tr(TR_AUTO_DISC_CHANGE_DESCRIPTION));
- #ifdef USE_DISCORD_PRESENCE
- m_checkbox_discord_presence->SetDescription(tr(TR_DISCORD_PRESENCE_DESCRIPTION));
- #endif
- m_combobox_speedlimit->SetTitle(tr("Speed Limit"));
- m_combobox_speedlimit->SetDescription(tr(TR_SPEEDLIMIT_DESCRIPTION));
- if (AutoUpdateChecker::SystemSupportsAutoUpdates())
- {
- m_combobox_update_track->SetTitle(tr("Auto Update"));
- m_combobox_update_track->SetDescription(tr(TR_UPDATE_TRACK_DESCRIPTION));
- }
- m_combobox_fallback_region->SetTitle(tr("Fallback Region"));
- m_combobox_fallback_region->SetDescription(tr(TR_FALLBACK_REGION_DESCRIPTION));
- #if defined(USE_ANALYTICS) && USE_ANALYTICS
- m_checkbox_enable_analytics->SetDescription(tr(TR_ENABLE_ANALYTICS_DESCRIPTION));
- m_button_generate_new_identity->SetTitle(tr("Generate a New Statistics Identity"));
- m_button_generate_new_identity->SetDescription(tr(TR_GENERATE_NEW_IDENTITY_DESCRIPTION));
- #endif
- }
|