diff --git a/Source/Core/DolphinQt2/CMakeLists.txt b/Source/Core/DolphinQt2/CMakeLists.txt index 63eefe3938..16ddc53540 100644 --- a/Source/Core/DolphinQt2/CMakeLists.txt +++ b/Source/Core/DolphinQt2/CMakeLists.txt @@ -31,8 +31,10 @@ set(SRCS Translation.cpp WiiUpdate.cpp WiiUpdate.h + Config/CheatWarningWidget.cpp Config/ControllersWindow.cpp Config/FilesystemWidget.cpp + Config/GeckoCodeWidget.cpp Config/Graphics/AdvancedWidget.cpp Config/Graphics/EnhancementsWidget.cpp Config/Graphics/GeneralWidget.cpp diff --git a/Source/Core/DolphinQt2/Config/CheatWarningWidget.cpp b/Source/Core/DolphinQt2/Config/CheatWarningWidget.cpp new file mode 100644 index 0000000000..85acbfc9bb --- /dev/null +++ b/Source/Core/DolphinQt2/Config/CheatWarningWidget.cpp @@ -0,0 +1,86 @@ +// Copyright 2017 Dolphin Emulator Project +// Licensed under GPLv2+ +// Refer to the license.txt file included. + +#include "DolphinQt2/Config/CheatWarningWidget.h" + +#include +#include +#include +#include +#include + +#include + +#include "Core/ConfigManager.h" +#include "Core/Core.h" +#include "DolphinQt2/Settings.h" + +CheatWarningWidget::CheatWarningWidget(const std::string& game_id) : m_game_id(game_id) +{ + CreateWidgets(); + ConnectWidgets(); + + connect(&Settings::Instance(), &Settings::EnableCheatsChanged, + [this] { Update(Core::IsRunning()); }); + connect(&Settings::Instance(), &Settings::EmulationStateChanged, [this](Core::State state) { + std::cout << (state == Core::State::Running) << std::endl; + Update(state == Core::State::Running); + }); + + Update(Core::IsRunning()); +} + +void CheatWarningWidget::CreateWidgets() +{ + auto* icon = new QLabel; + + const auto size = 1.5 * QFontMetrics(font()).height(); + + QPixmap warning_icon = style()->standardIcon(QStyle::SP_MessageBoxWarning).pixmap(size, size); + + icon->setPixmap(warning_icon); + + m_text = new QLabel(); + m_config_button = new QPushButton(tr("Configure Dolphin")); + + m_config_button->setHidden(true); + + auto* layout = new QHBoxLayout; + + layout->addWidget(icon); + layout->addWidget(m_text, 1); + layout->addWidget(m_config_button); + + layout->setContentsMargins(0, 0, 0, 0); + + setLayout(layout); +} + +void CheatWarningWidget::Update(bool running) +{ + bool hide_widget = true; + bool hide_config_button = true; + + if (running && SConfig::GetInstance().GetGameID() == m_game_id) + { + hide_widget = false; + m_text->setText(tr("Changing cheats will only take effect when the game is restarted.")); + } + + if (!Settings::Instance().GetCheatsEnabled()) + { + hide_widget = false; + hide_config_button = false; + m_text->setText(tr("Dolphin's cheat system is currently disabled.")); + } + + setHidden(hide_widget); + m_config_button->setHidden(hide_config_button); +} + +void CheatWarningWidget::ConnectWidgets() +{ + connect(m_config_button, &QPushButton::pressed, this, + &CheatWarningWidget::OpenCheatEnableSettings); +} diff --git a/Source/Core/DolphinQt2/Config/CheatWarningWidget.h b/Source/Core/DolphinQt2/Config/CheatWarningWidget.h new file mode 100644 index 0000000000..25e93e3b26 --- /dev/null +++ b/Source/Core/DolphinQt2/Config/CheatWarningWidget.h @@ -0,0 +1,32 @@ +// Copyright 2017 Dolphin Emulator Project +// Licensed under GPLv2+ +// Refer to the license.txt file included. + +#pragma once + +#include + +#include + +class QLabel; +class QPushButton; + +class CheatWarningWidget : public QWidget +{ + Q_OBJECT +public: + explicit CheatWarningWidget(const std::string& game_id); + +signals: + void OpenCheatEnableSettings(); + +private: + void CreateWidgets(); + void ConnectWidgets(); + + void Update(bool running); + + QLabel* m_text; + QPushButton* m_config_button; + const std::string m_game_id; +}; diff --git a/Source/Core/DolphinQt2/Config/GeckoCodeWidget.cpp b/Source/Core/DolphinQt2/Config/GeckoCodeWidget.cpp new file mode 100644 index 0000000000..ddb31b138b --- /dev/null +++ b/Source/Core/DolphinQt2/Config/GeckoCodeWidget.cpp @@ -0,0 +1,207 @@ +// Copyright 2017 Dolphin Emulator Project +// Licensed under GPLv2+ +// Refer to the license.txt file included. + +#include "DolphinQt2/Config/GeckoCodeWidget.h" + +#include +#include +#include +#include +#include +#include +#include +#include + +#include "Common/FileUtil.h" +#include "Common/IniFile.h" +#include "Core/ConfigManager.h" +#include "Core/GeckoCodeConfig.h" +#include "DolphinQt2/Config/CheatWarningWidget.h" +#include "DolphinQt2/GameList/GameFile.h" + +GeckoCodeWidget::GeckoCodeWidget(const GameFile& game) + : m_game(game), m_game_id(game.GetGameID().toStdString()), m_game_revision(game.GetRevision()) +{ + CreateWidgets(); + ConnectWidgets(); + + IniFile game_ini_local; + + // We don't use LoadLocalGameIni() here because user cheat codes that are installed via the UI + // will always be stored in GS/${GAMEID}.ini + game_ini_local.Load(File::GetUserPath(D_GAMESETTINGS_IDX) + m_game_id + ".ini"); + + IniFile game_ini_default = SConfig::GetInstance().LoadDefaultGameIni(m_game_id, m_game_revision); + m_gecko_codes = Gecko::LoadCodes(game_ini_default, game_ini_local); + + UpdateList(); +} + +void GeckoCodeWidget::CreateWidgets() +{ + m_warning = new CheatWarningWidget(m_game_id); + m_code_list = new QListWidget; + m_name_label = new QLabel; + m_creator_label = new QLabel; + + QFont monospace(QFontDatabase::systemFont(QFontDatabase::FixedFont).family()); + + const auto line_height = QFontMetrics(font()).lineSpacing(); + + m_code_description = new QTextEdit; + m_code_description->setFont(monospace); + m_code_description->setReadOnly(true); + m_code_description->setFixedHeight(line_height * 5); + + m_code_view = new QTextEdit; + m_code_view->setFont(monospace); + m_code_view->setReadOnly(true); + m_code_view->setFixedHeight(line_height * 10); + + m_download_codes = new QPushButton(tr("Download Codes (WiiRD Database)")); + m_download_codes->setEnabled(!m_game_id.empty()); + m_download_codes->setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Fixed); + + auto* layout = new QVBoxLayout; + + layout->addWidget(m_warning); + layout->addWidget(m_code_list); + + auto* info_layout = new QFormLayout; + + info_layout->addRow(tr("Name:"), m_name_label); + info_layout->addRow(tr("Creator:"), m_creator_label); + info_layout->addRow(tr("Description:"), static_cast(nullptr)); + + info_layout->setFormAlignment(Qt::AlignLeft | Qt::AlignTop); + + for (QLabel* label : {m_name_label, m_creator_label}) + { + label->setTextInteractionFlags(Qt::TextSelectableByMouse); + label->setCursor(Qt::IBeamCursor); + } + + layout->addLayout(info_layout); + layout->addWidget(m_code_description); + layout->addWidget(m_code_view); + layout->addWidget(m_download_codes, 0, Qt::AlignRight); + + setLayout(layout); +} + +void GeckoCodeWidget::ConnectWidgets() +{ + connect(m_code_list, &QListWidget::itemSelectionChanged, this, + &GeckoCodeWidget::OnSelectionChanged); + connect(m_code_list, &QListWidget::itemChanged, this, &GeckoCodeWidget::OnItemChanged); + + connect(m_download_codes, &QPushButton::pressed, this, &GeckoCodeWidget::DownloadCodes); + + connect(m_warning, &CheatWarningWidget::OpenCheatEnableSettings, this, + &GeckoCodeWidget::OpenGeneralSettings); +} + +void GeckoCodeWidget::OnSelectionChanged() +{ + auto items = m_code_list->selectedItems(); + + if (items.empty()) + return; + + auto selected = items[0]; + + const auto& code = m_gecko_codes[m_code_list->row(selected)]; + + m_name_label->setText(QString::fromStdString(code.name)); + m_creator_label->setText(QString::fromStdString(code.creator)); + + m_code_description->clear(); + + if (code.notes.empty()) + m_code_description->append(tr("N/A")); + + for (const auto& line : code.notes) + m_code_description->append(QString::fromStdString(line)); + + m_code_view->clear(); + + for (const auto& c : code.codes) + m_code_view->append(QStringLiteral("%1 %2") + .arg(c.address, 8, 16, QLatin1Char('0')) + .arg(c.data, 8, 16, QLatin1Char('0'))); +} + +void GeckoCodeWidget::OnItemChanged(QListWidgetItem* item) +{ + m_gecko_codes[m_code_list->row(item)].enabled = (item->checkState() == Qt::Checked); + + SaveCodes(); +} + +void GeckoCodeWidget::SaveCodes() +{ + IniFile game_ini_local; + game_ini_local.Load(File::GetUserPath(D_GAMESETTINGS_IDX) + m_game_id + ".ini"); + Gecko::SaveCodes(game_ini_local, m_gecko_codes); + + game_ini_local.Save(File::GetUserPath(D_GAMESETTINGS_IDX) + m_game_id + ".ini"); +} + +void GeckoCodeWidget::UpdateList() +{ + m_code_list->clear(); + + for (size_t i = 0; i < m_gecko_codes.size(); i++) + { + const auto& code = m_gecko_codes[i]; + + auto* item = new QListWidgetItem(QString::fromStdString(code.name) + .replace(QStringLiteral("<"), QStringLiteral("<")) + .replace(QStringLiteral(">"), QStringLiteral(">"))); + + item->setFlags(Qt::ItemIsEnabled | Qt::ItemIsSelectable | Qt::ItemIsUserCheckable); + item->setCheckState(code.enabled ? Qt::Checked : Qt::Unchecked); + + m_code_list->addItem(item); + } +} + +void GeckoCodeWidget::DownloadCodes() +{ + bool success; + + std::vector codes = Gecko::DownloadCodes(m_game_id, &success); + + if (!success) + { + QMessageBox::critical(this, tr("Error"), tr("Failed to download codes.")); + return; + } + + if (codes.empty()) + { + QMessageBox::critical(this, tr("Error"), tr("File contained no codes.")); + return; + } + + size_t added_count = 0; + + for (const auto& code : codes) + { + auto it = std::find(m_gecko_codes.begin(), m_gecko_codes.end(), code); + + if (it == m_gecko_codes.end()) + { + m_gecko_codes.push_back(code); + added_count++; + } + } + + UpdateList(); + SaveCodes(); + + QMessageBox::information(this, tr("Download complete"), + tr("Downloaded %1 codes. (added %2)") + .arg(QString::number(codes.size()), QString::number(added_count))); +} diff --git a/Source/Core/DolphinQt2/Config/GeckoCodeWidget.h b/Source/Core/DolphinQt2/Config/GeckoCodeWidget.h new file mode 100644 index 0000000000..877090b367 --- /dev/null +++ b/Source/Core/DolphinQt2/Config/GeckoCodeWidget.h @@ -0,0 +1,54 @@ +// Copyright 2017 Dolphin Emulator Project +// Licensed under GPLv2+ +// Refer to the license.txt file included. + +#pragma once + +#include + +#include +#include + +#include "Common/CommonTypes.h" +#include "Core/GeckoCode.h" + +class CheatWarningWidget; +class GameFile; +class QLabel; +class QListWidget; +class QListWidgetItem; +class QTextEdit; +class QPushButton; + +class GeckoCodeWidget : public QWidget +{ + Q_OBJECT +public: + explicit GeckoCodeWidget(const GameFile& game); + +signals: + void OpenGeneralSettings(); + +private: + void OnSelectionChanged(); + void OnItemChanged(QListWidgetItem* item); + + void CreateWidgets(); + void ConnectWidgets(); + void UpdateList(); + void DownloadCodes(); + void SaveCodes(); + + const GameFile& m_game; + std::string m_game_id; + u8 m_game_revision; + + CheatWarningWidget* m_warning; + QListWidget* m_code_list; + QLabel* m_name_label; + QLabel* m_creator_label; + QTextEdit* m_code_description; + QTextEdit* m_code_view; + QPushButton* m_download_codes; + std::vector m_gecko_codes; +}; diff --git a/Source/Core/DolphinQt2/Config/PropertiesDialog.cpp b/Source/Core/DolphinQt2/Config/PropertiesDialog.cpp index 3ab91deff8..d8c464f07c 100644 --- a/Source/Core/DolphinQt2/Config/PropertiesDialog.cpp +++ b/Source/Core/DolphinQt2/Config/PropertiesDialog.cpp @@ -7,6 +7,7 @@ #include #include "DolphinQt2/Config/FilesystemWidget.h" +#include "DolphinQt2/Config/GeckoCodeWidget.h" #include "DolphinQt2/Config/InfoWidget.h" #include "DolphinQt2/Config/PropertiesDialog.h" @@ -18,7 +19,14 @@ PropertiesDialog::PropertiesDialog(QWidget* parent, const GameFile& game) : QDia QTabWidget* tab_widget = new QTabWidget(this); InfoWidget* info = new InfoWidget(game); + + GeckoCodeWidget* gecko = new GeckoCodeWidget(game); + + connect(gecko, &GeckoCodeWidget::OpenGeneralSettings, this, + &PropertiesDialog::OpenGeneralSettings); + tab_widget->addTab(info, tr("Info")); + tab_widget->addTab(gecko, tr("Gecko Codes")); if (DiscIO::IsDisc(game.GetPlatformID())) { diff --git a/Source/Core/DolphinQt2/Config/PropertiesDialog.h b/Source/Core/DolphinQt2/Config/PropertiesDialog.h index f640f4e536..85f1ef64f5 100644 --- a/Source/Core/DolphinQt2/Config/PropertiesDialog.h +++ b/Source/Core/DolphinQt2/Config/PropertiesDialog.h @@ -13,4 +13,7 @@ class PropertiesDialog final : public QDialog Q_OBJECT public: explicit PropertiesDialog(QWidget* parent, const GameFile& game); + +signals: + void OpenGeneralSettings(); }; diff --git a/Source/Core/DolphinQt2/Config/SettingsWindow.cpp b/Source/Core/DolphinQt2/Config/SettingsWindow.cpp index 7bc5071527..59c9d33fd3 100644 --- a/Source/Core/DolphinQt2/Config/SettingsWindow.cpp +++ b/Source/Core/DolphinQt2/Config/SettingsWindow.cpp @@ -39,7 +39,7 @@ SettingsWindow::SettingsWindow(QWidget* parent) : QDialog(parent) m_tabs = new ListTabWidget(); layout->addWidget(m_tabs); - AddTab(m_tabs, tr("General"), new GeneralPane(), "config"); + m_general_pane_index = AddTab(m_tabs, tr("General"), new GeneralPane(), "config"); AddTab(m_tabs, tr("Interface"), new InterfacePane(), "browse"); m_audio_pane_index = AddTab(m_tabs, tr("Audio"), new AudioPane(), "play"); AddTab(m_tabs, tr("Paths"), new PathPane(), "browse"); @@ -57,3 +57,8 @@ void SettingsWindow::SelectAudioPane() { m_tabs->setCurrentIndex(m_audio_pane_index); } + +void SettingsWindow::SelectGeneralPane() +{ + m_tabs->setCurrentIndex(m_general_pane_index); +} diff --git a/Source/Core/DolphinQt2/Config/SettingsWindow.h b/Source/Core/DolphinQt2/Config/SettingsWindow.h index f61b41bfe6..50efd2ad8f 100644 --- a/Source/Core/DolphinQt2/Config/SettingsWindow.h +++ b/Source/Core/DolphinQt2/Config/SettingsWindow.h @@ -13,9 +13,11 @@ class SettingsWindow final : public QDialog Q_OBJECT public: explicit SettingsWindow(QWidget* parent = nullptr); + void SelectGeneralPane(); void SelectAudioPane(); private: ListTabWidget* m_tabs; int m_audio_pane_index = -1; + int m_general_pane_index = -1; }; diff --git a/Source/Core/DolphinQt2/DolphinQt2.vcxproj b/Source/Core/DolphinQt2/DolphinQt2.vcxproj index e6fc2256c9..8d55c59e73 100644 --- a/Source/Core/DolphinQt2/DolphinQt2.vcxproj +++ b/Source/Core/DolphinQt2/DolphinQt2.vcxproj @@ -60,8 +60,10 @@ + + @@ -113,14 +115,16 @@ - + + + @@ -156,8 +160,10 @@ + + @@ -229,8 +235,8 @@ - - + + diff --git a/Source/Core/DolphinQt2/GameList/GameList.cpp b/Source/Core/DolphinQt2/GameList/GameList.cpp index 072775165f..5e87376448 100644 --- a/Source/Core/DolphinQt2/GameList/GameList.cpp +++ b/Source/Core/DolphinQt2/GameList/GameList.cpp @@ -241,6 +241,9 @@ void GameList::ShowContextMenu(const QPoint&) void GameList::OpenProperties() { PropertiesDialog* properties = new PropertiesDialog(this, *GetSelectedGame()); + + connect(properties, &PropertiesDialog::OpenGeneralSettings, this, &GameList::OpenGeneralSettings); + properties->show(); } diff --git a/Source/Core/DolphinQt2/GameList/GameList.h b/Source/Core/DolphinQt2/GameList/GameList.h index 75103a42fd..fb564cde92 100644 --- a/Source/Core/DolphinQt2/GameList/GameList.h +++ b/Source/Core/DolphinQt2/GameList/GameList.h @@ -31,6 +31,7 @@ signals: void GameSelected(); void NetPlayHost(const QString& game_id); void SelectionChanged(QSharedPointer game_file); + void OpenGeneralSettings(); private: void ShowContextMenu(const QPoint&); diff --git a/Source/Core/DolphinQt2/MainWindow.cpp b/Source/Core/DolphinQt2/MainWindow.cpp index b8a7b0a871..fbb508f195 100644 --- a/Source/Core/DolphinQt2/MainWindow.cpp +++ b/Source/Core/DolphinQt2/MainWindow.cpp @@ -279,6 +279,8 @@ void MainWindow::ConnectGameList() { connect(m_game_list, &GameList::GameSelected, this, &MainWindow::Play); connect(m_game_list, &GameList::NetPlayHost, this, &MainWindow::NetPlayHost); + + connect(m_game_list, &GameList::OpenGeneralSettings, this, &MainWindow::ShowGeneralWindow); } void MainWindow::ConnectRenderWidget() @@ -550,6 +552,12 @@ void MainWindow::ShowAudioWindow() ShowSettingsWindow(); } +void MainWindow::ShowGeneralWindow() +{ + m_settings_window->SelectGeneralPane(); + ShowSettingsWindow(); +} + void MainWindow::ShowAboutDialog() { AboutDialog about{this}; diff --git a/Source/Core/DolphinQt2/MainWindow.h b/Source/Core/DolphinQt2/MainWindow.h index 84a2f04de6..93d5431334 100644 --- a/Source/Core/DolphinQt2/MainWindow.h +++ b/Source/Core/DolphinQt2/MainWindow.h @@ -91,6 +91,7 @@ private: void HideRenderWidget(); void ShowSettingsWindow(); + void ShowGeneralWindow(); void ShowAudioWindow(); void ShowControllersWindow(); void ShowGraphicsWindow(); diff --git a/Source/Core/DolphinQt2/Settings.cpp b/Source/Core/DolphinQt2/Settings.cpp index 0ed961cbff..bf1a07ea03 100644 --- a/Source/Core/DolphinQt2/Settings.cpp +++ b/Source/Core/DolphinQt2/Settings.cpp @@ -209,3 +209,17 @@ void Settings::ResetNetPlayServer(NetPlayServer* server) { m_server.reset(server); } + +bool Settings::GetCheatsEnabled() const +{ + return SConfig::GetInstance().bEnableCheats; +} + +void Settings::SetCheatsEnabled(bool enabled) +{ + if (SConfig::GetInstance().bEnableCheats != enabled) + { + SConfig::GetInstance().bEnableCheats = enabled; + emit EnableCheatsChanged(enabled); + } +} diff --git a/Source/Core/DolphinQt2/Settings.h b/Source/Core/DolphinQt2/Settings.h index c9ecc93a2d..56be9df511 100644 --- a/Source/Core/DolphinQt2/Settings.h +++ b/Source/Core/DolphinQt2/Settings.h @@ -77,6 +77,10 @@ public: NetPlayServer* GetNetPlayServer(); void ResetNetPlayServer(NetPlayServer* server = nullptr); + // Cheats + bool GetCheatsEnabled() const; + void SetCheatsEnabled(bool enabled); + // Other GameListModel* GetGameListModel() const; @@ -91,6 +95,7 @@ signals: void NANDRefresh(); void LogVisibilityChanged(bool visible); void LogConfigVisibilityChanged(bool visible); + void EnableCheatsChanged(bool enabled); private: std::unique_ptr m_client; diff --git a/Source/Core/DolphinQt2/Settings/GeneralPane.cpp b/Source/Core/DolphinQt2/Settings/GeneralPane.cpp index 9e092f4668..a89ecfe65d 100644 --- a/Source/Core/DolphinQt2/Settings/GeneralPane.cpp +++ b/Source/Core/DolphinQt2/Settings/GeneralPane.cpp @@ -145,7 +145,7 @@ void GeneralPane::LoadConfig() m_checkbox_enable_analytics->setChecked(SConfig::GetInstance().m_analytics_enabled); #endif m_checkbox_dualcore->setChecked(SConfig::GetInstance().bCPUThread); - m_checkbox_cheats->setChecked(SConfig::GetInstance().bEnableCheats); + m_checkbox_cheats->setChecked(Settings::Instance().GetCheatsEnabled()); int selection = qRound(SConfig::GetInstance().m_EmulationSpeed * 10); if (selection < m_combobox_speedlimit->count()) m_combobox_speedlimit->setCurrentIndex(selection); @@ -177,7 +177,7 @@ void GeneralPane::OnSaveConfig() SConfig::GetInstance().m_analytics_enabled = m_checkbox_enable_analytics->isChecked(); #endif SConfig::GetInstance().bCPUThread = m_checkbox_dualcore->isChecked(); - SConfig::GetInstance().bEnableCheats = m_checkbox_cheats->isChecked(); + Settings::Instance().SetCheatsEnabled(m_checkbox_cheats->isChecked()); SConfig::GetInstance().m_EmulationSpeed = m_combobox_speedlimit->currentIndex() * 0.1f; int engine_value = 0; if (m_radio_interpreter->isChecked())