This is part of a series of PRs made in preparation for the QML rewrite. this PR specifically moves a bunch of utility functions from main.cpp into qt_common, with the biggest benefit being that QML can reuse the exact same code through ctx passthrough. Also, QtCommon::Frontend is an abstraction layer over several previously Widgets-specific stuff like QMessageBox that gets used everywhere. The idea is that once QML is implemented, these functions can have a Quick version implemented for systems that don't work well with Widgets (sun) or for those on Plasma 6+ (reduces memory usage w/o Widgets linkage) although Quick from C++ is actually anal, but whatever. Other than that this should also just kinda reduce the size of main.cpp which is a 6000-line behemoth rn, and clangd straight up gives up with it for me (likely caused by the massive amount of headers, which this DOES reduce). In the future, I probably want to create a common strings lookup table that both Qt and QML can reference--though I'm not sure how much linguist likes that--which should give us a way to keep language consistent (use frozen-map). TODO: Docs for Qt stuff Co-authored-by: MaranBr <maranbr@outlook.com> Reviewed-on: #94 Reviewed-by: MaranBr <maranbr@eden-emu.dev> Reviewed-by: Shinmegumi <shinmegumi@eden-emu.dev>
577 lines
22 KiB
C++
577 lines
22 KiB
C++
// SPDX-FileCopyrightText: Copyright 2025 Eden Emulator Project
|
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
|
|
|
#include "qt_game_util.h"
|
|
#include "common/fs/fs.h"
|
|
#include "common/fs/path_util.h"
|
|
#include "core/file_sys/savedata_factory.h"
|
|
#include "core/hle/service/am/am_types.h"
|
|
#include "frontend_common/content_manager.h"
|
|
#include "qt_common.h"
|
|
#include "qt_common/uisettings.h"
|
|
#include "qt_frontend_util.h"
|
|
#include "yuzu/util/util.h"
|
|
|
|
#include <QDesktopServices>
|
|
#include <QStandardPaths>
|
|
#include <QUrl>
|
|
|
|
#ifdef _WIN32
|
|
#include "common/scope_exit.h"
|
|
#include "common/string_util.h"
|
|
#include <shlobj.h>
|
|
#include <windows.h>
|
|
#else
|
|
#include "fmt/ostream.h"
|
|
#include <fstream>
|
|
#endif
|
|
|
|
namespace QtCommon::Game {
|
|
|
|
bool CreateShortcutLink(const std::filesystem::path& shortcut_path,
|
|
const std::string& comment,
|
|
const std::filesystem::path& icon_path,
|
|
const std::filesystem::path& command,
|
|
const std::string& arguments,
|
|
const std::string& categories,
|
|
const std::string& keywords,
|
|
const std::string& name)
|
|
try {
|
|
#ifdef _WIN32 // Windows
|
|
HRESULT hr = CoInitialize(nullptr);
|
|
if (FAILED(hr)) {
|
|
LOG_ERROR(Frontend, "CoInitialize failed");
|
|
return false;
|
|
}
|
|
SCOPE_EXIT
|
|
{
|
|
CoUninitialize();
|
|
};
|
|
IShellLinkW* ps1 = nullptr;
|
|
IPersistFile* persist_file = nullptr;
|
|
SCOPE_EXIT
|
|
{
|
|
if (persist_file != nullptr) {
|
|
persist_file->Release();
|
|
}
|
|
if (ps1 != nullptr) {
|
|
ps1->Release();
|
|
}
|
|
};
|
|
HRESULT hres = CoCreateInstance(CLSID_ShellLink,
|
|
nullptr,
|
|
CLSCTX_INPROC_SERVER,
|
|
IID_IShellLinkW,
|
|
reinterpret_cast<void**>(&ps1));
|
|
if (FAILED(hres)) {
|
|
LOG_ERROR(Frontend, "Failed to create IShellLinkW instance");
|
|
return false;
|
|
}
|
|
hres = ps1->SetPath(command.c_str());
|
|
if (FAILED(hres)) {
|
|
LOG_ERROR(Frontend, "Failed to set path");
|
|
return false;
|
|
}
|
|
if (!arguments.empty()) {
|
|
hres = ps1->SetArguments(Common::UTF8ToUTF16W(arguments).data());
|
|
if (FAILED(hres)) {
|
|
LOG_ERROR(Frontend, "Failed to set arguments");
|
|
return false;
|
|
}
|
|
}
|
|
if (!comment.empty()) {
|
|
hres = ps1->SetDescription(Common::UTF8ToUTF16W(comment).data());
|
|
if (FAILED(hres)) {
|
|
LOG_ERROR(Frontend, "Failed to set description");
|
|
return false;
|
|
}
|
|
}
|
|
if (std::filesystem::is_regular_file(icon_path)) {
|
|
hres = ps1->SetIconLocation(icon_path.c_str(), 0);
|
|
if (FAILED(hres)) {
|
|
LOG_ERROR(Frontend, "Failed to set icon location");
|
|
return false;
|
|
}
|
|
}
|
|
hres = ps1->QueryInterface(IID_IPersistFile, reinterpret_cast<void**>(&persist_file));
|
|
if (FAILED(hres)) {
|
|
LOG_ERROR(Frontend, "Failed to get IPersistFile interface");
|
|
return false;
|
|
}
|
|
hres = persist_file->Save(std::filesystem::path{shortcut_path / (name + ".lnk")}.c_str(), TRUE);
|
|
if (FAILED(hres)) {
|
|
LOG_ERROR(Frontend, "Failed to save shortcut");
|
|
return false;
|
|
}
|
|
return true;
|
|
#elif defined(__unix__) && !defined(__APPLE__) && !defined(__ANDROID__) // Any desktop NIX
|
|
std::filesystem::path shortcut_path_full = shortcut_path / (name + ".desktop");
|
|
std::ofstream shortcut_stream(shortcut_path_full, std::ios::binary | std::ios::trunc);
|
|
if (!shortcut_stream.is_open()) {
|
|
LOG_ERROR(Frontend, "Failed to create shortcut");
|
|
return false;
|
|
}
|
|
// TODO: Migrate fmt::print to std::print in futures STD C++ 23.
|
|
fmt::print(shortcut_stream, "[Desktop Entry]\n");
|
|
fmt::print(shortcut_stream, "Type=Application\n");
|
|
fmt::print(shortcut_stream, "Version=1.0\n");
|
|
fmt::print(shortcut_stream, "Name={}\n", name);
|
|
if (!comment.empty()) {
|
|
fmt::print(shortcut_stream, "Comment={}\n", comment);
|
|
}
|
|
if (std::filesystem::is_regular_file(icon_path)) {
|
|
fmt::print(shortcut_stream, "Icon={}\n", icon_path.string());
|
|
}
|
|
fmt::print(shortcut_stream, "TryExec={}\n", command.string());
|
|
fmt::print(shortcut_stream, "Exec={} {}\n", command.string(), arguments);
|
|
if (!categories.empty()) {
|
|
fmt::print(shortcut_stream, "Categories={}\n", categories);
|
|
}
|
|
if (!keywords.empty()) {
|
|
fmt::print(shortcut_stream, "Keywords={}\n", keywords);
|
|
}
|
|
return true;
|
|
#else // Unsupported platform
|
|
return false;
|
|
#endif
|
|
} catch (const std::exception& e) {
|
|
LOG_ERROR(Frontend, "Failed to create shortcut: {}", e.what());
|
|
return false;
|
|
}
|
|
|
|
bool MakeShortcutIcoPath(const u64 program_id,
|
|
const std::string_view game_file_name,
|
|
std::filesystem::path& out_icon_path)
|
|
{
|
|
// Get path to Yuzu icons directory & icon extension
|
|
std::string ico_extension = "png";
|
|
#if defined(_WIN32)
|
|
out_icon_path = Common::FS::GetEdenPath(Common::FS::EdenPath::IconsDir);
|
|
ico_extension = "ico";
|
|
#elif defined(__linux__) || defined(__FreeBSD__)
|
|
out_icon_path = Common::FS::GetDataDirectory("XDG_DATA_HOME") / "icons/hicolor/256x256";
|
|
#endif
|
|
// Create icons directory if it doesn't exist
|
|
if (!Common::FS::CreateDirs(out_icon_path)) {
|
|
out_icon_path.clear();
|
|
return false;
|
|
}
|
|
|
|
// Create icon file path
|
|
out_icon_path /= (program_id == 0 ? fmt::format("eden-{}.{}", game_file_name, ico_extension)
|
|
: fmt::format("eden-{:016X}.{}", program_id, ico_extension));
|
|
return true;
|
|
}
|
|
|
|
void OpenEdenFolder(const Common::FS::EdenPath& path)
|
|
{
|
|
QDesktopServices::openUrl(QUrl::fromLocalFile(QString::fromStdString(Common::FS::GetEdenPathString(path))));
|
|
}
|
|
|
|
void OpenRootDataFolder()
|
|
{
|
|
OpenEdenFolder(Common::FS::EdenPath::EdenDir);
|
|
}
|
|
|
|
void OpenNANDFolder()
|
|
{
|
|
OpenEdenFolder(Common::FS::EdenPath::NANDDir);
|
|
}
|
|
|
|
void OpenSDMCFolder()
|
|
{
|
|
OpenEdenFolder(Common::FS::EdenPath::SDMCDir);
|
|
}
|
|
|
|
void OpenModFolder()
|
|
{
|
|
OpenEdenFolder(Common::FS::EdenPath::LoadDir);
|
|
}
|
|
|
|
void OpenLogFolder()
|
|
{
|
|
OpenEdenFolder(Common::FS::EdenPath::LogDir);
|
|
}
|
|
|
|
static QString GetGameListErrorRemoving(QtCommon::Game::InstalledEntryType type)
|
|
{
|
|
switch (type) {
|
|
case QtCommon::Game::InstalledEntryType::Game:
|
|
return tr("Error Removing Contents");
|
|
case QtCommon::Game::InstalledEntryType::Update:
|
|
return tr("Error Removing Update");
|
|
case QtCommon::Game::InstalledEntryType::AddOnContent:
|
|
return tr("Error Removing DLC");
|
|
default:
|
|
return QStringLiteral("Error Removing <Invalid Type>");
|
|
}
|
|
}
|
|
|
|
// Game Content //
|
|
void RemoveBaseContent(u64 program_id, InstalledEntryType type)
|
|
{
|
|
const auto res = ContentManager::RemoveBaseContent(system->GetFileSystemController(),
|
|
program_id);
|
|
if (res) {
|
|
QtCommon::Frontend::Information(rootObject,
|
|
"Successfully Removed",
|
|
"Successfully removed the installed base game.");
|
|
} else {
|
|
QtCommon::Frontend::Warning(
|
|
rootObject,
|
|
GetGameListErrorRemoving(type),
|
|
tr("The base game is not installed in the NAND and cannot be removed."));
|
|
}
|
|
}
|
|
|
|
void RemoveUpdateContent(u64 program_id, InstalledEntryType type)
|
|
{
|
|
const auto res = ContentManager::RemoveUpdate(system->GetFileSystemController(), program_id);
|
|
if (res) {
|
|
QtCommon::Frontend::Information(rootObject,
|
|
"Successfully Removed",
|
|
"Successfully removed the installed update.");
|
|
} else {
|
|
QtCommon::Frontend::Warning(rootObject,
|
|
GetGameListErrorRemoving(type),
|
|
tr("There is no update installed for this title."));
|
|
}
|
|
}
|
|
|
|
void RemoveAddOnContent(u64 program_id, InstalledEntryType type)
|
|
{
|
|
const size_t count = ContentManager::RemoveAllDLC(*system, program_id);
|
|
if (count == 0) {
|
|
QtCommon::Frontend::Warning(rootObject,
|
|
GetGameListErrorRemoving(type),
|
|
tr("There are no DLCs installed for this title."));
|
|
return;
|
|
}
|
|
|
|
QtCommon::Frontend::Information(rootObject,
|
|
tr("Successfully Removed"),
|
|
tr("Successfully removed %1 installed DLC.").arg(count));
|
|
}
|
|
|
|
// Global Content //
|
|
|
|
void RemoveTransferableShaderCache(u64 program_id, GameListRemoveTarget target)
|
|
{
|
|
const auto target_file_name = [target] {
|
|
switch (target) {
|
|
case GameListRemoveTarget::GlShaderCache:
|
|
return "opengl.bin";
|
|
case GameListRemoveTarget::VkShaderCache:
|
|
return "vulkan.bin";
|
|
default:
|
|
return "";
|
|
}
|
|
}();
|
|
const auto shader_cache_dir = Common::FS::GetEdenPath(Common::FS::EdenPath::ShaderDir);
|
|
const auto shader_cache_folder_path = shader_cache_dir / fmt::format("{:016x}", program_id);
|
|
const auto target_file = shader_cache_folder_path / target_file_name;
|
|
|
|
if (!Common::FS::Exists(target_file)) {
|
|
QtCommon::Frontend::Warning(rootObject,
|
|
tr("Error Removing Transferable Shader Cache"),
|
|
tr("A shader cache for this title does not exist."));
|
|
return;
|
|
}
|
|
if (Common::FS::RemoveFile(target_file)) {
|
|
QtCommon::Frontend::Information(rootObject,
|
|
tr("Successfully Removed"),
|
|
tr("Successfully removed the transferable shader cache."));
|
|
} else {
|
|
QtCommon::Frontend::Warning(rootObject,
|
|
tr("Error Removing Transferable Shader Cache"),
|
|
tr("Failed to remove the transferable shader cache."));
|
|
}
|
|
}
|
|
|
|
void RemoveVulkanDriverPipelineCache(u64 program_id)
|
|
{
|
|
static constexpr std::string_view target_file_name = "vulkan_pipelines.bin";
|
|
|
|
const auto shader_cache_dir = Common::FS::GetEdenPath(Common::FS::EdenPath::ShaderDir);
|
|
const auto shader_cache_folder_path = shader_cache_dir / fmt::format("{:016x}", program_id);
|
|
const auto target_file = shader_cache_folder_path / target_file_name;
|
|
|
|
if (!Common::FS::Exists(target_file)) {
|
|
return;
|
|
}
|
|
if (!Common::FS::RemoveFile(target_file)) {
|
|
QtCommon::Frontend::Warning(rootObject,
|
|
tr("Error Removing Vulkan Driver Pipeline Cache"),
|
|
tr("Failed to remove the driver pipeline cache."));
|
|
}
|
|
}
|
|
|
|
void RemoveAllTransferableShaderCaches(u64 program_id)
|
|
{
|
|
const auto shader_cache_dir = Common::FS::GetEdenPath(Common::FS::EdenPath::ShaderDir);
|
|
const auto program_shader_cache_dir = shader_cache_dir / fmt::format("{:016x}", program_id);
|
|
|
|
if (!Common::FS::Exists(program_shader_cache_dir)) {
|
|
QtCommon::Frontend::Warning(rootObject,
|
|
tr("Error Removing Transferable Shader Caches"),
|
|
tr("A shader cache for this title does not exist."));
|
|
return;
|
|
}
|
|
if (Common::FS::RemoveDirRecursively(program_shader_cache_dir)) {
|
|
QtCommon::Frontend::Information(rootObject,
|
|
tr("Successfully Removed"),
|
|
tr("Successfully removed the transferable shader caches."));
|
|
} else {
|
|
QtCommon::Frontend::Warning(
|
|
rootObject,
|
|
tr("Error Removing Transferable Shader Caches"),
|
|
tr("Failed to remove the transferable shader cache directory."));
|
|
}
|
|
}
|
|
|
|
void RemoveCustomConfiguration(u64 program_id, const std::string& game_path)
|
|
{
|
|
const auto file_path = std::filesystem::path(Common::FS::ToU8String(game_path));
|
|
const auto config_file_name = program_id == 0
|
|
? Common::FS::PathToUTF8String(file_path.filename())
|
|
.append(".ini")
|
|
: fmt::format("{:016X}.ini", program_id);
|
|
const auto custom_config_file_path = Common::FS::GetEdenPath(Common::FS::EdenPath::ConfigDir)
|
|
/ "custom" / config_file_name;
|
|
|
|
if (!Common::FS::Exists(custom_config_file_path)) {
|
|
QtCommon::Frontend::Warning(rootObject,
|
|
tr("Error Removing Custom Configuration"),
|
|
tr("A custom configuration for this title does not exist."));
|
|
return;
|
|
}
|
|
|
|
if (Common::FS::RemoveFile(custom_config_file_path)) {
|
|
QtCommon::Frontend::Information(rootObject,
|
|
tr("Successfully Removed"),
|
|
tr("Successfully removed the custom game configuration."));
|
|
} else {
|
|
QtCommon::Frontend::Warning(rootObject,
|
|
tr("Error Removing Custom Configuration"),
|
|
tr("Failed to remove the custom game configuration."));
|
|
}
|
|
}
|
|
|
|
void RemoveCacheStorage(u64 program_id)
|
|
{
|
|
const auto nand_dir = Common::FS::GetEdenPath(Common::FS::EdenPath::NANDDir);
|
|
auto vfs_nand_dir = vfs->OpenDirectory(Common::FS::PathToUTF8String(nand_dir),
|
|
FileSys::OpenMode::Read);
|
|
|
|
const auto cache_storage_path
|
|
= FileSys::SaveDataFactory::GetFullPath({},
|
|
vfs_nand_dir,
|
|
FileSys::SaveDataSpaceId::User,
|
|
FileSys::SaveDataType::Cache,
|
|
0 /* program_id */,
|
|
{},
|
|
0);
|
|
|
|
const auto path = Common::FS::ConcatPathSafe(nand_dir, cache_storage_path);
|
|
|
|
// Not an error if it wasn't cleared.
|
|
Common::FS::RemoveDirRecursively(path);
|
|
}
|
|
|
|
// Metadata //
|
|
void ResetMetadata()
|
|
{
|
|
const QString title = tr("Reset Metadata Cache");
|
|
|
|
if (!Common::FS::Exists(Common::FS::GetEdenPath(Common::FS::EdenPath::CacheDir)
|
|
/ "game_list/")) {
|
|
QtCommon::Frontend::Warning(rootObject, title, tr("The metadata cache is already empty."));
|
|
} else if (Common::FS::RemoveDirRecursively(
|
|
Common::FS::GetEdenPath(Common::FS::EdenPath::CacheDir) / "game_list")) {
|
|
QtCommon::Frontend::Information(rootObject,
|
|
title,
|
|
tr("The operation completed successfully."));
|
|
UISettings::values.is_game_list_reload_pending.exchange(true);
|
|
} else {
|
|
QtCommon::Frontend::Warning(
|
|
rootObject,
|
|
title,
|
|
tr("The metadata cache couldn't be deleted. It might be in use or non-existent."));
|
|
}
|
|
}
|
|
|
|
// Uhhh //
|
|
|
|
// Messages in pre-defined message boxes for less code spaghetti
|
|
inline constexpr bool CreateShortcutMessagesGUI(ShortcutMessages imsg, const QString& game_title)
|
|
{
|
|
int result = 0;
|
|
QMessageBox::StandardButtons buttons;
|
|
switch (imsg) {
|
|
case ShortcutMessages::Fullscreen:
|
|
buttons = QMessageBox::Yes | QMessageBox::No;
|
|
result
|
|
= QtCommon::Frontend::Information(tr("Create Shortcut"),
|
|
tr("Do you want to launch the game in fullscreen?"),
|
|
buttons);
|
|
return result == QMessageBox::Yes;
|
|
case ShortcutMessages::Success:
|
|
QtCommon::Frontend::Information(tr("Shortcut Created"),
|
|
tr("Successfully created a shortcut to %1").arg(game_title));
|
|
return false;
|
|
case ShortcutMessages::Volatile:
|
|
buttons = QMessageBox::StandardButton::Ok | QMessageBox::StandardButton::Cancel;
|
|
result = QtCommon::Frontend::Warning(
|
|
tr("Shortcut may be Volatile!"),
|
|
tr("This will create a shortcut to the current AppImage. This may "
|
|
"not work well if you update. Continue?"),
|
|
buttons);
|
|
return result == QMessageBox::Ok;
|
|
default:
|
|
buttons = QMessageBox::Ok;
|
|
QtCommon::Frontend::Critical(tr("Failed to Create Shortcut"),
|
|
tr("Failed to create a shortcut to %1").arg(game_title),
|
|
buttons);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
void CreateShortcut(const std::string& game_path,
|
|
const u64 program_id,
|
|
const std::string& game_title_,
|
|
const ShortcutTarget &target,
|
|
std::string arguments_,
|
|
const bool needs_title)
|
|
{
|
|
// Get path to Eden executable
|
|
std::filesystem::path command = GetEdenCommand();
|
|
|
|
// Shortcut path
|
|
std::filesystem::path shortcut_path = GetShortcutPath(target);
|
|
|
|
if (!std::filesystem::exists(shortcut_path)) {
|
|
CreateShortcutMessagesGUI(ShortcutMessages::Failed,
|
|
QString::fromStdString(shortcut_path.generic_string()));
|
|
LOG_ERROR(Frontend, "Invalid shortcut target {}", shortcut_path.generic_string());
|
|
return;
|
|
}
|
|
|
|
const FileSys::PatchManager pm{program_id, QtCommon::system->GetFileSystemController(),
|
|
QtCommon::system->GetContentProvider()};
|
|
const auto control = pm.GetControlMetadata();
|
|
const auto loader =
|
|
Loader::GetLoader(*QtCommon::system, QtCommon::vfs->OpenFile(game_path, FileSys::OpenMode::Read));
|
|
|
|
std::string game_title{game_title_};
|
|
|
|
// Delete illegal characters from title
|
|
if (needs_title) {
|
|
game_title = fmt::format("{:016X}", program_id);
|
|
if (control.first != nullptr) {
|
|
game_title = control.first->GetApplicationName();
|
|
} else {
|
|
loader->ReadTitle(game_title);
|
|
}
|
|
}
|
|
|
|
const std::string illegal_chars = "<>:\"/\\|?*.";
|
|
for (auto it = game_title.rbegin(); it != game_title.rend(); ++it) {
|
|
if (illegal_chars.find(*it) != std::string::npos) {
|
|
game_title.erase(it.base() - 1);
|
|
}
|
|
}
|
|
|
|
const QString qgame_title = QString::fromStdString(game_title);
|
|
|
|
// Get icon from game file
|
|
std::vector<u8> icon_image_file{};
|
|
if (control.second != nullptr) {
|
|
icon_image_file = control.second->ReadAllBytes();
|
|
} else if (loader->ReadIcon(icon_image_file) != Loader::ResultStatus::Success) {
|
|
LOG_WARNING(Frontend, "Could not read icon from {:s}", game_path);
|
|
}
|
|
|
|
QImage icon_data =
|
|
QImage::fromData(icon_image_file.data(), static_cast<int>(icon_image_file.size()));
|
|
std::filesystem::path out_icon_path;
|
|
if (QtCommon::Game::MakeShortcutIcoPath(program_id, game_title, out_icon_path)) {
|
|
if (!SaveIconToFile(out_icon_path, icon_data)) {
|
|
LOG_ERROR(Frontend, "Could not write icon to file");
|
|
}
|
|
} else {
|
|
QtCommon::Frontend::Critical(
|
|
tr("Create Icon"),
|
|
tr("Cannot create icon file. Path \"%1\" does not exist and cannot be created.")
|
|
.arg(QString::fromStdString(out_icon_path.string())));
|
|
}
|
|
|
|
#if defined(__unix__) && !defined(__APPLE__) && !defined(__ANDROID__)
|
|
// Special case for AppImages
|
|
// Warn once if we are making a shortcut to a volatile AppImage
|
|
if (command.string().ends_with(".AppImage") && !UISettings::values.shortcut_already_warned) {
|
|
if (!CreateShortcutMessagesGUI(ShortcutMessages::Volatile, qgame_title)) {
|
|
return;
|
|
}
|
|
UISettings::values.shortcut_already_warned = true;
|
|
}
|
|
#endif
|
|
|
|
// Create shortcut
|
|
std::string arguments{arguments_};
|
|
if (CreateShortcutMessagesGUI(ShortcutMessages::Fullscreen, qgame_title)) {
|
|
arguments = "-f " + arguments;
|
|
}
|
|
const std::string comment = fmt::format("Start {:s} with the Eden Emulator", game_title);
|
|
const std::string categories = "Game;Emulator;Qt;";
|
|
const std::string keywords = "Switch;Nintendo;";
|
|
|
|
if (QtCommon::Game::CreateShortcutLink(shortcut_path, comment, out_icon_path, command,
|
|
arguments, categories, keywords, game_title)) {
|
|
CreateShortcutMessagesGUI(ShortcutMessages::Success,
|
|
qgame_title);
|
|
return;
|
|
}
|
|
CreateShortcutMessagesGUI(ShortcutMessages::Failed,
|
|
qgame_title);
|
|
}
|
|
|
|
constexpr std::string GetShortcutPath(ShortcutTarget target) {
|
|
{
|
|
std::string shortcut_path{};
|
|
if (target == ShortcutTarget::Desktop) {
|
|
shortcut_path = QStandardPaths::writableLocation(QStandardPaths::DesktopLocation)
|
|
.toStdString();
|
|
} else if (target == ShortcutTarget::Applications) {
|
|
shortcut_path = QStandardPaths::writableLocation(QStandardPaths::ApplicationsLocation)
|
|
.toStdString();
|
|
}
|
|
|
|
return shortcut_path;
|
|
}
|
|
}
|
|
|
|
void CreateHomeMenuShortcut(ShortcutTarget target) {
|
|
constexpr u64 QLaunchId = static_cast<u64>(Service::AM::AppletProgramId::QLaunch);
|
|
auto bis_system = QtCommon::system->GetFileSystemController().GetSystemNANDContents();
|
|
if (!bis_system) {
|
|
QtCommon::Frontend::Warning(tr("No firmware available"),
|
|
tr("Please install firmware to use the home menu."));
|
|
return;
|
|
}
|
|
|
|
auto qlaunch_nca = bis_system->GetEntry(QLaunchId, FileSys::ContentRecordType::Program);
|
|
if (!qlaunch_nca) {
|
|
QtCommon::Frontend::Warning(tr("Home Menu Applet"),
|
|
tr("Home Menu is not available. Please reinstall firmware."));
|
|
return;
|
|
}
|
|
|
|
auto qlaunch_applet_nca = bis_system->GetEntry(QLaunchId, FileSys::ContentRecordType::Program);
|
|
const auto game_path = qlaunch_applet_nca->GetFullPath();
|
|
|
|
// TODO(crueter): Make this use the Eden icon
|
|
CreateShortcut(game_path, QLaunchId, "Switch Home Menu", target, "-qlaunch", false);
|
|
}
|
|
|
|
|
|
} // namespace QtCommon::Game
|