2024-03-29 05:43:46 +00:00
|
|
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2024-08-19 04:25:15 +00:00
|
|
|
#include <QClipboard>
|
2024-03-29 05:43:46 +00:00
|
|
|
#include <QDesktopServices>
|
|
|
|
#include <QMenu>
|
2024-07-05 23:57:54 +00:00
|
|
|
#include <QMessageBox>
|
2024-03-29 05:43:46 +00:00
|
|
|
#include <QTreeWidget>
|
|
|
|
#include <QTreeWidgetItem>
|
2024-07-05 23:57:54 +00:00
|
|
|
|
2024-03-29 05:43:46 +00:00
|
|
|
#include "game_info.h"
|
2024-06-11 02:42:21 +00:00
|
|
|
#include "trophy_viewer.h"
|
2024-03-29 05:43:46 +00:00
|
|
|
|
2024-07-05 23:57:54 +00:00
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
#include <ShlObj.h>
|
|
|
|
#include <Windows.h>
|
|
|
|
#include <objbase.h>
|
|
|
|
#include <shlguid.h>
|
|
|
|
#include <shobjidl.h>
|
|
|
|
#endif
|
|
|
|
|
2024-06-11 02:42:21 +00:00
|
|
|
class GuiContextMenus : public QObject {
|
|
|
|
Q_OBJECT
|
2024-03-29 05:43:46 +00:00
|
|
|
public:
|
|
|
|
void RequestGameMenu(const QPoint& pos, QVector<GameInfo> m_games, QTableWidget* widget,
|
|
|
|
bool isList) {
|
|
|
|
QPoint global_pos = widget->viewport()->mapToGlobal(pos);
|
|
|
|
int itemID = 0;
|
|
|
|
if (isList) {
|
|
|
|
itemID = widget->currentRow();
|
|
|
|
} else {
|
|
|
|
itemID = widget->currentRow() * widget->columnCount() + widget->currentColumn();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Setup menu.
|
|
|
|
QMenu menu(widget);
|
2024-07-05 23:57:54 +00:00
|
|
|
QAction createShortcut("Create Shortcut", widget);
|
2024-03-29 05:43:46 +00:00
|
|
|
QAction openFolder("Open Game Folder", widget);
|
|
|
|
QAction openSfoViewer("SFO Viewer", widget);
|
2024-06-11 02:42:21 +00:00
|
|
|
QAction openTrophyViewer("Trophy Viewer", widget);
|
2024-03-29 05:43:46 +00:00
|
|
|
|
|
|
|
menu.addAction(&openFolder);
|
2024-07-10 16:20:19 +00:00
|
|
|
menu.addAction(&createShortcut);
|
2024-03-29 05:43:46 +00:00
|
|
|
menu.addAction(&openSfoViewer);
|
2024-06-11 02:42:21 +00:00
|
|
|
menu.addAction(&openTrophyViewer);
|
2024-07-05 23:57:54 +00:00
|
|
|
|
2024-08-19 04:25:15 +00:00
|
|
|
// "Copy" submenu.
|
|
|
|
QMenu* copyMenu = new QMenu("Copy info", widget);
|
|
|
|
QAction* copyName = new QAction("Copy Name", widget);
|
|
|
|
QAction* copySerial = new QAction("Copy Serial", widget);
|
|
|
|
QAction* copyNameAll = new QAction("Copy All", widget);
|
|
|
|
|
|
|
|
copyMenu->addAction(copyName);
|
|
|
|
copyMenu->addAction(copySerial);
|
|
|
|
copyMenu->addAction(copyNameAll);
|
|
|
|
|
|
|
|
menu.addMenu(copyMenu);
|
|
|
|
|
2024-03-29 05:43:46 +00:00
|
|
|
// Show menu.
|
|
|
|
auto selected = menu.exec(global_pos);
|
|
|
|
if (!selected) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (selected == &openFolder) {
|
|
|
|
QString folderPath = QString::fromStdString(m_games[itemID].path);
|
|
|
|
QDesktopServices::openUrl(QUrl::fromLocalFile(folderPath));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (selected == &openSfoViewer) {
|
|
|
|
PSF psf;
|
2024-06-11 02:42:21 +00:00
|
|
|
if (psf.open(m_games[itemID].path + "/sce_sys/param.sfo", {})) {
|
2024-03-29 05:43:46 +00:00
|
|
|
int rows = psf.map_strings.size() + psf.map_integers.size();
|
|
|
|
QTableWidget* tableWidget = new QTableWidget(rows, 2);
|
2024-06-11 02:42:21 +00:00
|
|
|
tableWidget->setAttribute(Qt::WA_DeleteOnClose);
|
|
|
|
connect(widget->parent(), &QWidget::destroyed, tableWidget,
|
2024-08-23 19:38:55 +00:00
|
|
|
[tableWidget]() { tableWidget->deleteLater(); });
|
2024-06-11 02:42:21 +00:00
|
|
|
|
2024-03-29 05:43:46 +00:00
|
|
|
tableWidget->verticalHeader()->setVisible(false); // Hide vertical header
|
|
|
|
int row = 0;
|
|
|
|
|
|
|
|
for (const auto& pair : psf.map_strings) {
|
|
|
|
QTableWidgetItem* keyItem =
|
|
|
|
new QTableWidgetItem(QString::fromStdString(pair.first));
|
|
|
|
QTableWidgetItem* valueItem =
|
|
|
|
new QTableWidgetItem(QString::fromStdString(pair.second));
|
|
|
|
|
|
|
|
tableWidget->setItem(row, 0, keyItem);
|
|
|
|
tableWidget->setItem(row, 1, valueItem);
|
|
|
|
keyItem->setFlags(keyItem->flags() & ~Qt::ItemIsEditable);
|
|
|
|
valueItem->setFlags(valueItem->flags() & ~Qt::ItemIsEditable);
|
|
|
|
row++;
|
|
|
|
}
|
|
|
|
for (const auto& pair : psf.map_integers) {
|
|
|
|
QTableWidgetItem* keyItem =
|
|
|
|
new QTableWidgetItem(QString::fromStdString(pair.first));
|
|
|
|
QTableWidgetItem* valueItem = new QTableWidgetItem(
|
|
|
|
QString("0x").append(QString::number(pair.second, 16)));
|
|
|
|
|
|
|
|
tableWidget->setItem(row, 0, keyItem);
|
|
|
|
tableWidget->setItem(row, 1, valueItem);
|
|
|
|
keyItem->setFlags(keyItem->flags() & ~Qt::ItemIsEditable);
|
|
|
|
valueItem->setFlags(valueItem->flags() & ~Qt::ItemIsEditable);
|
|
|
|
row++;
|
|
|
|
}
|
|
|
|
tableWidget->resizeColumnsToContents();
|
|
|
|
tableWidget->resizeRowsToContents();
|
|
|
|
|
|
|
|
int width = tableWidget->horizontalHeader()->sectionSize(0) +
|
|
|
|
tableWidget->horizontalHeader()->sectionSize(1) + 2;
|
|
|
|
int height = (rows + 1) * (tableWidget->rowHeight(0));
|
|
|
|
tableWidget->setFixedSize(width, height);
|
|
|
|
tableWidget->sortItems(0, Qt::AscendingOrder);
|
|
|
|
tableWidget->horizontalHeader()->setVisible(false);
|
|
|
|
|
|
|
|
tableWidget->horizontalHeader()->setSectionResizeMode(QHeaderView::Fixed);
|
|
|
|
tableWidget->setWindowTitle("SFO Viewer");
|
|
|
|
tableWidget->show();
|
|
|
|
}
|
|
|
|
}
|
2024-06-11 02:42:21 +00:00
|
|
|
|
|
|
|
if (selected == &openTrophyViewer) {
|
|
|
|
QString trophyPath = QString::fromStdString(m_games[itemID].serial);
|
|
|
|
QString gameTrpPath = QString::fromStdString(m_games[itemID].path);
|
|
|
|
TrophyViewer* trophyViewer = new TrophyViewer(trophyPath, gameTrpPath);
|
|
|
|
trophyViewer->show();
|
|
|
|
connect(widget->parent(), &QWidget::destroyed, trophyViewer,
|
2024-08-23 19:38:55 +00:00
|
|
|
[trophyViewer]() { trophyViewer->deleteLater(); });
|
2024-06-11 02:42:21 +00:00
|
|
|
}
|
2024-07-05 23:57:54 +00:00
|
|
|
|
|
|
|
if (selected == &createShortcut) {
|
|
|
|
QString targetPath = QString::fromStdString(m_games[itemID].path);
|
|
|
|
QString ebootPath = targetPath + "/eboot.bin";
|
|
|
|
|
|
|
|
// Get the full path to the icon
|
|
|
|
QString iconPath = QString::fromStdString(m_games[itemID].icon_path);
|
|
|
|
QFileInfo iconFileInfo(iconPath);
|
|
|
|
QString icoPath = iconFileInfo.absolutePath() + "/" + iconFileInfo.baseName() + ".ico";
|
|
|
|
|
|
|
|
// Path to shortcut/link
|
|
|
|
QString linkPath;
|
|
|
|
|
|
|
|
// Path to the shadps4.exe executable
|
|
|
|
QString exePath;
|
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
linkPath = QStandardPaths::writableLocation(QStandardPaths::DesktopLocation) + "/" +
|
2024-07-09 03:55:48 +00:00
|
|
|
QString::fromStdString(m_games[itemID].name)
|
|
|
|
.remove(QRegularExpression("[\\\\/:*?\"<>|]")) +
|
|
|
|
".lnk";
|
2024-07-05 23:57:54 +00:00
|
|
|
|
|
|
|
exePath = QCoreApplication::applicationFilePath().replace("\\", "/");
|
|
|
|
|
|
|
|
#else
|
|
|
|
linkPath = QStandardPaths::writableLocation(QStandardPaths::DesktopLocation) + "/" +
|
2024-07-09 03:55:48 +00:00
|
|
|
QString::fromStdString(m_games[itemID].name)
|
|
|
|
.remove(QRegularExpression("[\\\\/:*?\"<>|]")) +
|
|
|
|
".desktop";
|
2024-07-05 23:57:54 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
// Convert the icon to .ico if necessary
|
|
|
|
if (iconFileInfo.suffix().toLower() == "png") {
|
|
|
|
// Convert icon from PNG to ICO
|
|
|
|
if (convertPngToIco(iconPath, icoPath)) {
|
|
|
|
|
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
if (createShortcutWin(linkPath, ebootPath, icoPath, exePath)) {
|
|
|
|
#else
|
|
|
|
if (createShortcutLinux(linkPath, ebootPath, iconPath)) {
|
|
|
|
#endif
|
|
|
|
QMessageBox::information(
|
2024-08-04 13:07:10 +00:00
|
|
|
nullptr, "Shortcut creation",
|
|
|
|
QString("Shortcut created successfully!\n %1").arg(linkPath));
|
2024-07-05 23:57:54 +00:00
|
|
|
} else {
|
|
|
|
QMessageBox::critical(
|
|
|
|
nullptr, "Error",
|
2024-08-04 13:07:10 +00:00
|
|
|
QString("Error creating shortcut!\n %1").arg(linkPath));
|
2024-07-05 23:57:54 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
QMessageBox::critical(nullptr, "Error", "Failed to convert icon.");
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// If the icon is already in ICO format, we just create the shortcut
|
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
if (createShortcutWin(linkPath, ebootPath, iconPath, exePath)) {
|
|
|
|
#else
|
|
|
|
if (createShortcutLinux(linkPath, ebootPath, iconPath)) {
|
|
|
|
#endif
|
|
|
|
QMessageBox::information(
|
2024-08-04 13:07:10 +00:00
|
|
|
nullptr, "Shortcut creation",
|
|
|
|
QString("Shortcut created successfully!\n %1").arg(linkPath));
|
2024-07-05 23:57:54 +00:00
|
|
|
} else {
|
|
|
|
QMessageBox::critical(nullptr, "Error",
|
2024-08-04 13:07:10 +00:00
|
|
|
QString("Error creating shortcut!\n %1").arg(linkPath));
|
2024-07-05 23:57:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-08-19 04:25:15 +00:00
|
|
|
|
|
|
|
// Handle the "Copy" actions
|
|
|
|
if (selected == copyName) {
|
|
|
|
QClipboard* clipboard = QGuiApplication::clipboard();
|
|
|
|
clipboard->setText(QString::fromStdString(m_games[itemID].name));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (selected == copySerial) {
|
|
|
|
QClipboard* clipboard = QGuiApplication::clipboard();
|
|
|
|
clipboard->setText(QString::fromStdString(m_games[itemID].serial));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (selected == copyNameAll) {
|
|
|
|
QClipboard* clipboard = QGuiApplication::clipboard();
|
|
|
|
QString combinedText = QString("Name:%1 | Serial:%2 | Version:%3 | Size:%4")
|
|
|
|
.arg(QString::fromStdString(m_games[itemID].name))
|
|
|
|
.arg(QString::fromStdString(m_games[itemID].serial))
|
|
|
|
.arg(QString::fromStdString(m_games[itemID].version))
|
|
|
|
.arg(QString::fromStdString(m_games[itemID].size));
|
|
|
|
clipboard->setText(combinedText);
|
|
|
|
}
|
2024-03-29 05:43:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int GetRowIndex(QTreeWidget* treeWidget, QTreeWidgetItem* item) {
|
|
|
|
int row = 0;
|
|
|
|
for (int i = 0; i < treeWidget->topLevelItemCount(); i++) { // check top level/parent items
|
|
|
|
QTreeWidgetItem* currentItem = treeWidget->topLevelItem(i);
|
|
|
|
if (currentItem == item) {
|
|
|
|
return row;
|
|
|
|
}
|
|
|
|
row++;
|
|
|
|
|
|
|
|
if (currentItem->childCount() > 0) { // check child items
|
|
|
|
for (int j = 0; j < currentItem->childCount(); j++) {
|
|
|
|
QTreeWidgetItem* childItem = currentItem->child(j);
|
|
|
|
if (childItem == item) {
|
|
|
|
return row;
|
|
|
|
}
|
|
|
|
row++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2024-06-11 02:42:21 +00:00
|
|
|
void RequestGameMenuPKGViewer(
|
|
|
|
const QPoint& pos, QStringList m_pkg_app_list, QTreeWidget* treeWidget,
|
|
|
|
std::function<void(std::filesystem::path, int, int)> InstallDragDropPkg) {
|
2024-03-29 05:43:46 +00:00
|
|
|
QPoint global_pos = treeWidget->viewport()->mapToGlobal(pos); // context menu position
|
|
|
|
QTreeWidgetItem* currentItem = treeWidget->currentItem(); // current clicked item
|
|
|
|
int itemIndex = GetRowIndex(treeWidget, currentItem); // row
|
|
|
|
|
|
|
|
QMenu menu(treeWidget);
|
|
|
|
QAction installPackage("Install PKG", treeWidget);
|
|
|
|
|
|
|
|
menu.addAction(&installPackage);
|
|
|
|
|
|
|
|
auto selected = menu.exec(global_pos);
|
|
|
|
if (!selected) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (selected == &installPackage) {
|
2024-03-30 06:03:15 +00:00
|
|
|
QStringList pkg_app_ = m_pkg_app_list[itemIndex].split(";;");
|
2024-06-11 02:42:21 +00:00
|
|
|
std::filesystem::path path(pkg_app_[9].toStdString());
|
|
|
|
#ifdef _WIN32
|
|
|
|
path = std::filesystem::path(pkg_app_[9].toStdWString());
|
|
|
|
#endif
|
|
|
|
InstallDragDropPkg(path, 1, 1);
|
2024-03-29 05:43:46 +00:00
|
|
|
}
|
|
|
|
}
|
2024-07-05 23:57:54 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
bool convertPngToIco(const QString& pngFilePath, const QString& icoFilePath) {
|
|
|
|
// Load the PNG image
|
|
|
|
QImage image(pngFilePath);
|
|
|
|
if (image.isNull()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Scale the image to the default icon size (256x256 pixels)
|
|
|
|
QImage scaledImage =
|
|
|
|
image.scaled(QSize(256, 256), Qt::KeepAspectRatio, Qt::SmoothTransformation);
|
|
|
|
|
|
|
|
// Convert the image to QPixmap
|
|
|
|
QPixmap pixmap = QPixmap::fromImage(scaledImage);
|
|
|
|
|
|
|
|
// Save the pixmap as an ICO file
|
|
|
|
if (pixmap.save(icoFilePath, "ICO")) {
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
bool createShortcutWin(const QString& linkPath, const QString& targetPath,
|
|
|
|
const QString& iconPath, const QString& exePath) {
|
|
|
|
CoInitializeEx(nullptr, COINIT_APARTMENTTHREADED);
|
|
|
|
|
|
|
|
// Create the ShellLink object
|
|
|
|
IShellLink* pShellLink = nullptr;
|
|
|
|
HRESULT hres = CoCreateInstance(CLSID_ShellLink, nullptr, CLSCTX_INPROC_SERVER,
|
|
|
|
IID_IShellLink, (LPVOID*)&pShellLink);
|
|
|
|
if (SUCCEEDED(hres)) {
|
|
|
|
// Defines the path to the program executable
|
|
|
|
pShellLink->SetPath((LPCWSTR)exePath.utf16());
|
|
|
|
|
|
|
|
// Sets the home directory ("Start in")
|
|
|
|
pShellLink->SetWorkingDirectory((LPCWSTR)QFileInfo(exePath).absolutePath().utf16());
|
|
|
|
|
|
|
|
// Set arguments, eboot.bin file location
|
|
|
|
QString arguments = QString("\"%1\"").arg(targetPath);
|
|
|
|
pShellLink->SetArguments((LPCWSTR)arguments.utf16());
|
|
|
|
|
|
|
|
// Set the icon for the shortcut
|
|
|
|
pShellLink->SetIconLocation((LPCWSTR)iconPath.utf16(), 0);
|
|
|
|
|
|
|
|
// Save the shortcut
|
|
|
|
IPersistFile* pPersistFile = nullptr;
|
|
|
|
hres = pShellLink->QueryInterface(IID_IPersistFile, (LPVOID*)&pPersistFile);
|
|
|
|
if (SUCCEEDED(hres)) {
|
|
|
|
hres = pPersistFile->Save((LPCWSTR)linkPath.utf16(), TRUE);
|
|
|
|
pPersistFile->Release();
|
|
|
|
}
|
|
|
|
pShellLink->Release();
|
|
|
|
}
|
|
|
|
|
|
|
|
CoUninitialize();
|
|
|
|
|
|
|
|
return SUCCEEDED(hres);
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
bool createShortcutLinux(const QString& linkPath, const QString& targetPath,
|
|
|
|
const QString& iconPath) {
|
|
|
|
QFile shortcutFile(linkPath);
|
|
|
|
if (!shortcutFile.open(QIODevice::WriteOnly | QIODevice::Text)) {
|
|
|
|
QMessageBox::critical(nullptr, "Error",
|
2024-08-04 13:07:10 +00:00
|
|
|
QString("Error creating shortcut!\n %1").arg(linkPath));
|
2024-07-05 23:57:54 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QTextStream out(&shortcutFile);
|
|
|
|
out << "[Desktop Entry]\n";
|
|
|
|
out << "Version=1.0\n";
|
2024-07-06 18:10:28 +00:00
|
|
|
out << "Name=" << QFileInfo(linkPath).baseName() << "\n";
|
2024-07-06 16:40:26 +00:00
|
|
|
out << "Exec=" << QCoreApplication::applicationFilePath() << " \"" << targetPath << "\"\n";
|
2024-07-06 16:42:53 +00:00
|
|
|
out << "Icon=" << iconPath << "\n";
|
2024-07-05 23:57:54 +00:00
|
|
|
out << "Terminal=false\n";
|
|
|
|
out << "Type=Application\n";
|
|
|
|
shortcutFile.close();
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#endif
|
2024-03-29 05:43:46 +00:00
|
|
|
};
|