2019-02-09 17:00:57 +01:00
|
|
|
// Copyright 2018 Citra Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
#include <QComboBox>
|
|
|
|
#include <QDialogButtonBox>
|
|
|
|
#include <QMessageBox>
|
|
|
|
#include <QString>
|
|
|
|
#include <QVBoxLayout>
|
|
|
|
#include "citra_qt/applets/mii_selector.h"
|
|
|
|
#include "common/file_util.h"
|
2019-04-07 15:58:42 +02:00
|
|
|
#include "common/string_util.h"
|
2019-02-09 17:00:57 +01:00
|
|
|
#include "core/file_sys/archive_extsavedata.h"
|
|
|
|
#include "core/file_sys/file_backend.h"
|
|
|
|
#include "core/hle/service/ptm/ptm.h"
|
|
|
|
|
2019-04-07 15:58:42 +02:00
|
|
|
/**
|
|
|
|
* Converts a UTF-16 text in a container to a UTF-8 std::string.
|
|
|
|
*/
|
|
|
|
template <typename T>
|
|
|
|
std::string TextFromBuffer(const T& text) {
|
|
|
|
const auto text_end = std::find(text.begin(), text.end(), u'\0');
|
|
|
|
const std::size_t text_size = std::distance(text.begin(), text_end);
|
|
|
|
std::u16string buffer(text_size, 0);
|
|
|
|
std::transform(text.begin(), text_end, buffer.begin(), [](u16_le character) {
|
|
|
|
return static_cast<char16_t>(static_cast<u16>(character));
|
|
|
|
});
|
|
|
|
return Common::UTF16ToUTF8(buffer);
|
|
|
|
}
|
|
|
|
|
2019-02-09 17:00:57 +01:00
|
|
|
QtMiiSelectorDialog::QtMiiSelectorDialog(QWidget* parent, QtMiiSelector* mii_selector_)
|
|
|
|
: QDialog(parent), mii_selector(mii_selector_) {
|
2019-02-15 19:20:06 +01:00
|
|
|
using namespace Frontend;
|
|
|
|
const auto config = mii_selector->config;
|
2019-02-09 17:00:57 +01:00
|
|
|
layout = new QVBoxLayout;
|
|
|
|
combobox = new QComboBox;
|
|
|
|
buttons = new QDialogButtonBox;
|
|
|
|
// Initialize buttons
|
2019-02-15 19:20:06 +01:00
|
|
|
buttons->addButton(tr(MII_BUTTON_OKAY), QDialogButtonBox::ButtonRole::AcceptRole);
|
2019-02-09 17:00:57 +01:00
|
|
|
if (config.enable_cancel_button) {
|
2019-02-15 19:20:06 +01:00
|
|
|
buttons->addButton(tr(MII_BUTTON_CANCEL), QDialogButtonBox::ButtonRole::RejectRole);
|
2019-02-09 17:00:57 +01:00
|
|
|
}
|
|
|
|
|
2019-02-26 23:36:58 +01:00
|
|
|
setWindowTitle(config.title.empty() || config.title.at(0) == '\x0000'
|
|
|
|
? tr("Mii Selector")
|
2019-04-07 15:58:42 +02:00
|
|
|
: QString::fromStdString(config.title));
|
2019-02-09 17:00:57 +01:00
|
|
|
|
2019-02-15 19:20:06 +01:00
|
|
|
miis.push_back(HLE::Applets::MiiSelector::GetStandardMiiResult().selected_mii_data);
|
|
|
|
combobox->addItem(tr("Standard Mii"));
|
|
|
|
|
2019-02-09 17:00:57 +01:00
|
|
|
std::string nand_directory{FileUtil::GetUserPath(FileUtil::UserPath::NANDDir)};
|
|
|
|
FileSys::ArchiveFactory_ExtSaveData extdata_archive_factory(nand_directory, true);
|
|
|
|
|
2019-02-26 23:36:58 +01:00
|
|
|
auto archive_result = extdata_archive_factory.Open(Service::PTM::ptm_shared_extdata_id, 0);
|
2019-02-15 19:20:06 +01:00
|
|
|
if (archive_result.Succeeded()) {
|
|
|
|
auto archive = std::move(archive_result).Unwrap();
|
|
|
|
|
|
|
|
FileSys::Path file_path = "/CFL_DB.dat";
|
|
|
|
FileSys::Mode mode{};
|
|
|
|
mode.read_flag.Assign(1);
|
|
|
|
|
|
|
|
auto file_result = archive->OpenFile(file_path, mode);
|
|
|
|
if (file_result.Succeeded()) {
|
|
|
|
auto file = std::move(file_result).Unwrap();
|
|
|
|
|
|
|
|
u32 saved_miis_offset = 0x8;
|
|
|
|
// The Mii Maker has a 100 Mii limit on the 3ds
|
|
|
|
for (int i = 0; i < 100; ++i) {
|
|
|
|
HLE::Applets::MiiData mii;
|
|
|
|
std::array<u8, sizeof(mii)> mii_raw;
|
|
|
|
file->Read(saved_miis_offset, sizeof(mii), mii_raw.data());
|
|
|
|
std::memcpy(&mii, mii_raw.data(), sizeof(mii));
|
|
|
|
if (mii.mii_id != 0) {
|
2019-04-07 15:58:42 +02:00
|
|
|
std::string name = TextFromBuffer(mii.mii_name);
|
2019-02-15 19:20:06 +01:00
|
|
|
miis.push_back(mii);
|
2019-04-07 15:58:42 +02:00
|
|
|
combobox->addItem(QString::fromStdString(name));
|
2019-02-15 19:20:06 +01:00
|
|
|
}
|
|
|
|
saved_miis_offset += sizeof(mii);
|
|
|
|
}
|
2019-02-09 17:00:57 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (combobox->count() > static_cast<int>(config.initially_selected_mii_index)) {
|
|
|
|
combobox->setCurrentIndex(static_cast<int>(config.initially_selected_mii_index));
|
|
|
|
}
|
|
|
|
|
|
|
|
connect(buttons, &QDialogButtonBox::accepted, this, [this] { accept(); });
|
|
|
|
connect(buttons, &QDialogButtonBox::rejected, this, [this] {
|
|
|
|
return_code = 1;
|
|
|
|
accept();
|
|
|
|
});
|
|
|
|
layout->addWidget(combobox);
|
|
|
|
layout->addWidget(buttons);
|
|
|
|
setLayout(layout);
|
|
|
|
}
|
|
|
|
|
|
|
|
QtMiiSelector::QtMiiSelector(QWidget& parent_) : parent(parent_) {}
|
|
|
|
|
2019-02-15 19:20:06 +01:00
|
|
|
void QtMiiSelector::Setup(const Frontend::MiiSelectorConfig& config) {
|
2019-02-09 17:00:57 +01:00
|
|
|
MiiSelector::Setup(config);
|
|
|
|
QMetaObject::invokeMethod(this, "OpenDialog", Qt::BlockingQueuedConnection);
|
|
|
|
}
|
|
|
|
|
|
|
|
void QtMiiSelector::OpenDialog() {
|
|
|
|
QtMiiSelectorDialog dialog(&parent, this);
|
|
|
|
dialog.setWindowFlags(Qt::Dialog | Qt::CustomizeWindowHint | Qt::WindowTitleHint |
|
|
|
|
Qt::WindowSystemMenuHint | Qt::WindowCloseButtonHint);
|
|
|
|
dialog.setWindowModality(Qt::WindowModal);
|
|
|
|
dialog.exec();
|
|
|
|
|
2019-02-15 19:20:06 +01:00
|
|
|
const auto index = dialog.combobox->currentIndex();
|
2019-02-09 17:00:57 +01:00
|
|
|
LOG_INFO(Frontend, "Mii Selector dialog finished (return_code={}, index={})",
|
|
|
|
dialog.return_code, index);
|
|
|
|
|
2019-02-15 19:20:06 +01:00
|
|
|
const auto mii_data = dialog.miis.at(index);
|
|
|
|
Finalize(dialog.return_code,
|
|
|
|
dialog.return_code == 0 ? std::move(mii_data) : HLE::Applets::MiiData{});
|
2019-02-09 17:00:57 +01:00
|
|
|
}
|