From 63f54478f0840a168a942999c894edad83bc6b59 Mon Sep 17 00:00:00 2001 From: offtkp Date: Sat, 27 Jan 2024 17:01:02 +0200 Subject: [PATCH 01/10] Add cheat picker window --- CMakeLists.txt | 6 +- include/panda_qt/cheats_window.hpp | 26 +++ include/panda_qt/main_window.hpp | 5 + src/panda_qt/cheats_window.cpp | 313 +++++++++++++++++++++++++++++ src/panda_qt/main_window.cpp | 14 +- 5 files changed, 358 insertions(+), 6 deletions(-) create mode 100644 include/panda_qt/cheats_window.hpp create mode 100644 src/panda_qt/cheats_window.cpp diff --git a/CMakeLists.txt b/CMakeLists.txt index baa37466..9748b8f8 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -14,7 +14,7 @@ if(CMAKE_CXX_COMPILER_ID STREQUAL "Clang" AND CMAKE_CXX_COMPILER_VERSION VERSION endif() if(NOT CMAKE_BUILD_TYPE) - set(CMAKE_BUILD_TYPE Release) + set(CMAKE_BUILD_TYPE RelWithDebInfo) endif() project(Alber) @@ -197,10 +197,10 @@ set(RENDERER_SW_SOURCE_FILES src/core/renderer_sw/renderer_sw.cpp) if(NOT ANDROID) if(ENABLE_QT_GUI) set(FRONTEND_SOURCE_FILES src/panda_qt/main.cpp src/panda_qt/screen.cpp src/panda_qt/main_window.cpp src/panda_qt/about_window.cpp - src/panda_qt/config_window.cpp src/panda_qt/zep.cpp src/panda_qt/text_editor.cpp + src/panda_qt/config_window.cpp src/panda_qt/zep.cpp src/panda_qt/text_editor.cpp src/panda_qt/cheats_window.cpp ) set(FRONTEND_HEADER_FILES include/panda_qt/screen.hpp include/panda_qt/main_window.hpp include/panda_qt/about_window.hpp - include/panda_qt/config_window.hpp include/panda_qt/text_editor.hpp + include/panda_qt/config_window.hpp include/panda_qt/text_editor.hpp include/panda_qt/cheats_window.hpp ) source_group("Source Files\\Qt" FILES ${FRONTEND_SOURCE_FILES}) diff --git a/include/panda_qt/cheats_window.hpp b/include/panda_qt/cheats_window.hpp new file mode 100644 index 00000000..2160a1f6 --- /dev/null +++ b/include/panda_qt/cheats_window.hpp @@ -0,0 +1,26 @@ +#pragma once + +#include +#include +#include +#include +#include "emulator.hpp" + +class QListWidget; + +class CheatsWindow final : public QWidget +{ + Q_OBJECT + +public: + CheatsWindow(Emulator* emu, const std::filesystem::path& path, QWidget* parent = nullptr); + ~CheatsWindow() = default; + +private: + void addEntry(); + void removeClicked(); + + QListWidget* cheatList; + std::filesystem::path cheatPath; + Emulator* emu; +}; diff --git a/include/panda_qt/main_window.hpp b/include/panda_qt/main_window.hpp index 7dfb91b7..39a8b35f 100644 --- a/include/panda_qt/main_window.hpp +++ b/include/panda_qt/main_window.hpp @@ -12,6 +12,7 @@ #include "emulator.hpp" #include "panda_qt/about_window.hpp" #include "panda_qt/config_window.hpp" +#include "panda_qt/cheats_window.hpp" #include "panda_qt/screen.hpp" #include "panda_qt/text_editor.hpp" #include "services/hid.hpp" @@ -54,15 +55,19 @@ class MainWindow : public QMainWindow { ScreenWidget screen; AboutWindow* aboutWindow; ConfigWindow* configWindow; + CheatsWindow* cheatsEditor; TextEditorWindow* luaEditor; QMenuBar* menuBar = nullptr; + QAction* cheatsEditorAction = nullptr; + void swapEmuBuffer(); void emuThreadMainLoop(); void selectLuaFile(); void selectROM(); void dumpRomFS(); void openLuaEditor(); + void openCheatsEditor(); void showAboutMenu(); void sendMessage(const EmulatorMessage& message); void dispatchMessage(const EmulatorMessage& message); diff --git a/src/panda_qt/cheats_window.cpp b/src/panda_qt/cheats_window.cpp new file mode 100644 index 00000000..909ba0cb --- /dev/null +++ b/src/panda_qt/cheats_window.cpp @@ -0,0 +1,313 @@ +#include "panda_qt/cheats_window.hpp" +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include "cheats.hpp" +#include "emulator.hpp" + +using CheatHandle = u32; + +CheatHandle BAD_CHEAT = 0xFFFFFFFF; + +struct CheatMetadata +{ + CheatHandle handle = BAD_CHEAT; + std::string name = "New cheat"; + std::string code; + bool enabled = true; +}; + +u32 addCheat(Emulator* emu, u8* data, size_t size) +{ + if ((size % 8) != 0) { + return BAD_CHEAT; + } + + Cheats::Cheat cheat; + cheat.enabled = true; + cheat.type = Cheats::CheatType::ActionReplay; + + for (size_t i = 0; i < size; i += 8) { + auto read32 = [](const u8* ptr) { return (u32(ptr[3]) << 24) | (u32(ptr[2]) << 16) | (u32(ptr[1]) << 8) | u32(ptr[0]); }; + + // Data is passed to us in big endian so we bswap + u32 firstWord = Common::swap32(read32(data + i)); + u32 secondWord = Common::swap32(read32(data + i + 4)); + cheat.instructions.insert(cheat.instructions.end(), {firstWord, secondWord}); + } + + return emu->getCheats().addCheat(cheat); +} + +class CheatEntryWidget : public QWidget +{ +public: + CheatEntryWidget(Emulator* emu, CheatMetadata metadata, QListWidget* parent); + + void Update() + { + name->setText(metadata.name.c_str()); + enabled->setChecked(metadata.enabled); + update(); + } + + void Remove() + { + emu->getCheats().removeCheat(metadata.handle); + cheatList->takeItem(cheatList->row(listItem)); + deleteLater(); + } + + const CheatMetadata& GetMetadata() + { + return metadata; + } + + void SetMetadata(const CheatMetadata& metadata) + { + this->metadata = metadata; + } + +private: + void checkboxChanged(int state); + void editClicked(); + + Emulator* emu; + CheatMetadata metadata; + u32 handle; + QLabel* name; + QCheckBox* enabled; + QListWidget* cheatList; + QListWidgetItem* listItem; +}; + +class CheatEditDialog : public QDialog +{ +public: + CheatEditDialog(Emulator* emu, CheatEntryWidget& cheatEntry); + + void accepted(); + void rejected(); +private: + Emulator* emu; + CheatEntryWidget& cheatEntry; + QTextEdit* codeEdit; + QLineEdit* nameEdit; +}; + +CheatEntryWidget::CheatEntryWidget(Emulator* emu, CheatMetadata metadata, QListWidget* parent) + : QWidget(), emu(emu), metadata(metadata), cheatList(parent) +{ + QHBoxLayout* layout = new QHBoxLayout; + + enabled = new QCheckBox; + enabled->setChecked(metadata.enabled); + + name = new QLabel(metadata.name.c_str()); + QPushButton* buttonEdit = new QPushButton(tr("Edit")); + + connect(enabled, &QCheckBox::stateChanged, this, &CheatEntryWidget::checkboxChanged); + connect(buttonEdit, &QPushButton::clicked, this, &CheatEntryWidget::editClicked); + + layout->addWidget(enabled); + layout->addWidget(name); + layout->addWidget(buttonEdit); + setLayout(layout); + + listItem = new QListWidgetItem; + listItem->setSizeHint(sizeHint()); + parent->addItem(listItem); + parent->setItemWidget(listItem, this); +} + +void CheatEntryWidget::checkboxChanged(int state) +{ + bool enabled = state == Qt::Checked; + if (metadata.handle == BAD_CHEAT) + { + printf("Cheat handle is bad, this shouldn't happen\n"); + return; + } + + if (enabled) + { + emu->getCheats().enableCheat(metadata.handle); + metadata.enabled = true; + } + else + { + emu->getCheats().disableCheat(metadata.handle); + metadata.enabled = false; + } +} + +void CheatEntryWidget::editClicked() +{ + CheatEditDialog* dialog = new CheatEditDialog(emu, *this); + dialog->show(); +} + +CheatEditDialog::CheatEditDialog(Emulator* emu, CheatEntryWidget& cheatEntry) : QDialog(), emu(emu), cheatEntry(cheatEntry) +{ + setAttribute(Qt::WA_DeleteOnClose); + setModal(true); + + QVBoxLayout* layout = new QVBoxLayout; + const CheatMetadata& metadata = cheatEntry.GetMetadata(); + codeEdit = new QTextEdit; + nameEdit = new QLineEdit; + nameEdit->setText(metadata.name.c_str()); + nameEdit->setPlaceholderText(tr("Cheat name")); + layout->addWidget(nameEdit); + + QFont font; + font.setFamily("Courier"); + font.setFixedPitch(true); + font.setPointSize(10); + codeEdit->setFont(font); + + if (metadata.code.size() != 0) + { + // Nicely format it like so: + // 01234567 89ABCDEF + // 01234567 89ABCDEF + std::string formattedCode; + for (size_t i = 0; i < metadata.code.size(); i += 2) + { + if (i != 0) { + if (i % 8 == 0 && i % 16 != 0) + { + formattedCode += " "; + } + else if (i % 16 == 0) + { + formattedCode += "\n"; + } + } + + formattedCode += metadata.code[i]; + formattedCode += metadata.code[i + 1]; + } + codeEdit->setText(formattedCode.c_str()); + } + + layout->addWidget(codeEdit); + setLayout(layout); + + auto buttons = QDialogButtonBox::Ok | QDialogButtonBox::Cancel; + QDialogButtonBox* button_box = new QDialogButtonBox(buttons); + layout->addWidget(button_box); + + connect(button_box, &QDialogButtonBox::accepted, this, &QDialog::accept); + connect(button_box, &QDialogButtonBox::rejected, this, &QDialog::reject); + connect(this, &QDialog::rejected, this, &CheatEditDialog::rejected); + connect(this, &QDialog::accepted, this, &CheatEditDialog::accepted); +} + +void CheatEditDialog::accepted() +{ + QString code = codeEdit->toPlainText(); + code.replace(QRegularExpression("[^0-9a-fA-F]"), ""); + + CheatMetadata metadata = cheatEntry.GetMetadata(); + bool isEditing = metadata.handle != BAD_CHEAT; + metadata.name = nameEdit->text().toStdString(); + metadata.code = code.toStdString(); + + std::vector bytes; + for (size_t i = 0; i < metadata.code.size(); i += 2) + { + std::string hex = metadata.code.substr(i, 2); + bytes.push_back((uint8_t)std::stoul(hex, nullptr, 16)); + } + + if (isEditing) + { + emu->getCheats().removeCheat(metadata.handle); + u32 handle = addCheat(emu, bytes.data(), bytes.size()); + metadata.handle = handle; + cheatEntry.SetMetadata(metadata); + } + else + { + if (metadata.name.empty()) + { + metadata.name = tr("Cheat code").toStdString(); + } + u32 handle = addCheat(emu, bytes.data(), bytes.size()); + metadata.handle = handle; + cheatEntry.SetMetadata(metadata); + } + + cheatEntry.Update(); +} + +void CheatEditDialog::rejected() +{ + bool isEditing = cheatEntry.GetMetadata().handle != BAD_CHEAT; + + if (!isEditing) + { + // Was adding a cheat but pressed cancel + cheatEntry.Remove(); + } +} + +CheatsWindow::CheatsWindow(Emulator* emu, const std::filesystem::path& cheatPath, QWidget* parent) +: QWidget(parent, Qt::Window), emu(emu), cheatPath(cheatPath) +{ + QVBoxLayout* layout = new QVBoxLayout; + layout->setContentsMargins(6, 6, 6, 6); + setLayout(layout); + + cheatList = new QListWidget; + layout->addWidget(cheatList); + + QWidget* buttonBox = new QWidget; + QHBoxLayout* buttonLayout = new QHBoxLayout; + + QPushButton* buttonAdd = new QPushButton(tr("Add")); + QPushButton* buttonRemove = new QPushButton(tr("Remove")); + + connect(buttonAdd, &QPushButton::clicked, this, &CheatsWindow::addEntry); + connect(buttonRemove, &QPushButton::clicked, this, &CheatsWindow::removeClicked); + + buttonLayout->addWidget(buttonAdd); + buttonLayout->addWidget(buttonRemove); + buttonBox->setLayout(buttonLayout); + + layout->addWidget(buttonBox); + + // TODO: load cheats from saved cheats per game + // for (const CheatMetadata& metadata : getSavedCheats()) + // { + // new CheatEntryWidget(emu, metadata, cheatList); + // } +} + +void CheatsWindow::addEntry() +{ + // CheatEntryWidget is added to the list when it's created + CheatEntryWidget* entry = new CheatEntryWidget(emu, {BAD_CHEAT, "New cheat", "", true}, cheatList); + CheatEditDialog* dialog = new CheatEditDialog(emu, *entry); + dialog->show(); +} + +void CheatsWindow::removeClicked() +{ + QListWidgetItem* item = cheatList->currentItem(); + if (item == nullptr) + { + return; + } + + CheatEntryWidget* entry = static_cast(cheatList->itemWidget(item)); + entry->Remove(); +} diff --git a/src/panda_qt/main_window.cpp b/src/panda_qt/main_window.cpp index 5c661119..be5e4fd6 100644 --- a/src/panda_qt/main_window.cpp +++ b/src/panda_qt/main_window.cpp @@ -48,20 +48,24 @@ MainWindow::MainWindow(QApplication* app, QWidget* parent) : QMainWindow(parent) auto dumpRomFSAction = toolsMenu->addAction(tr("Dump RomFS")); auto luaEditorAction = toolsMenu->addAction(tr("Open Lua Editor")); + cheatsEditorAction = toolsMenu->addAction(tr("Open Cheats Editor")); + cheatsEditorAction->setEnabled(false); connect(dumpRomFSAction, &QAction::triggered, this, &MainWindow::dumpRomFS); connect(luaEditorAction, &QAction::triggered, this, &MainWindow::openLuaEditor); + connect(cheatsEditorAction, &QAction::triggered, this, &MainWindow::openCheatsEditor); auto aboutAction = aboutMenu->addAction(tr("About Panda3DS")); connect(aboutAction, &QAction::triggered, this, &MainWindow::showAboutMenu); + emu = new Emulator(); + emu->setOutputSize(screen.surfaceWidth, screen.surfaceHeight); + // Set up misc objects aboutWindow = new AboutWindow(nullptr); configWindow = new ConfigWindow(this); + cheatsEditor = new CheatsWindow(emu, {}); luaEditor = new TextEditorWindow(this, "script.lua", ""); - emu = new Emulator(); - emu->setOutputSize(screen.surfaceWidth, screen.surfaceHeight); - auto args = QCoreApplication::arguments(); if (args.size() > 1) { auto romPath = std::filesystem::current_path() / args.at(1).toStdU16String(); @@ -184,6 +188,7 @@ MainWindow::~MainWindow() { delete menuBar; delete aboutWindow; delete configWindow; + delete cheatsEditor; delete luaEditor; } @@ -234,10 +239,13 @@ void MainWindow::showAboutMenu() { void MainWindow::openLuaEditor() { luaEditor->show(); } +void MainWindow::openCheatsEditor() { cheatsEditor->show(); } + void MainWindow::dispatchMessage(const EmulatorMessage& message) { switch (message.type) { case MessageType::LoadROM: emu->loadROM(*message.path.p); + cheatsEditorAction->setEnabled(true); // Clean up the allocated path delete message.path.p; break; From a473a34794e189c251079f50b95941ac1110fd91 Mon Sep 17 00:00:00 2001 From: offtkp Date: Sat, 27 Jan 2024 18:20:42 +0200 Subject: [PATCH 02/10] Use message queues --- CMakeLists.txt | 2 +- include/cheats.hpp | 3 + include/panda_qt/cheats_window.hpp | 28 +- include/panda_qt/main_window.hpp | 16 +- src/core/cheats.cpp | 22 ++ src/panda_qt/cheats_window.cpp | 429 ++++++++++++----------------- src/panda_qt/main_window.cpp | 35 ++- 7 files changed, 266 insertions(+), 269 deletions(-) diff --git a/CMakeLists.txt b/CMakeLists.txt index 9748b8f8..ac6cdfb8 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -14,7 +14,7 @@ if(CMAKE_CXX_COMPILER_ID STREQUAL "Clang" AND CMAKE_CXX_COMPILER_VERSION VERSION endif() if(NOT CMAKE_BUILD_TYPE) - set(CMAKE_BUILD_TYPE RelWithDebInfo) + set(CMAKE_BUILD_TYPE Release) endif() project(Alber) diff --git a/include/cheats.hpp b/include/cheats.hpp index 2be25827..ca8abe1d 100644 --- a/include/cheats.hpp +++ b/include/cheats.hpp @@ -9,6 +9,8 @@ // Forward-declare this since it's just passed and we don't want to include memory.hpp and increase compile time class Memory; +constexpr u32 badCheatHandle = 0xFFFFFFFF; + class Cheats { public: enum class CheatType { @@ -24,6 +26,7 @@ class Cheats { Cheats(Memory& mem, HIDService& hid); u32 addCheat(const Cheat& cheat); + u32 addCheat(const u8* data, size_t size); void removeCheat(u32 id); void enableCheat(u32 id); void disableCheat(u32 id); diff --git a/include/panda_qt/cheats_window.hpp b/include/panda_qt/cheats_window.hpp index 2160a1f6..c82b2bd8 100644 --- a/include/panda_qt/cheats_window.hpp +++ b/include/panda_qt/cheats_window.hpp @@ -1,26 +1,26 @@ #pragma once -#include -#include #include #include +#include +#include + #include "emulator.hpp" class QListWidget; -class CheatsWindow final : public QWidget -{ - Q_OBJECT +class CheatsWindow final : public QWidget { + Q_OBJECT -public: - CheatsWindow(Emulator* emu, const std::filesystem::path& path, QWidget* parent = nullptr); - ~CheatsWindow() = default; + public: + CheatsWindow(Emulator* emu, const std::filesystem::path& path, QWidget* parent = nullptr); + ~CheatsWindow() = default; -private: - void addEntry(); - void removeClicked(); + private: + void addEntry(); + void removeClicked(); - QListWidget* cheatList; - std::filesystem::path cheatPath; - Emulator* emu; + QListWidget* cheatList; + std::filesystem::path cheatPath; + Emulator* emu; }; diff --git a/include/panda_qt/main_window.hpp b/include/panda_qt/main_window.hpp index 39a8b35f..d5eccc93 100644 --- a/include/panda_qt/main_window.hpp +++ b/include/panda_qt/main_window.hpp @@ -17,12 +17,19 @@ #include "panda_qt/text_editor.hpp" #include "services/hid.hpp" +struct CheatMessage +{ + u32 handle; + std::vector cheat; + std::function callback; +}; + class MainWindow : public QMainWindow { Q_OBJECT private: // Types of messages we might send from the GUI thread to the emulator thread - enum class MessageType { LoadROM, Reset, Pause, Resume, TogglePause, DumpRomFS, PressKey, ReleaseKey, LoadLuaScript }; + enum class MessageType { LoadROM, Reset, Pause, Resume, TogglePause, DumpRomFS, PressKey, ReleaseKey, LoadLuaScript, EditCheat }; // Tagged union representing our message queue messages struct EmulatorMessage { @@ -40,6 +47,10 @@ class MainWindow : public QMainWindow { struct { std::string* str; } string; + + struct { + CheatMessage* c; + } cheat; }; }; @@ -59,8 +70,6 @@ class MainWindow : public QMainWindow { TextEditorWindow* luaEditor; QMenuBar* menuBar = nullptr; - QAction* cheatsEditorAction = nullptr; - void swapEmuBuffer(); void emuThreadMainLoop(); void selectLuaFile(); @@ -83,4 +92,5 @@ class MainWindow : public QMainWindow { void keyPressEvent(QKeyEvent* event) override; void keyReleaseEvent(QKeyEvent* event) override; void loadLuaScript(const std::string& code); + void editCheat(u32 handle, const std::vector& cheat, const std::function& callback); }; \ No newline at end of file diff --git a/src/core/cheats.cpp b/src/core/cheats.cpp index 83e7cdc4..7b8b71c2 100644 --- a/src/core/cheats.cpp +++ b/src/core/cheats.cpp @@ -1,4 +1,5 @@ #include "cheats.hpp" +#include "swap.hpp" Cheats::Cheats(Memory& mem, HIDService& hid) : ar(mem, hid) { reset(); } @@ -23,6 +24,27 @@ u32 Cheats::addCheat(const Cheat& cheat) { return cheats.size() - 1; } +u32 Cheats::addCheat(const u8* data, size_t size) { + if ((size % 8) != 0) { + return badCheatHandle; + } + + Cheats::Cheat cheat; + cheat.enabled = true; + cheat.type = Cheats::CheatType::ActionReplay; + + for (size_t i = 0; i < size; i += 8) { + auto read32 = [](const u8* ptr) { return (u32(ptr[3]) << 24) | (u32(ptr[2]) << 16) | (u32(ptr[1]) << 8) | u32(ptr[0]); }; + + // Data is passed to us in big endian so we bswap + u32 firstWord = Common::swap32(read32(data + i)); + u32 secondWord = Common::swap32(read32(data + i + 4)); + cheat.instructions.insert(cheat.instructions.end(), {firstWord, secondWord}); + } + + return addCheat(cheat); +} + void Cheats::removeCheat(u32 id) { if (id >= cheats.size()) { return; diff --git a/src/panda_qt/cheats_window.cpp b/src/panda_qt/cheats_window.cpp index 909ba0cb..c6628125 100644 --- a/src/panda_qt/cheats_window.cpp +++ b/src/panda_qt/cheats_window.cpp @@ -1,313 +1,248 @@ #include "panda_qt/cheats_window.hpp" -#include -#include -#include -#include -#include + #include #include -#include -#include #include +#include +#include +#include +#include +#include +#include + #include "cheats.hpp" #include "emulator.hpp" +#include "panda_qt/main_window.hpp" -using CheatHandle = u32; +MainWindow* mainWindow = nullptr; -CheatHandle BAD_CHEAT = 0xFFFFFFFF; - -struct CheatMetadata -{ - CheatHandle handle = BAD_CHEAT; - std::string name = "New cheat"; - std::string code; - bool enabled = true; +struct CheatMetadata { + u32 handle = badCheatHandle; + std::string name = "New cheat"; + std::string code; + bool enabled = true; }; -u32 addCheat(Emulator* emu, u8* data, size_t size) -{ - if ((size % 8) != 0) { - return BAD_CHEAT; +class CheatEntryWidget : public QWidget { + public: + CheatEntryWidget(Emulator* emu, CheatMetadata metadata, QListWidget* parent); + + void Update() { + name->setText(metadata.name.c_str()); + enabled->setChecked(metadata.enabled); + update(); } - Cheats::Cheat cheat; - cheat.enabled = true; - cheat.type = Cheats::CheatType::ActionReplay; - - for (size_t i = 0; i < size; i += 8) { - auto read32 = [](const u8* ptr) { return (u32(ptr[3]) << 24) | (u32(ptr[2]) << 16) | (u32(ptr[1]) << 8) | u32(ptr[0]); }; - - // Data is passed to us in big endian so we bswap - u32 firstWord = Common::swap32(read32(data + i)); - u32 secondWord = Common::swap32(read32(data + i + 4)); - cheat.instructions.insert(cheat.instructions.end(), {firstWord, secondWord}); + void Remove() { + emu->getCheats().removeCheat(metadata.handle); + cheatList->takeItem(cheatList->row(listItem)); + deleteLater(); } - return emu->getCheats().addCheat(cheat); -} + const CheatMetadata& GetMetadata() { return metadata; } -class CheatEntryWidget : public QWidget -{ -public: - CheatEntryWidget(Emulator* emu, CheatMetadata metadata, QListWidget* parent); + void SetMetadata(const CheatMetadata& metadata) { this->metadata = metadata; } - void Update() - { - name->setText(metadata.name.c_str()); - enabled->setChecked(metadata.enabled); - update(); - } + private: + void checkboxChanged(int state); + void editClicked(); - void Remove() - { - emu->getCheats().removeCheat(metadata.handle); - cheatList->takeItem(cheatList->row(listItem)); - deleteLater(); - } - - const CheatMetadata& GetMetadata() - { - return metadata; - } - - void SetMetadata(const CheatMetadata& metadata) - { - this->metadata = metadata; - } - -private: - void checkboxChanged(int state); - void editClicked(); - - Emulator* emu; - CheatMetadata metadata; - u32 handle; - QLabel* name; - QCheckBox* enabled; - QListWidget* cheatList; - QListWidgetItem* listItem; + Emulator* emu; + CheatMetadata metadata; + u32 handle; + QLabel* name; + QCheckBox* enabled; + QListWidget* cheatList; + QListWidgetItem* listItem; }; -class CheatEditDialog : public QDialog -{ -public: - CheatEditDialog(Emulator* emu, CheatEntryWidget& cheatEntry); +class CheatEditDialog : public QDialog { + public: + CheatEditDialog(Emulator* emu, CheatEntryWidget& cheatEntry); - void accepted(); - void rejected(); -private: - Emulator* emu; - CheatEntryWidget& cheatEntry; - QTextEdit* codeEdit; - QLineEdit* nameEdit; + void accepted(); + void rejected(); + + private: + Emulator* emu; + CheatEntryWidget& cheatEntry; + QTextEdit* codeEdit; + QLineEdit* nameEdit; }; CheatEntryWidget::CheatEntryWidget(Emulator* emu, CheatMetadata metadata, QListWidget* parent) - : QWidget(), emu(emu), metadata(metadata), cheatList(parent) -{ - QHBoxLayout* layout = new QHBoxLayout; + : QWidget(), emu(emu), metadata(metadata), cheatList(parent) { + QHBoxLayout* layout = new QHBoxLayout; - enabled = new QCheckBox; - enabled->setChecked(metadata.enabled); + enabled = new QCheckBox; + enabled->setChecked(metadata.enabled); - name = new QLabel(metadata.name.c_str()); - QPushButton* buttonEdit = new QPushButton(tr("Edit")); + name = new QLabel(metadata.name.c_str()); + QPushButton* buttonEdit = new QPushButton(tr("Edit")); - connect(enabled, &QCheckBox::stateChanged, this, &CheatEntryWidget::checkboxChanged); - connect(buttonEdit, &QPushButton::clicked, this, &CheatEntryWidget::editClicked); + connect(enabled, &QCheckBox::stateChanged, this, &CheatEntryWidget::checkboxChanged); + connect(buttonEdit, &QPushButton::clicked, this, &CheatEntryWidget::editClicked); - layout->addWidget(enabled); - layout->addWidget(name); - layout->addWidget(buttonEdit); - setLayout(layout); + layout->addWidget(enabled); + layout->addWidget(name); + layout->addWidget(buttonEdit); + setLayout(layout); - listItem = new QListWidgetItem; - listItem->setSizeHint(sizeHint()); - parent->addItem(listItem); - parent->setItemWidget(listItem, this); + listItem = new QListWidgetItem; + listItem->setSizeHint(sizeHint()); + parent->addItem(listItem); + parent->setItemWidget(listItem, this); } -void CheatEntryWidget::checkboxChanged(int state) -{ - bool enabled = state == Qt::Checked; - if (metadata.handle == BAD_CHEAT) - { - printf("Cheat handle is bad, this shouldn't happen\n"); - return; - } +void CheatEntryWidget::checkboxChanged(int state) { + bool enabled = state == Qt::Checked; + if (metadata.handle == badCheatHandle) { + printf("Cheat handle is bad, this shouldn't happen\n"); + return; + } - if (enabled) - { - emu->getCheats().enableCheat(metadata.handle); - metadata.enabled = true; - } - else - { - emu->getCheats().disableCheat(metadata.handle); - metadata.enabled = false; - } + if (enabled) { + emu->getCheats().enableCheat(metadata.handle); + metadata.enabled = true; + } else { + emu->getCheats().disableCheat(metadata.handle); + metadata.enabled = false; + } } -void CheatEntryWidget::editClicked() -{ - CheatEditDialog* dialog = new CheatEditDialog(emu, *this); - dialog->show(); +void CheatEntryWidget::editClicked() { + CheatEditDialog* dialog = new CheatEditDialog(emu, *this); + dialog->show(); } -CheatEditDialog::CheatEditDialog(Emulator* emu, CheatEntryWidget& cheatEntry) : QDialog(), emu(emu), cheatEntry(cheatEntry) -{ - setAttribute(Qt::WA_DeleteOnClose); - setModal(true); +CheatEditDialog::CheatEditDialog(Emulator* emu, CheatEntryWidget& cheatEntry) : QDialog(), emu(emu), cheatEntry(cheatEntry) { + setAttribute(Qt::WA_DeleteOnClose); + setModal(true); - QVBoxLayout* layout = new QVBoxLayout; - const CheatMetadata& metadata = cheatEntry.GetMetadata(); - codeEdit = new QTextEdit; - nameEdit = new QLineEdit; - nameEdit->setText(metadata.name.c_str()); - nameEdit->setPlaceholderText(tr("Cheat name")); - layout->addWidget(nameEdit); + QVBoxLayout* layout = new QVBoxLayout; + const CheatMetadata& metadata = cheatEntry.GetMetadata(); + codeEdit = new QTextEdit; + nameEdit = new QLineEdit; + nameEdit->setText(metadata.name.c_str()); + nameEdit->setPlaceholderText(tr("Cheat name")); + layout->addWidget(nameEdit); - QFont font; - font.setFamily("Courier"); - font.setFixedPitch(true); - font.setPointSize(10); - codeEdit->setFont(font); + QFont font; + font.setFamily("Courier"); + font.setFixedPitch(true); + font.setPointSize(10); + codeEdit->setFont(font); - if (metadata.code.size() != 0) - { - // Nicely format it like so: - // 01234567 89ABCDEF - // 01234567 89ABCDEF - std::string formattedCode; - for (size_t i = 0; i < metadata.code.size(); i += 2) - { - if (i != 0) { - if (i % 8 == 0 && i % 16 != 0) - { - formattedCode += " "; - } - else if (i % 16 == 0) - { - formattedCode += "\n"; - } - } + if (metadata.code.size() != 0) { + // Nicely format it like so: + // 01234567 89ABCDEF + // 01234567 89ABCDEF + std::string formattedCode; + for (size_t i = 0; i < metadata.code.size(); i += 2) { + if (i != 0) { + if (i % 8 == 0 && i % 16 != 0) { + formattedCode += " "; + } else if (i % 16 == 0) { + formattedCode += "\n"; + } + } - formattedCode += metadata.code[i]; - formattedCode += metadata.code[i + 1]; - } - codeEdit->setText(formattedCode.c_str()); - } + formattedCode += metadata.code[i]; + formattedCode += metadata.code[i + 1]; + } + codeEdit->setText(formattedCode.c_str()); + } - layout->addWidget(codeEdit); - setLayout(layout); + layout->addWidget(codeEdit); + setLayout(layout); - auto buttons = QDialogButtonBox::Ok | QDialogButtonBox::Cancel; - QDialogButtonBox* button_box = new QDialogButtonBox(buttons); - layout->addWidget(button_box); + auto buttons = QDialogButtonBox::Ok | QDialogButtonBox::Cancel; + QDialogButtonBox* button_box = new QDialogButtonBox(buttons); + layout->addWidget(button_box); - connect(button_box, &QDialogButtonBox::accepted, this, &QDialog::accept); - connect(button_box, &QDialogButtonBox::rejected, this, &QDialog::reject); - connect(this, &QDialog::rejected, this, &CheatEditDialog::rejected); - connect(this, &QDialog::accepted, this, &CheatEditDialog::accepted); + connect(button_box, &QDialogButtonBox::accepted, this, &QDialog::accept); + connect(button_box, &QDialogButtonBox::rejected, this, &QDialog::reject); + connect(this, &QDialog::rejected, this, &CheatEditDialog::rejected); + connect(this, &QDialog::accepted, this, &CheatEditDialog::accepted); } -void CheatEditDialog::accepted() -{ - QString code = codeEdit->toPlainText(); - code.replace(QRegularExpression("[^0-9a-fA-F]"), ""); +void CheatEditDialog::accepted() { + QString code = codeEdit->toPlainText(); + code.replace(QRegularExpression("[^0-9a-fA-F]"), ""); - CheatMetadata metadata = cheatEntry.GetMetadata(); - bool isEditing = metadata.handle != BAD_CHEAT; - metadata.name = nameEdit->text().toStdString(); - metadata.code = code.toStdString(); + CheatMetadata metadata = cheatEntry.GetMetadata(); + metadata.name = nameEdit->text().toStdString(); + metadata.code = code.toStdString(); - std::vector bytes; - for (size_t i = 0; i < metadata.code.size(); i += 2) - { - std::string hex = metadata.code.substr(i, 2); - bytes.push_back((uint8_t)std::stoul(hex, nullptr, 16)); - } + std::vector bytes; + for (size_t i = 0; i < metadata.code.size(); i += 2) { + std::string hex = metadata.code.substr(i, 2); + bytes.push_back((uint8_t)std::stoul(hex, nullptr, 16)); + } - if (isEditing) - { - emu->getCheats().removeCheat(metadata.handle); - u32 handle = addCheat(emu, bytes.data(), bytes.size()); - metadata.handle = handle; - cheatEntry.SetMetadata(metadata); - } - else - { - if (metadata.name.empty()) - { - metadata.name = tr("Cheat code").toStdString(); - } - u32 handle = addCheat(emu, bytes.data(), bytes.size()); - metadata.handle = handle; - cheatEntry.SetMetadata(metadata); - } - - cheatEntry.Update(); + mainWindow->editCheat(cheatEntry.GetMetadata().handle, bytes, [this](u32 handle) { + CheatMetadata metadata = cheatEntry.GetMetadata(); + metadata.handle = handle; + cheatEntry.SetMetadata(metadata); + cheatEntry.Update(); + }); } -void CheatEditDialog::rejected() -{ - bool isEditing = cheatEntry.GetMetadata().handle != BAD_CHEAT; +void CheatEditDialog::rejected() { + bool isEditing = cheatEntry.GetMetadata().handle != badCheatHandle; - if (!isEditing) - { - // Was adding a cheat but pressed cancel - cheatEntry.Remove(); - } + if (!isEditing) { + // Was adding a cheat but pressed cancel + cheatEntry.Remove(); + } } CheatsWindow::CheatsWindow(Emulator* emu, const std::filesystem::path& cheatPath, QWidget* parent) -: QWidget(parent, Qt::Window), emu(emu), cheatPath(cheatPath) -{ - QVBoxLayout* layout = new QVBoxLayout; - layout->setContentsMargins(6, 6, 6, 6); - setLayout(layout); + : QWidget(parent, Qt::Window), emu(emu), cheatPath(cheatPath) { + mainWindow = static_cast(parent); - cheatList = new QListWidget; - layout->addWidget(cheatList); + QVBoxLayout* layout = new QVBoxLayout; + layout->setContentsMargins(6, 6, 6, 6); + setLayout(layout); - QWidget* buttonBox = new QWidget; - QHBoxLayout* buttonLayout = new QHBoxLayout; + cheatList = new QListWidget; + layout->addWidget(cheatList); - QPushButton* buttonAdd = new QPushButton(tr("Add")); - QPushButton* buttonRemove = new QPushButton(tr("Remove")); + QWidget* buttonBox = new QWidget; + QHBoxLayout* buttonLayout = new QHBoxLayout; - connect(buttonAdd, &QPushButton::clicked, this, &CheatsWindow::addEntry); - connect(buttonRemove, &QPushButton::clicked, this, &CheatsWindow::removeClicked); + QPushButton* buttonAdd = new QPushButton(tr("Add")); + QPushButton* buttonRemove = new QPushButton(tr("Remove")); - buttonLayout->addWidget(buttonAdd); - buttonLayout->addWidget(buttonRemove); - buttonBox->setLayout(buttonLayout); + connect(buttonAdd, &QPushButton::clicked, this, &CheatsWindow::addEntry); + connect(buttonRemove, &QPushButton::clicked, this, &CheatsWindow::removeClicked); - layout->addWidget(buttonBox); + buttonLayout->addWidget(buttonAdd); + buttonLayout->addWidget(buttonRemove); + buttonBox->setLayout(buttonLayout); - // TODO: load cheats from saved cheats per game - // for (const CheatMetadata& metadata : getSavedCheats()) - // { - // new CheatEntryWidget(emu, metadata, cheatList); - // } + layout->addWidget(buttonBox); + + // TODO: load cheats from saved cheats per game + // for (const CheatMetadata& metadata : getSavedCheats()) + // { + // new CheatEntryWidget(emu, metadata, cheatList); + // } } -void CheatsWindow::addEntry() -{ - // CheatEntryWidget is added to the list when it's created - CheatEntryWidget* entry = new CheatEntryWidget(emu, {BAD_CHEAT, "New cheat", "", true}, cheatList); - CheatEditDialog* dialog = new CheatEditDialog(emu, *entry); - dialog->show(); +void CheatsWindow::addEntry() { + // CheatEntryWidget is added to the list when it's created + CheatEntryWidget* entry = new CheatEntryWidget(emu, {badCheatHandle, "New cheat", "", true}, cheatList); + CheatEditDialog* dialog = new CheatEditDialog(emu, *entry); + dialog->show(); } -void CheatsWindow::removeClicked() -{ - QListWidgetItem* item = cheatList->currentItem(); - if (item == nullptr) - { - return; - } +void CheatsWindow::removeClicked() { + QListWidgetItem* item = cheatList->currentItem(); + if (item == nullptr) { + return; + } - CheatEntryWidget* entry = static_cast(cheatList->itemWidget(item)); - entry->Remove(); + CheatEntryWidget* entry = static_cast(cheatList->itemWidget(item)); + entry->Remove(); } diff --git a/src/panda_qt/main_window.cpp b/src/panda_qt/main_window.cpp index be5e4fd6..d1f86173 100644 --- a/src/panda_qt/main_window.cpp +++ b/src/panda_qt/main_window.cpp @@ -6,6 +6,8 @@ #include #include +#include "cheats.hpp" + MainWindow::MainWindow(QApplication* app, QWidget* parent) : QMainWindow(parent), screen(this) { setWindowTitle("Alber"); // Enable drop events for loading ROMs @@ -48,8 +50,7 @@ MainWindow::MainWindow(QApplication* app, QWidget* parent) : QMainWindow(parent) auto dumpRomFSAction = toolsMenu->addAction(tr("Dump RomFS")); auto luaEditorAction = toolsMenu->addAction(tr("Open Lua Editor")); - cheatsEditorAction = toolsMenu->addAction(tr("Open Cheats Editor")); - cheatsEditorAction->setEnabled(false); + auto cheatsEditorAction = toolsMenu->addAction(tr("Open Cheats Editor")); connect(dumpRomFSAction, &QAction::triggered, this, &MainWindow::dumpRomFS); connect(luaEditorAction, &QAction::triggered, this, &MainWindow::openLuaEditor); connect(cheatsEditorAction, &QAction::triggered, this, &MainWindow::openCheatsEditor); @@ -63,7 +64,7 @@ MainWindow::MainWindow(QApplication* app, QWidget* parent) : QMainWindow(parent) // Set up misc objects aboutWindow = new AboutWindow(nullptr); configWindow = new ConfigWindow(this); - cheatsEditor = new CheatsWindow(emu, {}); + cheatsEditor = new CheatsWindow(emu, {}, this); luaEditor = new TextEditorWindow(this, "script.lua", ""); auto args = QCoreApplication::arguments(); @@ -245,7 +246,6 @@ void MainWindow::dispatchMessage(const EmulatorMessage& message) { switch (message.type) { case MessageType::LoadROM: emu->loadROM(*message.path.p); - cheatsEditorAction->setEnabled(true); // Clean up the allocated path delete message.path.p; break; @@ -255,6 +255,21 @@ void MainWindow::dispatchMessage(const EmulatorMessage& message) { delete message.string.str; break; + case MessageType::EditCheat: { + u32 handle = message.cheat.c->handle; + const std::vector& cheat = message.cheat.c->cheat; + const std::function& callback = message.cheat.c->callback; + bool isEditing = handle != badCheatHandle; + if (isEditing) { + emu->getCheats().removeCheat(handle); + u32 handle = emu->getCheats().addCheat(cheat.data(), cheat.size()); + } else { + u32 handle = emu->getCheats().addCheat(cheat.data(), cheat.size()); + callback(handle); + } + delete message.cheat.c; + } break; + case MessageType::Pause: emu->pause(); break; case MessageType::Resume: emu->resume(); break; case MessageType::TogglePause: emu->togglePause(); break; @@ -326,4 +341,16 @@ void MainWindow::loadLuaScript(const std::string& code) { // Make a copy of the code on the heap to send via the message queue message.string.str = new std::string(code); sendMessage(message); +} + +void MainWindow::editCheat(u32 handle, const std::vector& cheat, const std::function& callback) { + EmulatorMessage message{.type = MessageType::EditCheat}; + + CheatMessage* c = new CheatMessage(); + c->handle = handle; + c->cheat = cheat; + c->callback = callback; + + message.cheat.c = c; + sendMessage(message); } \ No newline at end of file From a2276c922f524629c5f3fe76be79b4eb454bc7c9 Mon Sep 17 00:00:00 2001 From: offtkp Date: Sat, 27 Jan 2024 18:21:44 +0200 Subject: [PATCH 03/10] Reduce code duplication --- src/hydra_core.cpp | 20 +------------------- 1 file changed, 1 insertion(+), 19 deletions(-) diff --git a/src/hydra_core.cpp b/src/hydra_core.cpp index d67ffe2f..acbf30a8 100644 --- a/src/hydra_core.cpp +++ b/src/hydra_core.cpp @@ -134,25 +134,7 @@ void HydraCore::setPollInputCallback(void (*callback)()) { pollInputCallback = c void HydraCore::setCheckButtonCallback(s32 (*callback)(u32 player, hydra::ButtonType button)) { checkButtonCallback = callback; } u32 HydraCore::addCheat(const u8* data, u32 size) { - // Every 3DS cheat is a multiple of 64 bits == 8 bytes - if ((size % 8) != 0) { - return hydra::BAD_CHEAT; - } - - Cheats::Cheat cheat; - cheat.enabled = true; - cheat.type = Cheats::CheatType::ActionReplay; - - for (u32 i = 0; i < size; i += 8) { - auto read32 = [](const u8* ptr) { return (u32(ptr[3]) << 24) | (u32(ptr[2]) << 16) | (u32(ptr[1]) << 8) | u32(ptr[0]); }; - - // Data is passed to us in big endian so we bswap - u32 firstWord = Common::swap32(read32(data + i)); - u32 secondWord = Common::swap32(read32(data + i + 4)); - cheat.instructions.insert(cheat.instructions.end(), {firstWord, secondWord}); - } - - return emulator->getCheats().addCheat(cheat); + return emulator->getCheats().addCheat(data, size); }; void HydraCore::removeCheat(u32 id) { emulator->getCheats().removeCheat(id); } From 3d52692536e541909f05e178305222b2f69d29c6 Mon Sep 17 00:00:00 2001 From: offtkp Date: Sat, 27 Jan 2024 19:01:05 +0200 Subject: [PATCH 04/10] Threading shenanigans --- include/panda_qt/main_window.hpp | 1 + src/panda_qt/cheats_window.cpp | 31 ++++++++++++++++++++++++++----- 2 files changed, 27 insertions(+), 5 deletions(-) diff --git a/include/panda_qt/main_window.hpp b/include/panda_qt/main_window.hpp index d5eccc93..f7757d73 100644 --- a/include/panda_qt/main_window.hpp +++ b/include/panda_qt/main_window.hpp @@ -5,6 +5,7 @@ #include #include #include +#include #include #include #include diff --git a/src/panda_qt/cheats_window.cpp b/src/panda_qt/cheats_window.cpp index c6628125..74be1e94 100644 --- a/src/panda_qt/cheats_window.cpp +++ b/src/panda_qt/cheats_window.cpp @@ -9,6 +9,8 @@ #include #include #include +#include +#include #include "cheats.hpp" #include "emulator.hpp" @@ -23,6 +25,19 @@ struct CheatMetadata { bool enabled = true; }; +void dispatchToMainThread(std::function callback) +{ + QTimer* timer = new QTimer(); + timer->moveToThread(qApp->thread()); + timer->setSingleShot(true); + QObject::connect(timer, &QTimer::timeout, [=]() + { + callback(); + timer->deleteLater(); + }); + QMetaObject::invokeMethod(timer, "start", Qt::QueuedConnection, Q_ARG(int, 0)); +} + class CheatEntryWidget : public QWidget { public: CheatEntryWidget(Emulator* emu, CheatMetadata metadata, QListWidget* parent); @@ -181,16 +196,22 @@ void CheatEditDialog::accepted() { } mainWindow->editCheat(cheatEntry.GetMetadata().handle, bytes, [this](u32 handle) { - CheatMetadata metadata = cheatEntry.GetMetadata(); - metadata.handle = handle; - cheatEntry.SetMetadata(metadata); - cheatEntry.Update(); + dispatchToMainThread([this, handle]() { + if (handle == badCheatHandle) { + cheatEntry.Remove(); + return; + } else { + CheatMetadata metadata = cheatEntry.GetMetadata(); + metadata.handle = handle; + cheatEntry.SetMetadata(metadata); + cheatEntry.Update(); + } + }); }); } void CheatEditDialog::rejected() { bool isEditing = cheatEntry.GetMetadata().handle != badCheatHandle; - if (!isEditing) { // Was adding a cheat but pressed cancel cheatEntry.Remove(); From 2229adbd21471aa1b0c693314c5a176214ff39c1 Mon Sep 17 00:00:00 2001 From: wheremyfoodat <44909372+wheremyfoodat@users.noreply.github.com> Date: Sat, 27 Jan 2024 19:52:54 +0200 Subject: [PATCH 05/10] More action replay opcodes --- src/core/action_replay.cpp | 59 ++++++++++++++++++++++++++++++++++++++ 1 file changed, 59 insertions(+) diff --git a/src/core/action_replay.cpp b/src/core/action_replay.cpp index e8467425..1ef494a2 100644 --- a/src/core/action_replay.cpp +++ b/src/core/action_replay.cpp @@ -139,6 +139,65 @@ void ActionReplay::executeDType(const Cheat& cheat, u32 instruction) { switch (instruction) { case 0xD3000000: offset1 = cheat[pc++]; break; case 0xD3000001: offset2 = cheat[pc++]; break; + + case 0xD6000000: + write32(*activeOffset + cheat[pc++], u32(*activeData)); + *activeOffset += 4; + break; + + case 0xD6000001: + write32(*activeOffset + cheat[pc++], u32(data1)); + *activeOffset += 4; + break; + + case 0xD6000002: + write32(*activeOffset + cheat[pc++], u32(data2)); + *activeOffset += 4; + break; + + case 0xD7000000: + write16(*activeOffset + cheat[pc++], u16(*activeData)); + *activeOffset += 2; + break; + + case 0xD7000001: + write16(*activeOffset + cheat[pc++], u16(data1)); + *activeOffset += 2; + break; + + case 0xD7000002: + write16(*activeOffset + cheat[pc++], u16(data2)); + *activeOffset += 2; + break; + + case 0xD8000000: + write8(*activeOffset + cheat[pc++], u8(*activeData)); + *activeOffset += 1; + break; + + case 0xD8000001: + write8(*activeOffset + cheat[pc++], u8(data1)); + *activeOffset += 1; + break; + + case 0xD8000002: + write8(*activeOffset + cheat[pc++], u8(data2)); + *activeOffset += 1; + break; + + + case 0xD9000000: *activeData = read32(cheat[pc++] + *activeOffset); break; + case 0xD9000001: data1 = read32(cheat[pc++] + *activeOffset); break; + case 0xD9000002: data2 = read32(cheat[pc++] + *activeOffset); break; + + case 0xDA000000: *activeData = read16(cheat[pc++] + *activeOffset); break; + case 0xDA000001: data1 = read16(cheat[pc++] + *activeOffset); break; + case 0xDA000002: data2 = read16(cheat[pc++] + *activeOffset); break; + + case 0xDB000000: *activeData = read8(cheat[pc++] + *activeOffset); break; + case 0xDB000001: data1 = read8(cheat[pc++] + *activeOffset); break; + case 0xDB000002: data2 = read8(cheat[pc++] + *activeOffset); break; + case 0xDC000000: *activeOffset += cheat[pc++]; break; // DD000000 XXXXXXXX - if KEYPAD has value XXXXXXXX execute next block From e3c9f0b219618a72741ef168fd0b8b4bbcb5f629 Mon Sep 17 00:00:00 2001 From: wheremyfoodat <44909372+wheremyfoodat@users.noreply.github.com> Date: Sat, 27 Jan 2024 20:11:53 +0200 Subject: [PATCH 06/10] [Qt] Implement circlepad --- include/panda_qt/main_window.hpp | 19 ++++++++++++++++++- src/panda_qt/main_window.cpp | 23 +++++++++++++++++++++++ 2 files changed, 41 insertions(+), 1 deletion(-) diff --git a/include/panda_qt/main_window.hpp b/include/panda_qt/main_window.hpp index f7757d73..9be065cf 100644 --- a/include/panda_qt/main_window.hpp +++ b/include/panda_qt/main_window.hpp @@ -30,7 +30,20 @@ class MainWindow : public QMainWindow { private: // Types of messages we might send from the GUI thread to the emulator thread - enum class MessageType { LoadROM, Reset, Pause, Resume, TogglePause, DumpRomFS, PressKey, ReleaseKey, LoadLuaScript, EditCheat }; + enum class MessageType { + LoadROM, + Reset, + Pause, + Resume, + TogglePause, + DumpRomFS, + PressKey, + ReleaseKey, + SetCirclePadX, + SetCirclePadY, + LoadLuaScript, + EditCheat, + }; // Tagged union representing our message queue messages struct EmulatorMessage { @@ -45,6 +58,10 @@ class MainWindow : public QMainWindow { u32 key; } key; + struct { + s16 value; + } circlepad; + struct { std::string* str; } string; diff --git a/src/panda_qt/main_window.cpp b/src/panda_qt/main_window.cpp index d1f86173..e603b780 100644 --- a/src/panda_qt/main_window.cpp +++ b/src/panda_qt/main_window.cpp @@ -276,6 +276,8 @@ void MainWindow::dispatchMessage(const EmulatorMessage& message) { case MessageType::Reset: emu->reset(Emulator::ReloadOption::Reload); break; case MessageType::PressKey: emu->getServiceManager().getHID().pressKey(message.key.key); break; case MessageType::ReleaseKey: emu->getServiceManager().getHID().releaseKey(message.key.key); break; + case MessageType::SetCirclePadX: emu->getServiceManager().getHID().setCirclepadX(message.circlepad.value); break; + case MessageType::SetCirclePadY: emu->getServiceManager().getHID().setCirclepadY(message.circlepad.value); break; } } @@ -283,7 +285,12 @@ void MainWindow::keyPressEvent(QKeyEvent* event) { auto pressKey = [this](u32 key) { EmulatorMessage message{.type = MessageType::PressKey}; message.key.key = key; + sendMessage(message); + }; + auto setCirclePad = [this](MessageType type, s16 value) { + EmulatorMessage message{.type = type}; + message.circlepad.value = value; sendMessage(message); }; @@ -296,6 +303,11 @@ void MainWindow::keyPressEvent(QKeyEvent* event) { case Qt::Key_Q: pressKey(HID::Keys::L); break; case Qt::Key_P: pressKey(HID::Keys::R); break; + case Qt::Key_W: setCirclePad(MessageType::SetCirclePadY, 0x9C); break; + case Qt::Key_A: setCirclePad(MessageType::SetCirclePadX, -0x9C); break; + case Qt::Key_S: setCirclePad(MessageType::SetCirclePadY, -0x9C); break; + case Qt::Key_D: setCirclePad(MessageType::SetCirclePadX, 0x9C); break; + case Qt::Key_Right: pressKey(HID::Keys::Right); break; case Qt::Key_Left: pressKey(HID::Keys::Left); break; case Qt::Key_Up: pressKey(HID::Keys::Up); break; @@ -312,7 +324,12 @@ void MainWindow::keyReleaseEvent(QKeyEvent* event) { auto releaseKey = [this](u32 key) { EmulatorMessage message{.type = MessageType::ReleaseKey}; message.key.key = key; + sendMessage(message); + }; + auto releaseCirclePad = [this](MessageType type) { + EmulatorMessage message{.type = type}; + message.circlepad.value = 0; sendMessage(message); }; @@ -325,6 +342,12 @@ void MainWindow::keyReleaseEvent(QKeyEvent* event) { case Qt::Key_Q: releaseKey(HID::Keys::L); break; case Qt::Key_P: releaseKey(HID::Keys::R); break; + case Qt::Key_W: + case Qt::Key_S: releaseCirclePad(MessageType::SetCirclePadY); break; + + case Qt::Key_A: + case Qt::Key_D: releaseCirclePad(MessageType::SetCirclePadX); break; + case Qt::Key_Right: releaseKey(HID::Keys::Right); break; case Qt::Key_Left: releaseKey(HID::Keys::Left); break; case Qt::Key_Up: releaseKey(HID::Keys::Up); break; From 864604c1e7100e7622bc6ef7d208f2d04fbd6c14 Mon Sep 17 00:00:00 2001 From: wheremyfoodat <44909372+wheremyfoodat@users.noreply.github.com> Date: Sat, 27 Jan 2024 20:27:46 +0200 Subject: [PATCH 07/10] Formatting --- src/panda_qt/cheats_window.cpp | 31 +++++++++++++++---------------- 1 file changed, 15 insertions(+), 16 deletions(-) diff --git a/src/panda_qt/cheats_window.cpp b/src/panda_qt/cheats_window.cpp index 74be1e94..1f359396 100644 --- a/src/panda_qt/cheats_window.cpp +++ b/src/panda_qt/cheats_window.cpp @@ -54,9 +54,8 @@ class CheatEntryWidget : public QWidget { deleteLater(); } - const CheatMetadata& GetMetadata() { return metadata; } - - void SetMetadata(const CheatMetadata& metadata) { this->metadata = metadata; } + const CheatMetadata& getMetadata() { return metadata; } + void setMetadata(const CheatMetadata& metadata) { this->metadata = metadata; } private: void checkboxChanged(int state); @@ -135,7 +134,7 @@ CheatEditDialog::CheatEditDialog(Emulator* emu, CheatEntryWidget& cheatEntry) : setModal(true); QVBoxLayout* layout = new QVBoxLayout; - const CheatMetadata& metadata = cheatEntry.GetMetadata(); + const CheatMetadata& metadata = cheatEntry.getMetadata(); codeEdit = new QTextEdit; nameEdit = new QLineEdit; nameEdit->setText(metadata.name.c_str()); @@ -172,11 +171,11 @@ CheatEditDialog::CheatEditDialog(Emulator* emu, CheatEntryWidget& cheatEntry) : setLayout(layout); auto buttons = QDialogButtonBox::Ok | QDialogButtonBox::Cancel; - QDialogButtonBox* button_box = new QDialogButtonBox(buttons); - layout->addWidget(button_box); + QDialogButtonBox* buttonBox = new QDialogButtonBox(buttons); + layout->addWidget(buttonBox); - connect(button_box, &QDialogButtonBox::accepted, this, &QDialog::accept); - connect(button_box, &QDialogButtonBox::rejected, this, &QDialog::reject); + connect(buttonBox, &QDialogButtonBox::accepted, this, &QDialog::accept); + connect(buttonBox, &QDialogButtonBox::rejected, this, &QDialog::reject); connect(this, &QDialog::rejected, this, &CheatEditDialog::rejected); connect(this, &QDialog::accepted, this, &CheatEditDialog::accepted); } @@ -185,25 +184,25 @@ void CheatEditDialog::accepted() { QString code = codeEdit->toPlainText(); code.replace(QRegularExpression("[^0-9a-fA-F]"), ""); - CheatMetadata metadata = cheatEntry.GetMetadata(); + CheatMetadata metadata = cheatEntry.getMetadata(); metadata.name = nameEdit->text().toStdString(); metadata.code = code.toStdString(); - std::vector bytes; + std::vector bytes; for (size_t i = 0; i < metadata.code.size(); i += 2) { std::string hex = metadata.code.substr(i, 2); - bytes.push_back((uint8_t)std::stoul(hex, nullptr, 16)); + bytes.push_back((u8)std::stoul(hex, nullptr, 16)); } - mainWindow->editCheat(cheatEntry.GetMetadata().handle, bytes, [this](u32 handle) { + mainWindow->editCheat(cheatEntry.getMetadata().handle, bytes, [this](u32 handle) { dispatchToMainThread([this, handle]() { if (handle == badCheatHandle) { cheatEntry.Remove(); return; } else { - CheatMetadata metadata = cheatEntry.GetMetadata(); + CheatMetadata metadata = cheatEntry.getMetadata(); metadata.handle = handle; - cheatEntry.SetMetadata(metadata); + cheatEntry.setMetadata(metadata); cheatEntry.Update(); } }); @@ -211,9 +210,9 @@ void CheatEditDialog::accepted() { } void CheatEditDialog::rejected() { - bool isEditing = cheatEntry.GetMetadata().handle != badCheatHandle; + bool isEditing = cheatEntry.getMetadata().handle != badCheatHandle; if (!isEditing) { - // Was adding a cheat but pressed cancel + // Was adding a cheat but user pressed cancel cheatEntry.Remove(); } } From 31eea40ea54d985f82573c7ac15550b52423131b Mon Sep 17 00:00:00 2001 From: wheremyfoodat <44909372+wheremyfoodat@users.noreply.github.com> Date: Sat, 27 Jan 2024 20:33:35 +0200 Subject: [PATCH 08/10] Fix cheat dialog forgetting cheat names/codes --- src/panda_qt/cheats_window.cpp | 1 + 1 file changed, 1 insertion(+) diff --git a/src/panda_qt/cheats_window.cpp b/src/panda_qt/cheats_window.cpp index 1f359396..6636cae7 100644 --- a/src/panda_qt/cheats_window.cpp +++ b/src/panda_qt/cheats_window.cpp @@ -187,6 +187,7 @@ void CheatEditDialog::accepted() { CheatMetadata metadata = cheatEntry.getMetadata(); metadata.name = nameEdit->text().toStdString(); metadata.code = code.toStdString(); + cheatEntry.setMetadata(metadata); std::vector bytes; for (size_t i = 0; i < metadata.code.size(); i += 2) { From 01b6380242e4deeee832ba2115b4c83d20fe8cc3 Mon Sep 17 00:00:00 2001 From: wheremyfoodat <44909372+wheremyfoodat@users.noreply.github.com> Date: Sat, 27 Jan 2024 20:38:17 +0200 Subject: [PATCH 09/10] Remove global definition of badCheatHandle --- include/cheats.hpp | 3 +-- include/panda_qt/main_window.hpp | 3 +-- src/panda_qt/cheats_window.cpp | 33 ++++++++++++++++---------------- src/panda_qt/main_window.cpp | 3 +-- 4 files changed, 19 insertions(+), 23 deletions(-) diff --git a/include/cheats.hpp b/include/cheats.hpp index ca8abe1d..b90c080b 100644 --- a/include/cheats.hpp +++ b/include/cheats.hpp @@ -9,8 +9,6 @@ // Forward-declare this since it's just passed and we don't want to include memory.hpp and increase compile time class Memory; -constexpr u32 badCheatHandle = 0xFFFFFFFF; - class Cheats { public: enum class CheatType { @@ -35,6 +33,7 @@ class Cheats { void clear(); bool haveCheats() const { return cheatsLoaded; } + static constexpr u32 badCheatHandle = 0xFFFFFFFF; private: ActionReplay ar; // An ActionReplay cheat machine for executing CTRPF codes diff --git a/include/panda_qt/main_window.hpp b/include/panda_qt/main_window.hpp index 9be065cf..c03cffbb 100644 --- a/include/panda_qt/main_window.hpp +++ b/include/panda_qt/main_window.hpp @@ -18,8 +18,7 @@ #include "panda_qt/text_editor.hpp" #include "services/hid.hpp" -struct CheatMessage -{ +struct CheatMessage { u32 handle; std::vector cheat; std::function callback; diff --git a/src/panda_qt/cheats_window.cpp b/src/panda_qt/cheats_window.cpp index 6636cae7..dbd251cc 100644 --- a/src/panda_qt/cheats_window.cpp +++ b/src/panda_qt/cheats_window.cpp @@ -19,14 +19,13 @@ MainWindow* mainWindow = nullptr; struct CheatMetadata { - u32 handle = badCheatHandle; + u32 handle = Cheats::badCheatHandle; std::string name = "New cheat"; std::string code; bool enabled = true; }; -void dispatchToMainThread(std::function callback) -{ +void dispatchToMainThread(std::function callback) { QTimer* timer = new QTimer(); timer->moveToThread(qApp->thread()); timer->setSingleShot(true); @@ -110,7 +109,7 @@ CheatEntryWidget::CheatEntryWidget(Emulator* emu, CheatMetadata metadata, QListW void CheatEntryWidget::checkboxChanged(int state) { bool enabled = state == Qt::Checked; - if (metadata.handle == badCheatHandle) { + if (metadata.handle == Cheats::badCheatHandle) { printf("Cheat handle is bad, this shouldn't happen\n"); return; } @@ -196,22 +195,22 @@ void CheatEditDialog::accepted() { } mainWindow->editCheat(cheatEntry.getMetadata().handle, bytes, [this](u32 handle) { - dispatchToMainThread([this, handle]() { - if (handle == badCheatHandle) { - cheatEntry.Remove(); - return; - } else { - CheatMetadata metadata = cheatEntry.getMetadata(); - metadata.handle = handle; - cheatEntry.setMetadata(metadata); - cheatEntry.Update(); - } - }); + dispatchToMainThread([this, handle]() { + if (handle == Cheats::badCheatHandle) { + cheatEntry.Remove(); + return; + } else { + CheatMetadata metadata = cheatEntry.getMetadata(); + metadata.handle = handle; + cheatEntry.setMetadata(metadata); + cheatEntry.Update(); + } + }); }); } void CheatEditDialog::rejected() { - bool isEditing = cheatEntry.getMetadata().handle != badCheatHandle; + bool isEditing = cheatEntry.getMetadata().handle != Cheats::badCheatHandle; if (!isEditing) { // Was adding a cheat but user pressed cancel cheatEntry.Remove(); @@ -253,7 +252,7 @@ CheatsWindow::CheatsWindow(Emulator* emu, const std::filesystem::path& cheatPath void CheatsWindow::addEntry() { // CheatEntryWidget is added to the list when it's created - CheatEntryWidget* entry = new CheatEntryWidget(emu, {badCheatHandle, "New cheat", "", true}, cheatList); + CheatEntryWidget* entry = new CheatEntryWidget(emu, {Cheats::badCheatHandle, "New cheat", "", true}, cheatList); CheatEditDialog* dialog = new CheatEditDialog(emu, *entry); dialog->show(); } diff --git a/src/panda_qt/main_window.cpp b/src/panda_qt/main_window.cpp index e603b780..1d7118bc 100644 --- a/src/panda_qt/main_window.cpp +++ b/src/panda_qt/main_window.cpp @@ -239,7 +239,6 @@ void MainWindow::showAboutMenu() { } void MainWindow::openLuaEditor() { luaEditor->show(); } - void MainWindow::openCheatsEditor() { cheatsEditor->show(); } void MainWindow::dispatchMessage(const EmulatorMessage& message) { @@ -259,7 +258,7 @@ void MainWindow::dispatchMessage(const EmulatorMessage& message) { u32 handle = message.cheat.c->handle; const std::vector& cheat = message.cheat.c->cheat; const std::function& callback = message.cheat.c->callback; - bool isEditing = handle != badCheatHandle; + bool isEditing = handle != Cheats::badCheatHandle; if (isEditing) { emu->getCheats().removeCheat(handle); u32 handle = emu->getCheats().addCheat(cheat.data(), cheat.size()); From c6d0769a95b5a9a0d3a584351e072f34b559aa3a Mon Sep 17 00:00:00 2001 From: wheremyfoodat <44909372+wheremyfoodat@users.noreply.github.com> Date: Sun, 28 Jan 2024 00:32:20 +0200 Subject: [PATCH 10/10] Update text_editor.cpp --- src/panda_qt/text_editor.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/panda_qt/text_editor.cpp b/src/panda_qt/text_editor.cpp index c189c2ce..a31a829f 100644 --- a/src/panda_qt/text_editor.cpp +++ b/src/panda_qt/text_editor.cpp @@ -16,8 +16,8 @@ TextEditorWindow::TextEditorWindow(QWidget* parent, const std::string& filename, ZepReplExCommand::Register(zepWidget.GetEditor(), &replProvider); // Default to standard mode instead of vim mode, initialize text box - zepWidget.GetEditor().SetGlobalMode(Zep::ZepMode_Standard::StaticName()); zepWidget.GetEditor().InitWithText(filename, initialText); + zepWidget.GetEditor().SetGlobalMode(Zep::ZepMode_Standard::StaticName()); // Layout for widgets QVBoxLayout* mainLayout = new QVBoxLayout(); @@ -41,4 +41,4 @@ TextEditorWindow::TextEditorWindow(QWidget* parent, const std::string& filename, mainLayout->addWidget(button); mainLayout->addWidget(&zepWidget); -} \ No newline at end of file +}