mirror of
				https://github.com/dolphin-emu/dolphin.git
				synced 2025-10-26 18:09:20 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			339 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			339 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2023 Dolphin Emulator Project
 | |
| // SPDX-License-Identifier: GPL-2.0-or-later
 | |
| 
 | |
| #include "DolphinQt/InfinityBase/InfinityBaseWindow.h"
 | |
| 
 | |
| #include <string>
 | |
| 
 | |
| #include <QCheckBox>
 | |
| #include <QComboBox>
 | |
| #include <QCompleter>
 | |
| #include <QDialogButtonBox>
 | |
| #include <QGroupBox>
 | |
| #include <QLabel>
 | |
| #include <QLineEdit>
 | |
| #include <QMessageBox>
 | |
| #include <QPushButton>
 | |
| #include <QScrollArea>
 | |
| #include <QString>
 | |
| #include <QVBoxLayout>
 | |
| 
 | |
| #include "Common/IOFile.h"
 | |
| 
 | |
| #include "Core/Config/MainSettings.h"
 | |
| #include "Core/Core.h"
 | |
| #include "Core/IOS/USB/Emulated/Infinity.h"
 | |
| #include "Core/System.h"
 | |
| 
 | |
| #include "DolphinQt/QtUtils/DolphinFileDialog.h"
 | |
| #include "DolphinQt/Resources.h"
 | |
| #include "DolphinQt/Settings.h"
 | |
| 
 | |
| // Qt is not guaranteed to keep track of file paths using native file pickers, so we use this
 | |
| // static variable to ensure we open at the most recent figure file location
 | |
| static QString s_last_figure_path;
 | |
| 
 | |
| using FigureUIPosition = IOS::HLE::USB::FigureUIPosition;
 | |
| 
 | |
| InfinityBaseWindow::InfinityBaseWindow(QWidget* parent) : QWidget(parent)
 | |
| {
 | |
|   // i18n: Window for managing Disney Infinity figures
 | |
|   setWindowTitle(tr("Infinity Manager"));
 | |
|   setWindowIcon(Resources::GetAppIcon());
 | |
|   setObjectName(QStringLiteral("infinity_manager"));
 | |
|   setMinimumSize(QSize(700, 200));
 | |
| 
 | |
|   CreateMainWindow();
 | |
| 
 | |
|   connect(&Settings::Instance(), &Settings::EmulationStateChanged, this,
 | |
|           &InfinityBaseWindow::OnEmulationStateChanged);
 | |
| 
 | |
|   installEventFilter(this);
 | |
| 
 | |
|   OnEmulationStateChanged(Core::GetState(Core::System::GetInstance()));
 | |
| }
 | |
| 
 | |
| InfinityBaseWindow::~InfinityBaseWindow() = default;
 | |
| 
 | |
| void InfinityBaseWindow::CreateMainWindow()
 | |
| {
 | |
|   auto* main_layout = new QVBoxLayout();
 | |
| 
 | |
|   auto* checkbox_group = new QGroupBox();
 | |
|   auto* checkbox_layout = new QHBoxLayout();
 | |
|   checkbox_layout->setAlignment(Qt::AlignHCenter);
 | |
|   m_checkbox = new QCheckBox(tr("Emulate Infinity Base"), this);
 | |
|   m_checkbox->setChecked(Config::Get(Config::MAIN_EMULATE_INFINITY_BASE));
 | |
|   connect(m_checkbox, &QCheckBox::toggled, this, &InfinityBaseWindow::EmulateBase);
 | |
|   checkbox_layout->addWidget(m_checkbox);
 | |
|   checkbox_group->setLayout(checkbox_layout);
 | |
|   main_layout->addWidget(checkbox_group);
 | |
| 
 | |
|   auto add_line = [](QVBoxLayout* vbox) {
 | |
|     auto* line = new QFrame();
 | |
|     line->setFrameShape(QFrame::HLine);
 | |
|     line->setFrameShadow(QFrame::Sunken);
 | |
|     vbox->addWidget(line);
 | |
|   };
 | |
| 
 | |
|   m_group_figures = new QGroupBox(tr("Active Infinity Figures:"));
 | |
|   auto* vbox_group = new QVBoxLayout();
 | |
|   auto* scroll_area = new QScrollArea();
 | |
| 
 | |
|   AddFigureSlot(vbox_group, tr("Play Set/Power Disc"), FigureUIPosition::HexagonDiscOne);
 | |
|   add_line(vbox_group);
 | |
|   AddFigureSlot(vbox_group, tr("Power Disc Two"), FigureUIPosition::HexagonDiscTwo);
 | |
|   add_line(vbox_group);
 | |
|   AddFigureSlot(vbox_group, tr("Power Disc Three"), FigureUIPosition::HexagonDiscThree);
 | |
|   add_line(vbox_group);
 | |
|   AddFigureSlot(vbox_group, tr("Player One"), FigureUIPosition::PlayerOne);
 | |
|   add_line(vbox_group);
 | |
|   AddFigureSlot(vbox_group, tr("Player One Ability One"), FigureUIPosition::P1AbilityOne);
 | |
|   add_line(vbox_group);
 | |
|   AddFigureSlot(vbox_group, tr("Player One Ability Two"), FigureUIPosition::P1AbilityTwo);
 | |
|   add_line(vbox_group);
 | |
|   AddFigureSlot(vbox_group, tr("Player Two"), FigureUIPosition::PlayerTwo);
 | |
|   add_line(vbox_group);
 | |
|   AddFigureSlot(vbox_group, tr("Player Two Ability One"), FigureUIPosition::P2AbilityOne);
 | |
|   add_line(vbox_group);
 | |
|   AddFigureSlot(vbox_group, tr("Player Two Ability Two"), FigureUIPosition::P2AbilityTwo);
 | |
| 
 | |
|   m_group_figures->setLayout(vbox_group);
 | |
|   scroll_area->setWidget(m_group_figures);
 | |
|   scroll_area->setWidgetResizable(true);
 | |
|   m_group_figures->setVisible(Config::Get(Config::MAIN_EMULATE_INFINITY_BASE));
 | |
|   main_layout->addWidget(scroll_area);
 | |
|   setLayout(main_layout);
 | |
| }
 | |
| 
 | |
| void InfinityBaseWindow::AddFigureSlot(QVBoxLayout* vbox_group, QString name, FigureUIPosition slot)
 | |
| {
 | |
|   auto* hbox_infinity = new QHBoxLayout();
 | |
| 
 | |
|   auto* label_skyname = new QLabel(name);
 | |
| 
 | |
|   auto* clear_btn = new QPushButton(tr("Clear"));
 | |
|   auto* create_btn = new QPushButton(tr("Create"));
 | |
|   auto* load_btn = new QPushButton(tr("Load"));
 | |
|   m_edit_figures[static_cast<u8>(slot)] = new QLineEdit();
 | |
|   m_edit_figures[static_cast<u8>(slot)]->setEnabled(false);
 | |
|   m_edit_figures[static_cast<u8>(slot)]->setText(tr("None"));
 | |
| 
 | |
|   connect(clear_btn, &QAbstractButton::clicked, this, [this, slot] { ClearFigure(slot); });
 | |
|   connect(create_btn, &QAbstractButton::clicked, this, [this, slot] { CreateFigure(slot); });
 | |
|   connect(load_btn, &QAbstractButton::clicked, this, [this, slot] { LoadFigure(slot); });
 | |
| 
 | |
|   hbox_infinity->addWidget(label_skyname);
 | |
|   hbox_infinity->addWidget(m_edit_figures[static_cast<u8>(slot)]);
 | |
|   hbox_infinity->addWidget(clear_btn);
 | |
|   hbox_infinity->addWidget(create_btn);
 | |
|   hbox_infinity->addWidget(load_btn);
 | |
| 
 | |
|   vbox_group->addLayout(hbox_infinity);
 | |
| }
 | |
| 
 | |
| void InfinityBaseWindow::ClearFigure(FigureUIPosition slot)
 | |
| {
 | |
|   auto& system = Core::System::GetInstance();
 | |
|   m_edit_figures[static_cast<u8>(slot)]->setText(tr("None"));
 | |
| 
 | |
|   system.GetInfinityBase().RemoveFigure(slot);
 | |
| }
 | |
| 
 | |
| void InfinityBaseWindow::LoadFigure(FigureUIPosition slot)
 | |
| {
 | |
|   const QString file_path =
 | |
|       DolphinFileDialog::getOpenFileName(this, tr("Select Figure File"), s_last_figure_path,
 | |
|                                          QStringLiteral("Infinity Figure (*.bin);;"));
 | |
|   if (file_path.isEmpty())
 | |
|   {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   s_last_figure_path = QFileInfo(file_path).absolutePath() + QLatin1Char('/');
 | |
| 
 | |
|   LoadFigurePath(slot, file_path);
 | |
| }
 | |
| 
 | |
| void InfinityBaseWindow::CreateFigure(FigureUIPosition slot)
 | |
| {
 | |
|   CreateFigureDialog create_dlg(this, slot);
 | |
|   if (create_dlg.exec() == CreateFigureDialog::Accepted)
 | |
|   {
 | |
|     LoadFigurePath(slot, create_dlg.GetFilePath());
 | |
|   }
 | |
| }
 | |
| 
 | |
| void InfinityBaseWindow::LoadFigurePath(FigureUIPosition slot, const QString& path)
 | |
| {
 | |
|   File::IOFile inf_file(path.toStdString(), "r+b");
 | |
|   if (!inf_file)
 | |
|   {
 | |
|     QMessageBox::warning(
 | |
|         this, tr("Failed to open the Infinity file!"),
 | |
|         tr("Failed to open the Infinity file:\n%1\n\nThe file may already be in use on the base.")
 | |
|             .arg(path),
 | |
|         QMessageBox::Ok);
 | |
|     return;
 | |
|   }
 | |
|   std::array<u8, IOS::HLE::USB::INFINITY_NUM_BLOCKS * IOS::HLE::USB::INFINITY_BLOCK_SIZE> file_data;
 | |
|   if (!inf_file.ReadBytes(file_data.data(), file_data.size()))
 | |
|   {
 | |
|     QMessageBox::warning(
 | |
|         this, tr("Failed to read the Infinity file!"),
 | |
|         tr("Failed to read the Infinity file:\n%1\n\nThe file was too small.").arg(path),
 | |
|         QMessageBox::Ok);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   auto& system = Core::System::GetInstance();
 | |
| 
 | |
|   system.GetInfinityBase().RemoveFigure(slot);
 | |
|   m_edit_figures[static_cast<u8>(slot)]->setText(QString::fromStdString(
 | |
|       system.GetInfinityBase().LoadFigure(file_data, std::move(inf_file), slot)));
 | |
| }
 | |
| 
 | |
| CreateFigureDialog::CreateFigureDialog(QWidget* parent, FigureUIPosition slot) : QDialog(parent)
 | |
| {
 | |
|   setWindowTitle(tr("Infinity Figure Creator"));
 | |
|   setObjectName(QStringLiteral("infinity_creator"));
 | |
|   setMinimumSize(QSize(500, 150));
 | |
|   auto* layout = new QVBoxLayout;
 | |
| 
 | |
|   auto* combo_figlist = new QComboBox();
 | |
|   QStringList filterlist;
 | |
|   u32 first_entry = 0;
 | |
|   for (const auto& entry : IOS::HLE::USB::InfinityBase::GetFigureList())
 | |
|   {
 | |
|     const auto figure = entry.second;
 | |
|     // Only display entry if it is a piece appropriate for the slot
 | |
|     if ((slot == FigureUIPosition::HexagonDiscOne &&
 | |
|          ((figure > 0x1E8480 && figure < 0x2DC6BF) || (figure > 0x3D0900 && figure < 0x4C4B3F))) ||
 | |
|         ((slot == FigureUIPosition::HexagonDiscTwo || slot == FigureUIPosition::HexagonDiscThree) &&
 | |
|          (figure > 0x3D0900 && figure < 0x4C4B3F)) ||
 | |
|         ((slot == FigureUIPosition::PlayerOne || slot == FigureUIPosition::PlayerTwo) &&
 | |
|          figure < 0x1E847F) ||
 | |
|         ((slot == FigureUIPosition::P1AbilityOne || slot == FigureUIPosition::P1AbilityTwo ||
 | |
|           slot == FigureUIPosition::P2AbilityOne || slot == FigureUIPosition::P2AbilityTwo) &&
 | |
|          (figure > 0x2DC6C0 && figure < 0x3D08FF)))
 | |
|     {
 | |
|       const auto figure_name = QString::fromStdString(entry.first);
 | |
|       combo_figlist->addItem(figure_name, QVariant(figure));
 | |
|       filterlist << figure_name;
 | |
|       if (first_entry == 0)
 | |
|       {
 | |
|         first_entry = figure;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   combo_figlist->addItem(tr("--Unknown--"), QVariant(0xFFFF));
 | |
|   combo_figlist->setEditable(true);
 | |
|   combo_figlist->setInsertPolicy(QComboBox::NoInsert);
 | |
| 
 | |
|   auto* co_compl = new QCompleter(filterlist, this);
 | |
|   co_compl->setCaseSensitivity(Qt::CaseInsensitive);
 | |
|   co_compl->setCompletionMode(QCompleter::PopupCompletion);
 | |
|   co_compl->setFilterMode(Qt::MatchContains);
 | |
|   combo_figlist->setCompleter(co_compl);
 | |
| 
 | |
|   layout->addWidget(combo_figlist);
 | |
| 
 | |
|   auto* line = new QFrame();
 | |
|   line->setFrameShape(QFrame::HLine);
 | |
|   line->setFrameShadow(QFrame::Sunken);
 | |
|   layout->addWidget(line);
 | |
| 
 | |
|   auto* hbox_idvar = new QHBoxLayout();
 | |
|   auto* label_id = new QLabel(tr("Figure Number:"));
 | |
|   auto* edit_num = new QLineEdit(QString::number(first_entry));
 | |
|   auto* rxv = new QRegularExpressionValidator(QRegularExpression(QStringLiteral("\\d*")), this);
 | |
|   edit_num->setValidator(rxv);
 | |
|   hbox_idvar->addWidget(label_id);
 | |
|   hbox_idvar->addWidget(edit_num);
 | |
|   layout->addLayout(hbox_idvar);
 | |
| 
 | |
|   auto* buttons = new QDialogButtonBox(QDialogButtonBox::Ok | QDialogButtonBox::Cancel);
 | |
|   buttons->button(QDialogButtonBox::Ok)->setText(tr("Create"));
 | |
|   layout->addWidget(buttons);
 | |
| 
 | |
|   setLayout(layout);
 | |
| 
 | |
|   connect(combo_figlist, &QComboBox::currentIndexChanged, [=](int index) {
 | |
|     const u32 char_info = combo_figlist->itemData(index).toUInt();
 | |
|     if (char_info != 0xFFFFFFFF)
 | |
|     {
 | |
|       edit_num->setText(QString::number(char_info));
 | |
|     }
 | |
|   });
 | |
| 
 | |
|   connect(buttons, &QDialogButtonBox::accepted, this, [=, this] {
 | |
|     bool ok_char = false;
 | |
|     const u32 char_number = edit_num->text().toULong(&ok_char);
 | |
|     if (!ok_char)
 | |
|     {
 | |
|       QMessageBox::warning(this, tr("Error converting value"), tr("Character entered is invalid!"),
 | |
|                            QMessageBox::Ok);
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     QString predef_name = s_last_figure_path;
 | |
| 
 | |
|     auto& system = Core::System::GetInstance();
 | |
|     const auto found_fig = system.GetInfinityBase().FindFigure(char_number);
 | |
|     if (!found_fig.empty())
 | |
|     {
 | |
|       predef_name += QString::fromStdString(found_fig + ".bin");
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|       // i18n: This is used to create a file name. The string must end in ".bin".
 | |
|       QString str = tr("Unknown(%1).bin");
 | |
|       predef_name += str.arg(char_number);
 | |
|     }
 | |
| 
 | |
|     m_file_path = DolphinFileDialog::getSaveFileName(this, tr("Create Infinity File"), predef_name,
 | |
|                                                      tr("Infinity Object (*.bin);;"));
 | |
|     if (m_file_path.isEmpty())
 | |
|     {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     if (!system.GetInfinityBase().CreateFigure(m_file_path.toStdString(), char_number))
 | |
|     {
 | |
|       QMessageBox::warning(
 | |
|           this, tr("Failed to create Infinity file"),
 | |
|           tr("Blank figure creation failed at:\n%1\n\nTry again with a different character.")
 | |
|               .arg(m_file_path),
 | |
|           QMessageBox::Ok);
 | |
|       return;
 | |
|     }
 | |
|     s_last_figure_path = QFileInfo(m_file_path).absolutePath() + QLatin1Char('/');
 | |
|     accept();
 | |
|   });
 | |
| 
 | |
|   connect(buttons, &QDialogButtonBox::rejected, this, &QDialog::reject);
 | |
| 
 | |
|   connect(co_compl, QOverload<const QString&>::of(&QCompleter::activated),
 | |
|           [=](const QString& text) {
 | |
|             combo_figlist->setCurrentText(text);
 | |
|             combo_figlist->setCurrentIndex(combo_figlist->findText(text));
 | |
|           });
 | |
| }
 | |
| 
 | |
| QString CreateFigureDialog::GetFilePath() const
 | |
| {
 | |
|   return m_file_path;
 | |
| }
 | |
| 
 | |
| void InfinityBaseWindow::EmulateBase(bool emulate)
 | |
| {
 | |
|   Config::SetBaseOrCurrent(Config::MAIN_EMULATE_INFINITY_BASE, emulate);
 | |
|   m_group_figures->setVisible(emulate);
 | |
| }
 | |
| 
 | |
| void InfinityBaseWindow::OnEmulationStateChanged(Core::State state)
 | |
| {
 | |
|   const bool running = state != Core::State::Uninitialized;
 | |
| 
 | |
|   m_checkbox->setEnabled(!running);
 | |
| }
 |