mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-10-24 17:09:06 +00:00
353 lines
12 KiB
C++
353 lines
12 KiB
C++
// Copyright 2021 Dolphin Emulator Project
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
#include "DolphinQt/RiivolutionBootWidget.h"
|
|
|
|
#include <unordered_map>
|
|
|
|
#include <fmt/format.h>
|
|
|
|
#include <QComboBox>
|
|
#include <QDialogButtonBox>
|
|
#include <QDir>
|
|
#include <QFileDialog>
|
|
#include <QFileInfo>
|
|
#include <QGridLayout>
|
|
#include <QGroupBox>
|
|
#include <QLabel>
|
|
#include <QLineEdit>
|
|
#include <QMetaType>
|
|
#include <QPushButton>
|
|
#include <QScrollArea>
|
|
#include <QVBoxLayout>
|
|
|
|
#include "Common/FileSearch.h"
|
|
#include "Common/FileUtil.h"
|
|
#include "DiscIO/GameModDescriptor.h"
|
|
#include "DiscIO/RiivolutionParser.h"
|
|
#include "DiscIO/RiivolutionPatcher.h"
|
|
#include "DolphinQt/Config/HardcoreWarningWidget.h"
|
|
#include "DolphinQt/QtUtils/ModalMessageBox.h"
|
|
|
|
struct GuiRiivolutionPatchIndex
|
|
{
|
|
size_t m_disc_index;
|
|
size_t m_section_index;
|
|
size_t m_option_index;
|
|
size_t m_choice_index;
|
|
};
|
|
|
|
Q_DECLARE_METATYPE(GuiRiivolutionPatchIndex);
|
|
|
|
RiivolutionBootWidget::RiivolutionBootWidget(std::string game_id, std::optional<u16> revision,
|
|
std::optional<u8> disc, std::string base_game_path,
|
|
QWidget* parent)
|
|
: QDialog(parent), m_game_id(std::move(game_id)), m_revision(revision), m_disc_number(disc),
|
|
m_base_game_path(std::move(base_game_path))
|
|
{
|
|
setWindowTitle(tr("Start with Riivolution Patches"));
|
|
|
|
CreateWidgets();
|
|
ConnectWidgets();
|
|
LoadMatchingXMLs();
|
|
|
|
resize(QSize(400, 600));
|
|
}
|
|
|
|
RiivolutionBootWidget::~RiivolutionBootWidget() = default;
|
|
|
|
void RiivolutionBootWidget::CreateWidgets()
|
|
{
|
|
#ifdef USE_RETRO_ACHIEVEMENTS
|
|
m_hc_warning = new HardcoreWarningWidget(this);
|
|
#endif // USE_RETRO_ACHIEVEMENTS
|
|
auto* open_xml_button = new QPushButton(tr("Open Riivolution XML..."));
|
|
auto* boot_game_button = new QPushButton(tr("Start"));
|
|
boot_game_button->setDefault(true);
|
|
auto* save_preset_button = new QPushButton(tr("Save as Preset..."));
|
|
auto* group_box = new QGroupBox();
|
|
auto* scroll_area = new QScrollArea();
|
|
|
|
auto* stretch_helper = new QVBoxLayout();
|
|
m_patch_section_layout = new QVBoxLayout();
|
|
stretch_helper->addLayout(m_patch_section_layout);
|
|
stretch_helper->addStretch();
|
|
group_box->setLayout(stretch_helper);
|
|
scroll_area->setWidget(group_box);
|
|
scroll_area->setWidgetResizable(true);
|
|
|
|
auto* button_layout = new QHBoxLayout();
|
|
button_layout->addStretch();
|
|
button_layout->addWidget(open_xml_button, 0, Qt::AlignRight);
|
|
button_layout->addWidget(save_preset_button, 0, Qt::AlignRight);
|
|
button_layout->addWidget(boot_game_button, 0, Qt::AlignRight);
|
|
|
|
auto* layout = new QVBoxLayout();
|
|
#ifdef USE_RETRO_ACHIEVEMENTS
|
|
layout->addWidget(m_hc_warning);
|
|
#endif // USE_RETRO_ACHIEVEMENTS
|
|
layout->addWidget(scroll_area);
|
|
layout->addLayout(button_layout);
|
|
setLayout(layout);
|
|
|
|
connect(open_xml_button, &QPushButton::clicked, this, &RiivolutionBootWidget::OpenXML);
|
|
connect(boot_game_button, &QPushButton::clicked, this, &RiivolutionBootWidget::BootGame);
|
|
connect(save_preset_button, &QPushButton::clicked, this, &RiivolutionBootWidget::SaveAsPreset);
|
|
}
|
|
|
|
void RiivolutionBootWidget::ConnectWidgets()
|
|
{
|
|
#ifdef USE_RETRO_ACHIEVEMENTS
|
|
connect(m_hc_warning, &HardcoreWarningWidget::OpenAchievementSettings, this,
|
|
&RiivolutionBootWidget::OpenAchievementSettings);
|
|
connect(m_hc_warning, &HardcoreWarningWidget::OpenAchievementSettings, this,
|
|
&RiivolutionBootWidget::reject);
|
|
#endif // USE_RETRO_ACHIEVEMENTS
|
|
}
|
|
|
|
void RiivolutionBootWidget::LoadMatchingXMLs()
|
|
{
|
|
const std::string& riivolution_dir = File::GetUserPath(D_RIIVOLUTION_IDX);
|
|
const auto config = LoadConfigXML(riivolution_dir);
|
|
for (const std::string& path : Common::DoFileSearch({riivolution_dir + "riivolution"}, {".xml"}))
|
|
{
|
|
auto parsed = DiscIO::Riivolution::ParseFile(path);
|
|
if (!parsed || !parsed->IsValidForGame(m_game_id, m_revision, m_disc_number))
|
|
continue;
|
|
if (config)
|
|
DiscIO::Riivolution::ApplyConfigDefaults(&*parsed, *config);
|
|
MakeGUIForParsedFile(path, riivolution_dir, *parsed);
|
|
}
|
|
}
|
|
|
|
static std::string FindRoot(const std::string& path)
|
|
{
|
|
// Try to set the virtual SD root to directory one up from current.
|
|
// This mimics where the XML would be on a real SD card.
|
|
QDir dir = QFileInfo(QString::fromStdString(path)).dir();
|
|
if (dir.cdUp())
|
|
return dir.absolutePath().toStdString();
|
|
return File::GetUserPath(D_RIIVOLUTION_IDX);
|
|
}
|
|
|
|
void RiivolutionBootWidget::OpenXML()
|
|
{
|
|
const std::string& riivolution_dir = File::GetUserPath(D_RIIVOLUTION_IDX);
|
|
QStringList paths = QFileDialog::getOpenFileNames(
|
|
this, tr("Select Riivolution XML file"), QString::fromStdString(riivolution_dir),
|
|
QStringLiteral("%1 (*.xml);;%2 (*)").arg(tr("Riivolution XML files")).arg(tr("All Files")));
|
|
if (paths.isEmpty())
|
|
return;
|
|
|
|
for (const QString& path : paths)
|
|
{
|
|
std::string p = path.toStdString();
|
|
auto parsed = DiscIO::Riivolution::ParseFile(p);
|
|
if (!parsed)
|
|
{
|
|
ModalMessageBox::warning(
|
|
this, tr("Failed loading XML."),
|
|
tr("Did not recognize %1 as a valid Riivolution XML file.").arg(path));
|
|
continue;
|
|
}
|
|
|
|
if (!parsed->IsValidForGame(m_game_id, m_revision, m_disc_number))
|
|
{
|
|
ModalMessageBox::warning(
|
|
this, tr("Invalid game."),
|
|
tr("The patches in %1 are not for the selected game or game revision.").arg(path));
|
|
continue;
|
|
}
|
|
|
|
auto root = FindRoot(p);
|
|
const auto config = LoadConfigXML(root);
|
|
if (config)
|
|
DiscIO::Riivolution::ApplyConfigDefaults(&*parsed, *config);
|
|
MakeGUIForParsedFile(p, std::move(root), *parsed);
|
|
}
|
|
}
|
|
|
|
void RiivolutionBootWidget::MakeGUIForParsedFile(std::string path, std::string root,
|
|
DiscIO::Riivolution::Disc input_disc)
|
|
{
|
|
const size_t disc_index = m_discs.size();
|
|
const auto& disc =
|
|
m_discs.emplace_back(DiscWithRoot{std::move(input_disc), std::move(root), std::move(path)});
|
|
|
|
auto* disc_box = new QGroupBox(QFileInfo(QString::fromStdString(disc.path)).fileName());
|
|
auto* disc_layout = new QVBoxLayout();
|
|
disc_box->setLayout(disc_layout);
|
|
|
|
auto* xml_root_line_edit = new QLineEdit(QString::fromStdString(disc.root));
|
|
xml_root_line_edit->setReadOnly(true);
|
|
auto* xml_root_layout = new QHBoxLayout();
|
|
auto* xml_root_open = new QPushButton(tr("..."));
|
|
xml_root_layout->addWidget(new QLabel(tr("SD Root:")), 0);
|
|
xml_root_layout->addWidget(xml_root_line_edit, 0);
|
|
xml_root_layout->addWidget(xml_root_open, 0);
|
|
disc_layout->addLayout(xml_root_layout);
|
|
connect(xml_root_open, &QPushButton::clicked, this, [this, xml_root_line_edit, disc_index] {
|
|
QString dir = QDir::toNativeSeparators(QFileDialog::getExistingDirectory(
|
|
this, tr("Select the Virtual SD Card Root"), xml_root_line_edit->text()));
|
|
if (!dir.isEmpty())
|
|
{
|
|
xml_root_line_edit->setText(dir);
|
|
m_discs[disc_index].root = dir.toStdString();
|
|
}
|
|
});
|
|
|
|
for (size_t section_index = 0; section_index < disc.disc.m_sections.size(); ++section_index)
|
|
{
|
|
const auto& section = disc.disc.m_sections[section_index];
|
|
auto* group_box = new QGroupBox(QString::fromStdString(section.m_name));
|
|
auto* grid_layout = new QGridLayout();
|
|
group_box->setLayout(grid_layout);
|
|
|
|
int row = 0;
|
|
for (size_t option_index = 0; option_index < section.m_options.size(); ++option_index)
|
|
{
|
|
const auto& option = section.m_options[option_index];
|
|
auto* label = new QLabel(QString::fromStdString(option.m_name));
|
|
auto* selection = new QComboBox();
|
|
const GuiRiivolutionPatchIndex gui_disabled_index{disc_index, section_index, option_index, 0};
|
|
selection->addItem(tr("Disabled"), QVariant::fromValue(gui_disabled_index));
|
|
for (size_t choice_index = 0; choice_index < option.m_choices.size(); ++choice_index)
|
|
{
|
|
const auto& choice = option.m_choices[choice_index];
|
|
const GuiRiivolutionPatchIndex gui_index{disc_index, section_index, option_index,
|
|
choice_index + 1};
|
|
selection->addItem(QString::fromStdString(choice.m_name), QVariant::fromValue(gui_index));
|
|
}
|
|
if (option.m_selected_choice <= option.m_choices.size())
|
|
selection->setCurrentIndex(static_cast<int>(option.m_selected_choice));
|
|
|
|
connect(selection, &QComboBox::currentIndexChanged, this, [this, selection](int idx) {
|
|
const auto gui_index = selection->currentData().value<GuiRiivolutionPatchIndex>();
|
|
auto& selected_disc = m_discs[gui_index.m_disc_index].disc;
|
|
auto& selected_section = selected_disc.m_sections[gui_index.m_section_index];
|
|
auto& selected_option = selected_section.m_options[gui_index.m_option_index];
|
|
selected_option.m_selected_choice = static_cast<u32>(gui_index.m_choice_index);
|
|
});
|
|
|
|
grid_layout->addWidget(label, row, 0, 1, 1);
|
|
grid_layout->addWidget(selection, row, 1, 1, 1);
|
|
++row;
|
|
}
|
|
|
|
disc_layout->addWidget(group_box);
|
|
}
|
|
|
|
m_patch_section_layout->addWidget(disc_box);
|
|
}
|
|
|
|
std::optional<DiscIO::Riivolution::Config>
|
|
RiivolutionBootWidget::LoadConfigXML(const std::string& root_directory)
|
|
{
|
|
// The way Riivolution stores settings only makes sense for standard game IDs.
|
|
if (!(m_game_id.size() == 4 || m_game_id.size() == 6))
|
|
return std::nullopt;
|
|
|
|
return DiscIO::Riivolution::ParseConfigFile(
|
|
fmt::format("{}/riivolution/config/{}.xml", root_directory, m_game_id.substr(0, 4)));
|
|
}
|
|
|
|
void RiivolutionBootWidget::SaveConfigXMLs()
|
|
{
|
|
if (!(m_game_id.size() == 4 || m_game_id.size() == 6))
|
|
return;
|
|
|
|
std::unordered_map<std::string, DiscIO::Riivolution::Config> map;
|
|
for (const auto& disc : m_discs)
|
|
{
|
|
auto config = map.try_emplace(disc.root);
|
|
auto& config_options = config.first->second.m_options;
|
|
for (const auto& section : disc.disc.m_sections)
|
|
{
|
|
for (const auto& option : section.m_options)
|
|
{
|
|
std::string id = option.m_id.empty() ? (section.m_name + option.m_name) : option.m_id;
|
|
config_options.emplace_back(
|
|
DiscIO::Riivolution::ConfigOption{std::move(id), option.m_selected_choice});
|
|
}
|
|
}
|
|
}
|
|
|
|
for (const auto& config : map)
|
|
{
|
|
DiscIO::Riivolution::WriteConfigFile(
|
|
fmt::format("{}/riivolution/config/{}.xml", config.first, m_game_id.substr(0, 4)),
|
|
config.second);
|
|
}
|
|
}
|
|
|
|
void RiivolutionBootWidget::BootGame()
|
|
{
|
|
SaveConfigXMLs();
|
|
|
|
m_patches.clear();
|
|
for (const auto& disc : m_discs)
|
|
{
|
|
auto patches = disc.disc.GeneratePatches(m_game_id);
|
|
|
|
// set the file loader for each patch
|
|
for (auto& patch : patches)
|
|
{
|
|
patch.m_file_data_loader = std::make_shared<DiscIO::Riivolution::FileDataLoaderHostFS>(
|
|
disc.root, disc.disc.m_xml_path, patch.m_root);
|
|
}
|
|
|
|
m_patches.insert(m_patches.end(), patches.begin(), patches.end());
|
|
}
|
|
|
|
m_should_boot = true;
|
|
close();
|
|
}
|
|
|
|
void RiivolutionBootWidget::SaveAsPreset()
|
|
{
|
|
DiscIO::GameModDescriptor descriptor;
|
|
descriptor.base_file = m_base_game_path;
|
|
|
|
DiscIO::GameModDescriptorRiivolution riivolution_descriptor;
|
|
for (const auto& disc : m_discs)
|
|
{
|
|
// filter out XMLs that don't actually contribute to the preset
|
|
auto patches = disc.disc.GeneratePatches(m_game_id);
|
|
if (patches.empty())
|
|
continue;
|
|
|
|
auto& descriptor_patch = riivolution_descriptor.patches.emplace_back();
|
|
descriptor_patch.xml = disc.path;
|
|
descriptor_patch.root = disc.root;
|
|
for (const auto& section : disc.disc.m_sections)
|
|
{
|
|
for (const auto& option : section.m_options)
|
|
{
|
|
auto& descriptor_option = descriptor_patch.options.emplace_back();
|
|
descriptor_option.section_name = section.m_name;
|
|
if (!option.m_id.empty())
|
|
descriptor_option.option_id = option.m_id;
|
|
else
|
|
descriptor_option.option_name = option.m_name;
|
|
descriptor_option.choice = option.m_selected_choice;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!riivolution_descriptor.patches.empty())
|
|
descriptor.riivolution = std::move(riivolution_descriptor);
|
|
|
|
QDir dir = QFileInfo(QString::fromStdString(m_base_game_path)).dir();
|
|
QString target_path = QFileDialog::getSaveFileName(this, tr("Save Preset"), dir.absolutePath(),
|
|
QStringLiteral("%1 (*.json);;%2 (*)")
|
|
.arg(tr("Dolphin Game Mod Preset"))
|
|
.arg(tr("All Files")));
|
|
if (target_path.isEmpty())
|
|
return;
|
|
|
|
descriptor.display_name = QFileInfo(target_path).fileName().toStdString();
|
|
auto dot = descriptor.display_name.rfind('.');
|
|
if (dot != std::string::npos)
|
|
descriptor.display_name = descriptor.display_name.substr(0, dot);
|
|
DiscIO::WriteGameModDescriptorFile(target_path.toStdString(), descriptor, true);
|
|
}
|