2018-01-24 05:35:13 -07:00
|
|
|
// Copyright 2018 Dolphin Emulator Project
|
2021-07-04 19:22:19 -06:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2018-01-24 05:35:13 -07:00
|
|
|
|
2018-07-06 16:40:15 -06:00
|
|
|
#include "DolphinQt/Config/CheatCodeEditor.h"
|
2018-01-24 05:35:13 -07:00
|
|
|
|
|
|
|
#include <QDialogButtonBox>
|
|
|
|
#include <QFontDatabase>
|
|
|
|
#include <QGridLayout>
|
|
|
|
#include <QLabel>
|
|
|
|
#include <QLineEdit>
|
|
|
|
#include <QStringList>
|
|
|
|
#include <QTextEdit>
|
|
|
|
|
2021-08-10 10:29:54 -06:00
|
|
|
#include "Common/StringUtil.h"
|
|
|
|
|
2018-01-24 05:35:13 -07:00
|
|
|
#include "Core/ARDecrypt.h"
|
|
|
|
#include "Core/ActionReplay.h"
|
|
|
|
#include "Core/GeckoCodeConfig.h"
|
|
|
|
|
2019-03-04 12:49:00 -07:00
|
|
|
#include "DolphinQt/QtUtils/ModalMessageBox.h"
|
|
|
|
|
2018-05-06 10:02:39 -06:00
|
|
|
CheatCodeEditor::CheatCodeEditor(QWidget* parent) : QDialog(parent)
|
2018-01-24 05:35:13 -07:00
|
|
|
{
|
2018-05-06 10:02:39 -06:00
|
|
|
setWindowFlags(windowFlags() & ~Qt::WindowContextHelpButtonHint);
|
|
|
|
setWindowTitle(tr("Cheat Code Editor"));
|
|
|
|
|
2018-01-24 05:35:13 -07:00
|
|
|
CreateWidgets();
|
|
|
|
ConnectWidgets();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CheatCodeEditor::SetARCode(ActionReplay::ARCode* code)
|
|
|
|
{
|
|
|
|
m_name_edit->setText(QString::fromStdString(code->name));
|
|
|
|
|
|
|
|
QString s;
|
|
|
|
|
|
|
|
for (ActionReplay::AREntry& e : code->ops)
|
|
|
|
{
|
|
|
|
s += QStringLiteral("%1 %2\n")
|
|
|
|
.arg(e.cmd_addr, 8, 16, QLatin1Char('0'))
|
|
|
|
.arg(e.value, 8, 16, QLatin1Char('0'));
|
|
|
|
}
|
|
|
|
|
|
|
|
m_code_edit->setText(s);
|
|
|
|
|
|
|
|
m_creator_label->setHidden(true);
|
|
|
|
m_creator_edit->setHidden(true);
|
|
|
|
m_notes_label->setHidden(true);
|
|
|
|
m_notes_edit->setHidden(true);
|
|
|
|
|
|
|
|
m_ar_code = code;
|
|
|
|
m_gecko_code = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CheatCodeEditor::SetGeckoCode(Gecko::GeckoCode* code)
|
|
|
|
{
|
|
|
|
m_name_edit->setText(QString::fromStdString(code->name));
|
|
|
|
m_creator_edit->setText(QString::fromStdString(code->creator));
|
|
|
|
|
|
|
|
QString code_string;
|
|
|
|
|
|
|
|
for (const auto& c : code->codes)
|
|
|
|
code_string += QStringLiteral("%1 %2\n")
|
|
|
|
.arg(c.address, 8, 16, QLatin1Char('0'))
|
|
|
|
.arg(c.data, 8, 16, QLatin1Char('0'));
|
|
|
|
|
|
|
|
m_code_edit->setText(code_string);
|
|
|
|
|
|
|
|
QString notes_string;
|
|
|
|
for (const auto& line : code->notes)
|
|
|
|
notes_string += QStringLiteral("%1\n").arg(QString::fromStdString(line));
|
|
|
|
|
|
|
|
m_notes_edit->setText(notes_string);
|
|
|
|
|
|
|
|
m_creator_label->setHidden(false);
|
|
|
|
m_creator_edit->setHidden(false);
|
|
|
|
m_notes_label->setHidden(false);
|
|
|
|
m_notes_edit->setHidden(false);
|
|
|
|
|
|
|
|
m_gecko_code = code;
|
|
|
|
m_ar_code = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CheatCodeEditor::CreateWidgets()
|
|
|
|
{
|
|
|
|
m_name_edit = new QLineEdit;
|
|
|
|
m_creator_edit = new QLineEdit;
|
|
|
|
m_notes_edit = new QTextEdit;
|
|
|
|
m_code_edit = new QTextEdit;
|
|
|
|
m_button_box = new QDialogButtonBox(QDialogButtonBox::Cancel | QDialogButtonBox::Save);
|
|
|
|
|
|
|
|
m_creator_label = new QLabel(tr("Creator:"));
|
|
|
|
m_notes_label = new QLabel(tr("Notes:"));
|
|
|
|
|
|
|
|
QGridLayout* grid_layout = new QGridLayout;
|
|
|
|
|
|
|
|
grid_layout->addWidget(new QLabel(tr("Name:")), 0, 0);
|
|
|
|
grid_layout->addWidget(m_name_edit, 0, 1);
|
|
|
|
grid_layout->addWidget(m_creator_label, 1, 0);
|
|
|
|
grid_layout->addWidget(m_creator_edit, 1, 1);
|
|
|
|
grid_layout->addWidget(m_notes_label, 2, 0);
|
|
|
|
grid_layout->addWidget(m_notes_edit, 2, 1);
|
|
|
|
grid_layout->addWidget(new QLabel(tr("Code:")), 3, 0);
|
|
|
|
grid_layout->addWidget(m_code_edit, 3, 1);
|
|
|
|
grid_layout->addWidget(m_button_box, 4, 1);
|
|
|
|
|
|
|
|
QFont monospace(QFontDatabase::systemFont(QFontDatabase::FixedFont).family());
|
|
|
|
|
|
|
|
m_code_edit->setFont(monospace);
|
|
|
|
|
2018-06-05 04:45:08 -06:00
|
|
|
m_code_edit->setAcceptRichText(false);
|
|
|
|
m_notes_edit->setAcceptRichText(false);
|
|
|
|
|
2018-01-24 05:35:13 -07:00
|
|
|
setLayout(grid_layout);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CheatCodeEditor::ConnectWidgets()
|
|
|
|
{
|
|
|
|
connect(m_button_box, &QDialogButtonBox::accepted, this, &CheatCodeEditor::accept);
|
|
|
|
connect(m_button_box, &QDialogButtonBox::rejected, this, &QDialog::reject);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CheatCodeEditor::AcceptAR()
|
|
|
|
{
|
2021-08-10 11:19:34 -06:00
|
|
|
QString name = m_name_edit->text();
|
|
|
|
|
2018-01-24 05:35:13 -07:00
|
|
|
std::vector<ActionReplay::AREntry> entries;
|
|
|
|
std::vector<std::string> encrypted_lines;
|
|
|
|
|
2019-07-30 05:57:06 -06:00
|
|
|
QStringList lines = m_code_edit->toPlainText().split(QLatin1Char{'\n'});
|
2018-01-24 05:35:13 -07:00
|
|
|
|
|
|
|
for (int i = 0; i < lines.size(); i++)
|
|
|
|
{
|
|
|
|
QString line = lines[i];
|
|
|
|
|
|
|
|
if (line.isEmpty())
|
|
|
|
continue;
|
|
|
|
|
2021-08-10 11:19:34 -06:00
|
|
|
if (i == 0 && line[0] == u'$')
|
|
|
|
{
|
|
|
|
if (name.isEmpty())
|
|
|
|
name = line.right(line.size() - 1);
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-07-30 05:57:06 -06:00
|
|
|
QStringList values = line.split(QLatin1Char{' '});
|
2018-01-24 05:35:13 -07:00
|
|
|
|
|
|
|
bool good = true;
|
|
|
|
|
|
|
|
u32 addr = 0;
|
|
|
|
u32 value = 0;
|
|
|
|
|
|
|
|
if (values.size() == 2)
|
|
|
|
{
|
|
|
|
addr = values[0].toUInt(&good, 16);
|
|
|
|
|
|
|
|
if (good)
|
|
|
|
value = values[1].toUInt(&good, 16);
|
|
|
|
|
|
|
|
if (good)
|
|
|
|
entries.push_back(ActionReplay::AREntry(addr, value));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-07-30 05:57:06 -06:00
|
|
|
QStringList blocks = line.split(QLatin1Char{'-'});
|
2018-01-24 05:35:13 -07:00
|
|
|
|
|
|
|
if (blocks.size() == 3 && blocks[0].size() == 4 && blocks[1].size() == 4 &&
|
2018-03-22 16:42:20 -06:00
|
|
|
blocks[2].size() == 5)
|
2018-01-24 05:35:13 -07:00
|
|
|
{
|
|
|
|
encrypted_lines.emplace_back(blocks[0].toStdString() + blocks[1].toStdString() +
|
|
|
|
blocks[2].toStdString());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
good = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!good)
|
|
|
|
{
|
2019-03-04 12:49:00 -07:00
|
|
|
auto result = ModalMessageBox::warning(
|
2018-01-24 05:35:13 -07:00
|
|
|
this, tr("Parsing Error"),
|
|
|
|
tr("Unable to parse line %1 of the entered AR code as a valid "
|
|
|
|
"encrypted or decrypted code. Make sure you typed it correctly.\n\n"
|
|
|
|
"Would you like to ignore this line and continue parsing?")
|
|
|
|
.arg(i + 1),
|
|
|
|
QMessageBox::Ok | QMessageBox::Abort);
|
|
|
|
|
|
|
|
if (result == QMessageBox::Abort)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!encrypted_lines.empty())
|
|
|
|
{
|
|
|
|
if (!entries.empty())
|
|
|
|
{
|
2019-03-04 12:49:00 -07:00
|
|
|
auto result = ModalMessageBox::warning(
|
2018-01-24 05:35:13 -07:00
|
|
|
this, tr("Invalid Mixed Code"),
|
|
|
|
tr("This Action Replay code contains both encrypted and unencrypted lines; "
|
|
|
|
"you should check that you have entered it correctly.\n\n"
|
|
|
|
"Do you want to discard all unencrypted lines?"),
|
|
|
|
QMessageBox::Yes | QMessageBox::No | QMessageBox::Cancel);
|
|
|
|
|
|
|
|
// YES = Discard the unencrypted lines then decrypt the encrypted ones instead.
|
|
|
|
// NO = Discard the encrypted lines, keep the unencrypted ones
|
|
|
|
// CANCEL = Stop and let the user go back to editing
|
|
|
|
switch (result)
|
|
|
|
{
|
|
|
|
case QMessageBox::Yes:
|
|
|
|
entries.clear();
|
|
|
|
break;
|
|
|
|
case QMessageBox::No:
|
|
|
|
encrypted_lines.clear();
|
|
|
|
break;
|
|
|
|
case QMessageBox::Cancel:
|
|
|
|
return false;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ActionReplay::DecryptARCode(encrypted_lines, &entries);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (entries.empty())
|
|
|
|
{
|
2019-03-04 12:49:00 -07:00
|
|
|
ModalMessageBox::critical(this, tr("Error"),
|
|
|
|
tr("The resulting decrypted AR code doesn't contain any lines."));
|
2018-01-24 05:35:13 -07:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-08-10 11:19:34 -06:00
|
|
|
m_ar_code->name = name.toStdString();
|
2018-01-24 05:35:13 -07:00
|
|
|
m_ar_code->ops = std::move(entries);
|
|
|
|
m_ar_code->user_defined = true;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CheatCodeEditor::AcceptGecko()
|
|
|
|
{
|
2021-08-10 11:19:34 -06:00
|
|
|
QString name = m_name_edit->text();
|
|
|
|
|
2018-01-24 05:35:13 -07:00
|
|
|
std::vector<Gecko::GeckoCode::Code> entries;
|
|
|
|
|
2019-07-30 05:57:06 -06:00
|
|
|
QStringList lines = m_code_edit->toPlainText().split(QLatin1Char{'\n'});
|
2018-01-24 05:35:13 -07:00
|
|
|
|
|
|
|
for (int i = 0; i < lines.size(); i++)
|
|
|
|
{
|
|
|
|
QString line = lines[i];
|
|
|
|
|
|
|
|
if (line.isEmpty())
|
|
|
|
continue;
|
|
|
|
|
2021-08-10 11:19:34 -06:00
|
|
|
if (i == 0 && line[0] == u'$')
|
|
|
|
{
|
|
|
|
if (name.isEmpty())
|
|
|
|
name = line.right(line.size() - 1);
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-07-30 05:57:06 -06:00
|
|
|
QStringList values = line.split(QLatin1Char{' '});
|
2018-01-24 05:35:13 -07:00
|
|
|
|
2018-04-27 14:45:40 -06:00
|
|
|
bool good = values.size() == 2;
|
2018-01-24 05:35:13 -07:00
|
|
|
|
|
|
|
u32 addr = 0;
|
|
|
|
u32 value = 0;
|
|
|
|
|
2018-04-27 14:45:40 -06:00
|
|
|
if (good)
|
|
|
|
addr = values[0].toUInt(&good, 16);
|
2018-01-24 05:35:13 -07:00
|
|
|
|
|
|
|
if (good)
|
|
|
|
value = values[1].toUInt(&good, 16);
|
|
|
|
|
|
|
|
if (!good)
|
|
|
|
{
|
2019-03-04 12:49:00 -07:00
|
|
|
auto result = ModalMessageBox::warning(
|
|
|
|
this, tr("Parsing Error"),
|
|
|
|
tr("Unable to parse line %1 of the entered Gecko code as a valid "
|
|
|
|
"code. Make sure you typed it correctly.\n\n"
|
|
|
|
"Would you like to ignore this line and continue parsing?")
|
|
|
|
.arg(i + 1),
|
|
|
|
QMessageBox::Ok | QMessageBox::Abort);
|
2018-01-24 05:35:13 -07:00
|
|
|
|
|
|
|
if (result == QMessageBox::Abort)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Gecko::GeckoCode::Code c;
|
|
|
|
c.address = addr;
|
|
|
|
c.data = value;
|
|
|
|
c.original_line = line.toStdString();
|
|
|
|
|
|
|
|
entries.push_back(c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (entries.empty())
|
|
|
|
{
|
2021-08-10 10:29:54 -06:00
|
|
|
ModalMessageBox::critical(this, tr("Error"), tr("This Gecko code doesn't contain any lines."));
|
2018-04-27 14:45:40 -06:00
|
|
|
return false;
|
2018-01-24 05:35:13 -07:00
|
|
|
}
|
|
|
|
|
2021-08-10 11:19:34 -06:00
|
|
|
m_gecko_code->name = name.toStdString();
|
2018-01-24 05:35:13 -07:00
|
|
|
m_gecko_code->creator = m_creator_edit->text().toStdString();
|
|
|
|
m_gecko_code->codes = std::move(entries);
|
2021-08-10 10:29:54 -06:00
|
|
|
m_gecko_code->notes = SplitString(m_notes_edit->toPlainText().toStdString(), '\n');
|
2018-01-24 05:35:13 -07:00
|
|
|
m_gecko_code->user_defined = true;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CheatCodeEditor::accept()
|
|
|
|
{
|
|
|
|
bool success = m_gecko_code != nullptr ? AcceptGecko() : AcceptAR();
|
|
|
|
|
|
|
|
if (success)
|
|
|
|
QDialog::accept();
|
|
|
|
}
|