mirror of
https://github.com/DarkMatterCore/nxdumptool.git
synced 2025-01-10 03:27:23 -03:00
50deeeb41b
The directory layout is partially based on the C++ namespaces we're currently using. Other changes include: * devoptab: move directory into "core". * fatfs: move directory into "core". * GameCardTab: move portions of logic from PopulateList() into their own methods. * GameCardTab: use a macro to generate the properties table. * GameCardTab: use a macro to add ListItem elements. * GameCardTab: update AddApplicationMetadataItems() method to also display the number of DLCs available in the inserted gamecard for each application whenever possible. * Makefile: remove all extra entries from the INCLUDES variable. * nxdt_includes: move HOS version structs into their own header file. * tasks: move code for each individual task into its own file(s). * title: update titleGetGameCardApplicationMetadataEntries() to also count the number of DLCs available in the inserted gamecard for any given base application. * title: reorder gamecard application metadata entries by name before returning the buffer in titleGetGameCardApplicationMetadataEntries().
257 lines
11 KiB
C++
257 lines
11 KiB
C++
/*
|
|
* dump_options_frame.cpp
|
|
*
|
|
* Copyright (c) 2020-2024, DarkMatterCore <pabloacurielz@gmail.com>.
|
|
*
|
|
* This file is part of nxdumptool (https://github.com/DarkMatterCore/nxdumptool).
|
|
*
|
|
* nxdumptool is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* nxdumptool is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <views/dump_options_frame.hpp>
|
|
|
|
namespace i18n = brls::i18n; /* For getStr(). */
|
|
using namespace i18n::literals; /* For _i18n. */
|
|
|
|
namespace nxdt::views
|
|
{
|
|
DumpOptionsFrame::DumpOptionsFrame(RootView *root_view, const std::string& title, const std::string& base_output_path, const std::string& raw_filename) :
|
|
brls::ThumbnailFrame(), root_view(root_view), base_output_path(base_output_path), raw_filename(raw_filename)
|
|
{
|
|
/* Generate icon using the default image. */
|
|
brls::Image *icon = new brls::Image();
|
|
icon->setImage(BOREALIS_ASSET("icon/" APP_TITLE ".jpg"));
|
|
icon->setScaleType(brls::ImageScaleType::SCALE);
|
|
|
|
/* Initialize the rest of the elements. */
|
|
this->Initialize(title, icon);
|
|
}
|
|
|
|
DumpOptionsFrame::DumpOptionsFrame(RootView *root_view, const std::string& title, brls::Image *icon, const std::string& base_output_path, const std::string& raw_filename) :
|
|
brls::ThumbnailFrame(), root_view(root_view), base_output_path(base_output_path), raw_filename(raw_filename)
|
|
{
|
|
/* Initialize the rest of the elements. */
|
|
this->Initialize(title, icon);
|
|
}
|
|
|
|
DumpOptionsFrame::~DumpOptionsFrame()
|
|
{
|
|
/* Unregister all button click event listeners. */
|
|
this->button_click_event->unsubscribeAll();
|
|
|
|
/* Unregister UMS task listener. */
|
|
this->root_view->UnregisterUmsTaskListener(this->ums_task_sub);
|
|
}
|
|
|
|
void DumpOptionsFrame::Initialize(const std::string& title, brls::Image *icon)
|
|
{
|
|
/* Set UI properties. */
|
|
this->setTitle(title);
|
|
this->setIcon(icon);
|
|
|
|
/* Generate List view. */
|
|
this->list = new brls::List();
|
|
this->list->setSpacing(this->list->getSpacing() / 2);
|
|
this->list->setMarginBottom(20);
|
|
|
|
/* Filename. */
|
|
this->filename = new brls::InputListItem("dump_options/filename/label"_i18n, this->SanitizeUserFileName(), "", "dump_options/filename/description"_i18n,
|
|
FS_MAX_FILENAME_LENGTH);
|
|
|
|
this->filename->getClickEvent()->subscribe([this](brls::View *view) {
|
|
/* Sanitize the string entered by the user. */
|
|
this->raw_filename = this->filename->getValue();
|
|
this->filename->setValue(this->SanitizeUserFileName());
|
|
});
|
|
|
|
this->list->addView(this->filename);
|
|
|
|
/* Output storage. */
|
|
this->output_storage = new brls::SelectListItem("dump_options/output_storage/label"_i18n, this->GenerateOutputStoragesVector(this->root_view->GetUmsDevices()),
|
|
static_cast<u32>(this->root_view->GetOutputStorage()),
|
|
i18n::getStr("dump_options/output_storage/description", GITHUB_REPOSITORY_URL));
|
|
|
|
this->output_storage->getValueSelectedEvent()->subscribe([this](int selected) {
|
|
/* Make sure the current value isn't out of bounds. */
|
|
if (selected < ConfigOutputStorage_SdCard || selected >= static_cast<int>(this->root_view->GetUmsDevices().size() + ConfigOutputStorage_Count)) return;
|
|
|
|
/* Update configuration. */
|
|
if (selected == ConfigOutputStorage_SdCard || selected == ConfigOutputStorage_UsbHost) configSetInteger("output_storage", selected);
|
|
|
|
/* Update cached output storage value. */
|
|
this->root_view->SetOutputStorage(selected);
|
|
|
|
/* Sanitize output filename for the selected storage. */
|
|
this->filename->setValue(this->SanitizeUserFileName());
|
|
|
|
/* Update the storage prefix. */
|
|
this->UpdateStoragePrefix(static_cast<u32>(selected));
|
|
});
|
|
|
|
/* Manually update the storage prefix. */
|
|
this->UpdateStoragePrefix(this->output_storage->getSelectedValue());
|
|
|
|
this->list->addView(this->output_storage);
|
|
|
|
/* Subscribe to the UMS device event. */
|
|
this->ums_task_sub = this->root_view->RegisterUmsTaskListener([this](const nxdt::tasks::UmsDeviceVector& ums_devices) {
|
|
/* Update SelectListItem values. */
|
|
/* If the dropdown menu currently is being displayed, it'll be reloaded. */
|
|
this->output_storage->updateValues(this->GenerateOutputStoragesVector(ums_devices));
|
|
|
|
/* Get selected output storage. */
|
|
u32 selected = this->output_storage->getSelectedValue();
|
|
if (selected > ConfigOutputStorage_UsbHost)
|
|
{
|
|
/* Switch the current output storage to the SD card if a UMS device was previously selected. */
|
|
this->output_storage->setSelectedValue(ConfigOutputStorage_SdCard);
|
|
|
|
/* Manually trigger selection event. */
|
|
/* This will take care of updating the JSON configuration, updating the cached output storage value and saniziting the filename provided by the user. */
|
|
this->output_storage->getValueSelectedEvent()->fire(ConfigOutputStorage_SdCard);
|
|
} else {
|
|
/* Set the current output storage once more. This will make sure the string for the selected storage gets updated. */
|
|
this->output_storage->setSelectedValue(selected);
|
|
}
|
|
});
|
|
|
|
/* Start dump button. */
|
|
brls::Button *button = this->getSidebar()->getButton();
|
|
button->setLabel("dump_options/start_dump"_i18n);
|
|
|
|
/* Retrieve button event pointer. */
|
|
this->button_click_event = button->getClickEvent();
|
|
|
|
/* Set content view. */
|
|
this->setContentView(this->list);
|
|
}
|
|
|
|
std::string DumpOptionsFrame::SanitizeUserFileName(void)
|
|
{
|
|
char *raw_filename_dup = nullptr;
|
|
|
|
if (this->raw_filename.empty() || !(raw_filename_dup = strdup(this->raw_filename.c_str()))) return "dummy";
|
|
|
|
int selected = this->root_view->GetOutputStorage();
|
|
utilsReplaceIllegalCharacters(raw_filename_dup, selected == ConfigOutputStorage_SdCard);
|
|
|
|
std::string output = std::string(raw_filename_dup);
|
|
free(raw_filename_dup);
|
|
|
|
return output;
|
|
}
|
|
|
|
std::vector<std::string> DumpOptionsFrame::GenerateOutputStoragesVector(const nxdt::tasks::UmsDeviceVector& ums_devices)
|
|
{
|
|
std::vector<std::string> storages{};
|
|
size_t elem_count = (ConfigOutputStorage_Count + ums_devices.size());
|
|
|
|
/* Fill storages vector. */
|
|
for(size_t i = 0; i < elem_count; i++)
|
|
{
|
|
if (i == 1)
|
|
{
|
|
storages.push_back("dump_options/output_storage/value_01"_i18n);
|
|
continue;
|
|
}
|
|
|
|
u64 total_sz = 0, free_sz = 0;
|
|
char total_sz_str[0x40] = {0}, free_sz_str[0x40] = {0};
|
|
|
|
const nxdt::tasks::UmsDeviceVectorEntry *ums_device_entry = (i >= ConfigOutputStorage_Count ? &(ums_devices.at(i - ConfigOutputStorage_Count)) : nullptr);
|
|
const UsbHsFsDevice *cur_ums_device = (ums_device_entry ? ums_device_entry->first : nullptr);
|
|
|
|
sprintf(total_sz_str, "%s/", cur_ums_device ? cur_ums_device->name : DEVOPTAB_SDMC_DEVICE);
|
|
utilsGetFileSystemStatsByPath(total_sz_str, &total_sz, &free_sz);
|
|
utilsGenerateFormattedSizeString(static_cast<double>(total_sz), total_sz_str, sizeof(total_sz_str));
|
|
utilsGenerateFormattedSizeString(static_cast<double>(free_sz), free_sz_str, sizeof(free_sz_str));
|
|
|
|
if (cur_ums_device)
|
|
{
|
|
storages.push_back(i18n::getStr("dump_options/output_storage/value_02", ums_device_entry->second, free_sz_str, total_sz_str));
|
|
} else {
|
|
storages.push_back(i18n::getStr("dump_options/output_storage/value_00", free_sz_str, total_sz_str));
|
|
}
|
|
}
|
|
|
|
return storages;
|
|
}
|
|
|
|
void DumpOptionsFrame::UpdateStoragePrefix(u32 selected)
|
|
{
|
|
switch(selected)
|
|
{
|
|
case ConfigOutputStorage_SdCard:
|
|
this->storage_prefix = DEVOPTAB_SDMC_DEVICE "/";
|
|
break;
|
|
case ConfigOutputStorage_UsbHost:
|
|
this->storage_prefix = "/";
|
|
break;
|
|
default:
|
|
{
|
|
const nxdt::tasks::UmsDeviceVector& ums_devices = this->root_view->GetUmsDevices();
|
|
this->storage_prefix = std::string(ums_devices.at(selected - ConfigOutputStorage_Count).first->name);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
bool DumpOptionsFrame::onCancel(void)
|
|
{
|
|
/* Pop view. This will invoke this class' destructor. */
|
|
brls::Application::popView(brls::ViewAnimation::SLIDE_RIGHT);
|
|
return true;
|
|
}
|
|
|
|
void DumpOptionsFrame::addView(brls::View *view, bool fill)
|
|
{
|
|
this->list->addView(view, fill);
|
|
}
|
|
|
|
bool DumpOptionsFrame::GetOutputFilePath(const std::string& extension, std::string& output)
|
|
{
|
|
std::string tmp = this->storage_prefix;
|
|
u32 selected = this->output_storage->getSelectedValue();
|
|
char *sanitized_path = nullptr;
|
|
|
|
if (selected == ConfigOutputStorage_SdCard || selected >= ConfigOutputStorage_Count)
|
|
{
|
|
/* Remove the trailing path separator (if available) and append the application's base path if we're dealing with an SD card or a UMS device. */
|
|
if (tmp.back() == '/') tmp.pop_back();
|
|
tmp += APP_BASE_PATH;
|
|
}
|
|
|
|
/* Append a path separator, if needed. */
|
|
if (tmp.back() != '/' && this->base_output_path.front() != '/') tmp.push_back('/');
|
|
|
|
/* Append the base output path string. */
|
|
tmp += this->base_output_path;
|
|
|
|
/* Generate the sanitized file path. */
|
|
sanitized_path = utilsGeneratePath(tmp.c_str(), this->filename->getValue().c_str(), extension.c_str());
|
|
if (!sanitized_path)
|
|
{
|
|
brls::Application::notify("dump_options/notifications/get_output_path_error"_i18n);
|
|
return false;
|
|
}
|
|
|
|
/* Update output. */
|
|
output = std::string(sanitized_path);
|
|
|
|
/* Free sanitized path. */
|
|
free(sanitized_path);
|
|
|
|
return true;
|
|
}
|
|
}
|