diff --git a/Source/Core/DolphinQt2/CMakeLists.txt b/Source/Core/DolphinQt2/CMakeLists.txt index 73fd776468..dae253410a 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 index b365e3048b..85acbfc9bb 100644 --- a/Source/Core/DolphinQt2/Config/CheatWarningWidget.cpp +++ b/Source/Core/DolphinQt2/Config/CheatWarningWidget.cpp @@ -10,6 +10,8 @@ #include #include +#include + #include "Core/ConfigManager.h" #include "Core/Core.h" #include "DolphinQt2/Settings.h" @@ -21,8 +23,10 @@ CheatWarningWidget::CheatWarningWidget(const std::string& game_id) : m_game_id(g connect(&Settings::Instance(), &Settings::EnableCheatsChanged, [this] { Update(Core::IsRunning()); }); - connect(this, &CheatWarningWidget::EmulationStarted, [this] { Update(true); }); - connect(this, &CheatWarningWidget::EmulationStopped, [this] { Update(false); }); + 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()); } diff --git a/Source/Core/DolphinQt2/Config/CheatWarningWidget.h b/Source/Core/DolphinQt2/Config/CheatWarningWidget.h index 15b3334620..25e93e3b26 100644 --- a/Source/Core/DolphinQt2/Config/CheatWarningWidget.h +++ b/Source/Core/DolphinQt2/Config/CheatWarningWidget.h @@ -19,8 +19,6 @@ public: signals: void OpenCheatEnableSettings(); - void EmulationStarted(); - void EmulationStopped(); private: void CreateWidgets(); 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/DolphinQt2.vcxproj b/Source/Core/DolphinQt2/DolphinQt2.vcxproj index be07c59b2a..bfd4dcb868 100644 --- a/Source/Core/DolphinQt2/DolphinQt2.vcxproj +++ b/Source/Core/DolphinQt2/DolphinQt2.vcxproj @@ -60,8 +60,10 @@ + + @@ -111,14 +113,16 @@ - + + + @@ -154,8 +158,10 @@ + + @@ -226,8 +232,8 @@ - - + + diff --git a/Source/Core/DolphinQt2/GameList/GameList.cpp b/Source/Core/DolphinQt2/GameList/GameList.cpp index df4e9d161c..7588bd12b4 100644 --- a/Source/Core/DolphinQt2/GameList/GameList.cpp +++ b/Source/Core/DolphinQt2/GameList/GameList.cpp @@ -239,6 +239,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 a55b5b94b2..bfa3aff851 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() @@ -551,6 +553,12 @@ void MainWindow::ShowAudioWindow() ShowSettingsWindow(); } +void MainWindow::ShowGeneralWindow() +{ + m_settings_window->SelectGeneralPane(); + ShowSettingsWindow(); +} + void MainWindow::ShowAboutDialog() { AboutDialog* about = new AboutDialog(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();