340 lines
12 KiB
C++
340 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/QtUtils/SetWindowDecorations.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"));
|
|
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);
|
|
SetQWidgetWindowDecorations(&create_dlg);
|
|
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);
|
|
}
|