mirror of
				https://github.com/dolphin-emu/dolphin.git
				synced 2025-10-25 17:39:09 +00:00 
			
		
		
		
	Refactors the AR/Gecko/Patch code approval process to verify from every possible game ini, not just the base game ID. This fixes codes on specific revisions or codes general to any region.
		
			
				
	
	
		
			320 lines
		
	
	
	
		
			8.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			320 lines
		
	
	
	
		
			8.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2018 Dolphin Emulator Project
 | |
| // SPDX-License-Identifier: GPL-2.0-or-later
 | |
| 
 | |
| #include "DolphinQt/Config/ARCodeWidget.h"
 | |
| 
 | |
| #include <algorithm>
 | |
| #include <utility>
 | |
| 
 | |
| #include <QCursor>
 | |
| #include <QHBoxLayout>
 | |
| #include <QListWidget>
 | |
| #include <QMenu>
 | |
| #include <QPushButton>
 | |
| #include <QVBoxLayout>
 | |
| 
 | |
| #include "Common/FileUtil.h"
 | |
| #include "Common/IniFile.h"
 | |
| 
 | |
| #include "Core/ActionReplay.h"
 | |
| #include "Core/ConfigManager.h"
 | |
| 
 | |
| #include "DolphinQt/Config/CheatCodeEditor.h"
 | |
| #include "DolphinQt/Config/CheatWarningWidget.h"
 | |
| #include "DolphinQt/Config/HardcoreWarningWidget.h"
 | |
| #include "DolphinQt/QtUtils/NonDefaultQPushButton.h"
 | |
| #include "DolphinQt/QtUtils/SetWindowDecorations.h"
 | |
| #include "DolphinQt/QtUtils/WrapInScrollArea.h"
 | |
| 
 | |
| #include "UICommon/GameFile.h"
 | |
| 
 | |
| ARCodeWidget::ARCodeWidget(std::string game_id, u16 game_revision, bool restart_required)
 | |
|     : m_game_id(std::move(game_id)), m_game_revision(game_revision),
 | |
|       m_restart_required(restart_required)
 | |
| {
 | |
|   CreateWidgets();
 | |
|   ConnectWidgets();
 | |
| 
 | |
|   LoadCodes();
 | |
| }
 | |
| 
 | |
| ARCodeWidget::~ARCodeWidget() = default;
 | |
| 
 | |
| void ARCodeWidget::ChangeGame(std::string game_id, const u16 game_revision)
 | |
| {
 | |
|   m_game_id = std::move(game_id);
 | |
|   m_game_revision = game_revision;
 | |
|   m_restart_required = false;
 | |
| 
 | |
|   m_ar_codes.clear();
 | |
| 
 | |
|   // If a CheatCodeEditor is open, it's now trying to add or edit a code in the previous game's code
 | |
|   // list which is no longer loaded. Letting the user save the code wouldn't make sense, so close
 | |
|   // the dialog instead.
 | |
|   m_cheat_code_editor->reject();
 | |
| 
 | |
|   LoadCodes();
 | |
| }
 | |
| 
 | |
| void ARCodeWidget::CreateWidgets()
 | |
| {
 | |
|   m_warning = new CheatWarningWidget(m_game_id, m_restart_required, this);
 | |
| #ifdef USE_RETRO_ACHIEVEMENTS
 | |
|   m_hc_warning = new HardcoreWarningWidget(this);
 | |
| #endif  // USE_RETRO_ACHIEVEMENTS
 | |
|   m_code_list = new QListWidget;
 | |
|   m_code_add = new NonDefaultQPushButton(tr("&Add New Code..."));
 | |
|   m_code_edit = new NonDefaultQPushButton(tr("&Edit Code..."));
 | |
|   m_code_remove = new NonDefaultQPushButton(tr("&Remove Code"));
 | |
| 
 | |
|   m_cheat_code_editor = new CheatCodeEditor(this);
 | |
| 
 | |
|   m_code_list->setContextMenuPolicy(Qt::CustomContextMenu);
 | |
| 
 | |
|   auto* button_layout = new QHBoxLayout;
 | |
| 
 | |
|   button_layout->addWidget(m_code_add);
 | |
|   button_layout->addWidget(m_code_edit);
 | |
|   button_layout->addWidget(m_code_remove);
 | |
| 
 | |
|   QVBoxLayout* layout = new QVBoxLayout;
 | |
| 
 | |
|   layout->addWidget(m_warning);
 | |
| #ifdef USE_RETRO_ACHIEVEMENTS
 | |
|   layout->addWidget(m_hc_warning);
 | |
| #endif  // USE_RETRO_ACHIEVEMENTS
 | |
|   layout->addWidget(m_code_list);
 | |
|   layout->addLayout(button_layout);
 | |
| 
 | |
|   WrapInScrollArea(this, layout);
 | |
| }
 | |
| 
 | |
| void ARCodeWidget::ConnectWidgets()
 | |
| {
 | |
|   connect(m_warning, &CheatWarningWidget::OpenCheatEnableSettings, this,
 | |
|           &ARCodeWidget::OpenGeneralSettings);
 | |
| #ifdef USE_RETRO_ACHIEVEMENTS
 | |
|   connect(m_hc_warning, &HardcoreWarningWidget::OpenAchievementSettings, this,
 | |
|           &ARCodeWidget::OpenAchievementSettings);
 | |
| #endif  // USE_RETRO_ACHIEVEMENTS
 | |
| 
 | |
|   connect(m_code_list, &QListWidget::itemChanged, this, &ARCodeWidget::OnItemChanged);
 | |
|   connect(m_code_list, &QListWidget::itemSelectionChanged, this, &ARCodeWidget::OnSelectionChanged);
 | |
|   connect(m_code_list->model(), &QAbstractItemModel::rowsMoved, this,
 | |
|           &ARCodeWidget::OnListReordered);
 | |
|   connect(m_code_list, &QListWidget::customContextMenuRequested, this,
 | |
|           &ARCodeWidget::OnContextMenuRequested);
 | |
| 
 | |
|   connect(m_code_add, &QPushButton::clicked, this, &ARCodeWidget::OnCodeAddClicked);
 | |
|   connect(m_code_edit, &QPushButton::clicked, this, &ARCodeWidget::OnCodeEditClicked);
 | |
|   connect(m_code_remove, &QPushButton::clicked, this, &ARCodeWidget::OnCodeRemoveClicked);
 | |
| }
 | |
| 
 | |
| void ARCodeWidget::OnItemChanged(QListWidgetItem* item)
 | |
| {
 | |
|   m_ar_codes[m_code_list->row(item)].enabled = (item->checkState() == Qt::Checked);
 | |
| 
 | |
|   if (!m_restart_required)
 | |
|     ActionReplay::ApplyCodes(m_ar_codes, m_game_id, m_game_revision);
 | |
| 
 | |
|   UpdateList();
 | |
|   SaveCodes();
 | |
| }
 | |
| 
 | |
| void ARCodeWidget::OnContextMenuRequested()
 | |
| {
 | |
|   QMenu menu;
 | |
| 
 | |
|   menu.addAction(tr("Sort Alphabetically"), this, &ARCodeWidget::SortAlphabetically);
 | |
|   menu.addAction(tr("Show Enabled Codes First"), this, &ARCodeWidget::SortEnabledCodesFirst);
 | |
|   menu.addAction(tr("Show Disabled Codes First"), this, &ARCodeWidget::SortDisabledCodesFirst);
 | |
| 
 | |
|   menu.exec(QCursor::pos());
 | |
| }
 | |
| 
 | |
| void ARCodeWidget::SortAlphabetically()
 | |
| {
 | |
|   m_code_list->sortItems();
 | |
|   OnListReordered();
 | |
| }
 | |
| 
 | |
| void ARCodeWidget::SortEnabledCodesFirst()
 | |
| {
 | |
|   std::ranges::stable_partition(m_ar_codes, std::identity{}, &ActionReplay::ARCode::enabled);
 | |
|   UpdateList();
 | |
|   SaveCodes();
 | |
| }
 | |
| 
 | |
| void ARCodeWidget::SortDisabledCodesFirst()
 | |
| {
 | |
|   std::ranges::stable_partition(m_ar_codes, std::logical_not{}, &ActionReplay::ARCode::enabled);
 | |
|   UpdateList();
 | |
|   SaveCodes();
 | |
| }
 | |
| 
 | |
| void ARCodeWidget::OnListReordered()
 | |
| {
 | |
|   // Reorder codes based on the indices of table item
 | |
|   std::vector<ActionReplay::ARCode> codes;
 | |
|   codes.reserve(m_ar_codes.size());
 | |
| 
 | |
|   for (int i = 0; i < m_code_list->count(); i++)
 | |
|   {
 | |
|     const int index = m_code_list->item(i)->data(Qt::UserRole).toInt();
 | |
| 
 | |
|     codes.push_back(std::move(m_ar_codes[index]));
 | |
|   }
 | |
| 
 | |
|   m_ar_codes = std::move(codes);
 | |
| 
 | |
|   SaveCodes();
 | |
| }
 | |
| 
 | |
| void ARCodeWidget::OnSelectionChanged()
 | |
| {
 | |
|   const QList<QListWidgetItem*> items = m_code_list->selectedItems();
 | |
|   const bool empty = items.empty();
 | |
| 
 | |
|   m_code_edit->setDisabled(empty);
 | |
|   m_code_remove->setDisabled(empty);
 | |
| 
 | |
|   if (empty)
 | |
|     return;
 | |
| 
 | |
|   const QListWidgetItem* const selected = items[0];
 | |
| 
 | |
|   const bool user_defined = m_ar_codes[m_code_list->row(selected)].user_defined;
 | |
| 
 | |
|   m_code_remove->setEnabled(user_defined);
 | |
|   m_code_edit->setText(user_defined ? tr("&Edit Code...") : tr("Clone and &Edit Code..."));
 | |
| }
 | |
| 
 | |
| void ARCodeWidget::UpdateList()
 | |
| {
 | |
|   m_code_list->clear();
 | |
| 
 | |
|   for (size_t i = 0; i < m_ar_codes.size(); i++)
 | |
|   {
 | |
|     const auto& ar = m_ar_codes[i];
 | |
|     auto* item = new QListWidgetItem(QString::fromStdString(ar.name)
 | |
|                                          .replace(QStringLiteral("<"), QChar::fromLatin1('<'))
 | |
|                                          .replace(QStringLiteral(">"), QChar::fromLatin1('>')));
 | |
| 
 | |
|     item->setFlags(Qt::ItemIsEnabled | Qt::ItemIsSelectable | Qt::ItemIsUserCheckable |
 | |
|                    Qt::ItemIsDragEnabled);
 | |
|     item->setCheckState(ar.enabled ? Qt::Checked : Qt::Unchecked);
 | |
|     item->setData(Qt::UserRole, static_cast<int>(i));
 | |
| 
 | |
|     m_code_list->addItem(item);
 | |
|   }
 | |
| 
 | |
|   m_code_list->setDragDropMode(QAbstractItemView::InternalMove);
 | |
| }
 | |
| 
 | |
| void ARCodeWidget::LoadCodes()
 | |
| {
 | |
|   if (!m_game_id.empty())
 | |
|   {
 | |
|     Common::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");
 | |
| 
 | |
|     const Common::IniFile game_ini_default =
 | |
|         SConfig::LoadDefaultGameIni(m_game_id, m_game_revision);
 | |
|     m_ar_codes = ActionReplay::LoadCodes(game_ini_default, game_ini_local);
 | |
|   }
 | |
| 
 | |
|   m_code_list->setEnabled(!m_game_id.empty());
 | |
|   m_code_add->setEnabled(!m_game_id.empty());
 | |
|   m_code_edit->setEnabled(false);
 | |
|   m_code_remove->setEnabled(false);
 | |
| 
 | |
|   UpdateList();
 | |
| }
 | |
| 
 | |
| void ARCodeWidget::SaveCodes()
 | |
| {
 | |
|   if (m_game_id.empty())
 | |
|     return;
 | |
| 
 | |
|   const auto ini_path =
 | |
|       std::string(File::GetUserPath(D_GAMESETTINGS_IDX)).append(m_game_id).append(".ini");
 | |
| 
 | |
|   Common::IniFile game_ini_local;
 | |
|   game_ini_local.Load(ini_path);
 | |
|   ActionReplay::SaveCodes(&game_ini_local, m_ar_codes);
 | |
|   game_ini_local.Save(ini_path);
 | |
| }
 | |
| 
 | |
| void ARCodeWidget::AddCode(ActionReplay::ARCode code)
 | |
| {
 | |
|   m_ar_codes.push_back(std::move(code));
 | |
| 
 | |
|   UpdateList();
 | |
|   SaveCodes();
 | |
| }
 | |
| 
 | |
| void ARCodeWidget::OnCodeAddClicked()
 | |
| {
 | |
|   ActionReplay::ARCode ar;
 | |
|   ar.enabled = true;
 | |
| 
 | |
|   m_cheat_code_editor->SetARCode(&ar);
 | |
|   SetQWidgetWindowDecorations(m_cheat_code_editor);
 | |
|   if (m_cheat_code_editor->exec() == QDialog::Rejected)
 | |
|     return;
 | |
| 
 | |
|   m_ar_codes.push_back(std::move(ar));
 | |
| 
 | |
|   UpdateList();
 | |
|   SaveCodes();
 | |
| }
 | |
| 
 | |
| void ARCodeWidget::OnCodeEditClicked()
 | |
| {
 | |
|   const auto items = m_code_list->selectedItems();
 | |
|   if (items.empty())
 | |
|     return;
 | |
| 
 | |
|   const auto* const selected = items[0];
 | |
|   auto& current_ar = m_ar_codes[m_code_list->row(selected)];
 | |
|   SetQWidgetWindowDecorations(m_cheat_code_editor);
 | |
| 
 | |
|   if (current_ar.user_defined)
 | |
|   {
 | |
|     m_cheat_code_editor->SetARCode(¤t_ar);
 | |
|     if (m_cheat_code_editor->exec() == QDialog::Rejected)
 | |
|       return;
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     ActionReplay::ARCode ar = current_ar;
 | |
|     m_cheat_code_editor->SetARCode(&ar);
 | |
|     if (m_cheat_code_editor->exec() == QDialog::Rejected)
 | |
|       return;
 | |
| 
 | |
|     m_ar_codes.push_back(std::move(ar));
 | |
|   }
 | |
| 
 | |
|   SaveCodes();
 | |
|   UpdateList();
 | |
| }
 | |
| 
 | |
| void ARCodeWidget::OnCodeRemoveClicked()
 | |
| {
 | |
|   auto items = m_code_list->selectedItems();
 | |
| 
 | |
|   if (items.empty())
 | |
|     return;
 | |
| 
 | |
|   const auto* selected = items[0];
 | |
| 
 | |
|   m_ar_codes.erase(m_ar_codes.begin() + m_code_list->row(selected));
 | |
| 
 | |
|   SaveCodes();
 | |
|   UpdateList();
 | |
| 
 | |
|   m_code_remove->setEnabled(false);
 | |
| }
 |