455 lines
17 KiB
C++
Raw Normal View History

2015-08-31 21:35:33 -07:00
// Copyright 2015 Citra Emulator Project
// Licensed under GPLv2 or any later version
// Refer to the license.txt file included.
#pragma once
#include <algorithm>
2018-04-16 00:42:58 +02:00
#include <map>
#include <unordered_map>
#include <utility>
Qt: Add missing #include after 2f8bd1829670 In file included from src/citra_qt/citra-qt_autogen/mocs_compilation.cpp:14: In file included from src/citra_qt/citra-qt_autogen/EWIEGA46WW/moc_game_list_p.cpp:9: src/citra_qt/game_list_p.h:160:17: error: use of undeclared identifier 'QCoreApplication'; did you mean 'QApplication'? setText(QCoreApplication::translate("GameList", status.text)); ^~~~~~~~~~~~~~~~ QApplication /usr/local/include/qt5/QtGui/qwindowdefs.h:81:7: note: 'QApplication' declared here class QApplication; ^ In file included from src/citra_qt/citra-qt_autogen/mocs_compilation.cpp:14: In file included from src/citra_qt/citra-qt_autogen/EWIEGA46WW/moc_game_list_p.cpp:9: src/citra_qt/game_list_p.h:160:17: error: incomplete type 'QApplication' named in nested name specifier setText(QCoreApplication::translate("GameList", status.text)); ^~~~~~~~~~~~~~~~~~ /usr/local/include/qt5/QtCore/qobject.h:446:18: note: forward declaration of 'QApplication' friend class QApplication; ^ In file included from src/citra_qt/citra-qt_autogen/mocs_compilation.cpp:14: In file included from src/citra_qt/citra-qt_autogen/EWIEGA46WW/moc_game_list_p.cpp:9: src/citra_qt/game_list_p.h:161:20: error: use of undeclared identifier 'QCoreApplication'; did you mean 'QApplication'? setToolTip(QCoreApplication::translate("GameList", status.tooltip)); ^~~~~~~~~~~~~~~~ QApplication /usr/local/include/qt5/QtGui/qwindowdefs.h:81:7: note: 'QApplication' declared here class QApplication; ^ In file included from src/citra_qt/citra-qt_autogen/mocs_compilation.cpp:14: In file included from src/citra_qt/citra-qt_autogen/EWIEGA46WW/moc_game_list_p.cpp:9: src/citra_qt/game_list_p.h:161:20: error: incomplete type 'QApplication' named in nested name specifier setToolTip(QCoreApplication::translate("GameList", status.tooltip)); ^~~~~~~~~~~~~~~~~~ /usr/local/include/qt5/QtCore/qobject.h:446:18: note: forward declaration of 'QApplication' friend class QApplication; ^
2018-04-18 20:01:45 +00:00
#include <QCoreApplication>
#include <QFileInfo>
2016-04-14 00:04:05 +03:00
#include <QImage>
2018-04-16 00:42:58 +02:00
#include <QObject>
2015-08-31 21:35:33 -07:00
#include <QRunnable>
#include <QStandardItem>
#include <QString>
#include <QWidget>
2019-08-14 22:38:54 -06:00
#include "citra_qt/uisettings.h"
2015-08-31 21:35:33 -07:00
#include "citra_qt/util/util.h"
#include "common/file_util.h"
2018-04-16 00:42:58 +02:00
#include "common/logging/log.h"
#include "common/string_util.h"
#include "core/loader/smdh.h"
2016-04-14 00:04:05 +03:00
enum class GameListItemType {
Game = QStandardItem::UserType + 1,
CustomDir = QStandardItem::UserType + 2,
InstalledDir = QStandardItem::UserType + 3,
SystemDir = QStandardItem::UserType + 4,
AddDir = QStandardItem::UserType + 5
};
Q_DECLARE_METATYPE(GameListItemType);
2016-04-14 00:04:05 +03:00
/**
* Gets the game icon from SMDH data.
* @param smdh SMDH data
2016-04-14 00:04:05 +03:00
* @param large If true, returns large icon (48x48), otherwise returns small icon (24x24)
* @return QPixmap game icon
*/
static QPixmap GetQPixmapFromSMDH(const Loader::SMDH& smdh, bool large) {
std::vector<u16> icon_data = smdh.GetIcon(large);
const uchar* data = reinterpret_cast<const uchar*>(icon_data.data());
int size = large ? 48 : 24;
QImage icon(data, size, size, QImage::Format::Format_RGB16);
2016-04-14 00:04:05 +03:00
return QPixmap::fromImage(icon);
}
/**
* Gets the default icon (for games without valid SMDH)
* @param large If true, returns large icon (48x48), otherwise returns small icon (24x24)
* @return QPixmap default icon
*/
static QPixmap GetDefaultIcon(bool large) {
int size = large ? 48 : 24;
QPixmap icon(size, size);
icon.fill(Qt::transparent);
return icon;
}
/**
* Gets the short game title from SMDH data.
* @param smdh SMDH data
2016-04-14 00:04:05 +03:00
* @param language title language
* @return QString short title
*/
static QString GetQStringShortTitleFromSMDH(const Loader::SMDH& smdh,
Loader::SMDH::TitleLanguage language) {
return QString::fromUtf16(smdh.GetShortTitle(language).data());
2016-04-14 00:04:05 +03:00
}
2015-08-31 21:35:33 -07:00
/**
* Gets the long game title from SMDH data.
* @param smdh SMDH data
* @param language title language
* @return QString long title
*/
static QString GetQStringLongTitleFromSMDH(const Loader::SMDH& smdh,
2019-09-05 22:54:39 -03:00
Loader::SMDH::TitleLanguage language) {
return QString::fromUtf16(smdh.GetLongTitle(language).data());
}
2018-05-01 19:57:01 +02:00
/**
* Gets the game region from SMDH data.
* @param smdh SMDH data
* @return QString region
*/
static QString GetRegionFromSMDH(const Loader::SMDH& smdh) {
2019-08-11 13:52:08 +02:00
using GameRegion = Loader::SMDH::GameRegion;
2019-08-11 16:44:54 +02:00
static const std::map<GameRegion, const char*> regions_map = {
{GameRegion::Japan, QT_TRANSLATE_NOOP("GameRegion", "Japan")},
{GameRegion::NorthAmerica, QT_TRANSLATE_NOOP("GameRegion", "North America")},
{GameRegion::Europe, QT_TRANSLATE_NOOP("GameRegion", "Europe")},
{GameRegion::Australia, QT_TRANSLATE_NOOP("GameRegion", "Australia")},
{GameRegion::China, QT_TRANSLATE_NOOP("GameRegion", "China")},
{GameRegion::Korea, QT_TRANSLATE_NOOP("GameRegion", "Korea")},
{GameRegion::Taiwan, QT_TRANSLATE_NOOP("GameRegion", "Taiwan")}};
2018-05-01 19:57:01 +02:00
2019-08-11 13:52:08 +02:00
std::vector<GameRegion> regions = smdh.GetRegions();
if (regions.empty()) {
return QCoreApplication::translate("GameRegion", "Invalid region");
2019-08-11 13:52:08 +02:00
}
2019-08-20 21:03:41 +08:00
const bool region_free =
std::all_of(regions_map.begin(), regions_map.end(), [&regions](const auto& it) {
return std::find(regions.begin(), regions.end(), it.first) != regions.end();
});
2019-08-20 21:03:41 +08:00
if (region_free) {
return QCoreApplication::translate("GameRegion", "Region free");
}
const QString separator =
UISettings::values.game_list_single_line_mode ? QStringLiteral(", ") : QStringLiteral("\n");
QString result = QCoreApplication::translate("GameRegion", regions_map.at(regions.front()));
2019-08-11 13:52:08 +02:00
for (auto region = ++regions.begin(); region != regions.end(); ++region) {
result += separator + QCoreApplication::translate("GameRegion", regions_map.at(*region));
2019-08-11 13:52:08 +02:00
}
return result;
2018-05-01 19:57:01 +02:00
}
2018-04-16 00:42:58 +02:00
class GameListItem : public QStandardItem {
2015-08-31 21:35:33 -07:00
public:
// used to access type from item index
static constexpr int TypeRole = Qt::UserRole + 1;
static constexpr int SortRole = Qt::UserRole + 2;
GameListItem() = default;
explicit GameListItem(const QString& string) : QStandardItem(string) {
setData(string, SortRole);
}
2015-08-31 21:35:33 -07:00
};
/// Game list icon sizes (in px)
static const std::unordered_map<UISettings::GameListIconSize, int> IconSizes{
{UISettings::GameListIconSize::NoIcon, 0},
{UISettings::GameListIconSize::SmallIcon, 24},
{UISettings::GameListIconSize::LargeIcon, 48},
};
2015-08-31 21:35:33 -07:00
/**
* A specialization of GameListItem for path values.
* This class ensures that for every full path value it holds, a correct string representation
* of just the filename (with no extension) will be displayed to the user.
2016-10-20 12:26:59 -02:00
* If this class receives valid SMDH data, it will also display game icons and titles.
2015-08-31 21:35:33 -07:00
*/
class GameListItemPath : public GameListItem {
public:
static constexpr int TitleRole = SortRole + 1;
static constexpr int FullPathRole = SortRole + 2;
static constexpr int ProgramIdRole = SortRole + 3;
static constexpr int ExtdataIdRole = SortRole + 4;
static constexpr int LongTitleRole = SortRole + 5;
2015-08-31 21:35:33 -07:00
GameListItemPath() = default;
GameListItemPath(const QString& game_path, const std::vector<u8>& smdh_data, u64 program_id,
u64 extdata_id) {
setData(type(), TypeRole);
2015-08-31 21:35:33 -07:00
setData(game_path, FullPathRole);
setData(qulonglong(program_id), ProgramIdRole);
setData(qulonglong(extdata_id), ExtdataIdRole);
2016-04-14 00:04:05 +03:00
if (UISettings::values.game_list_icon_size == UISettings::GameListIconSize::NoIcon) {
// Do not display icons
setData(QPixmap(), Qt::DecorationRole);
}
bool large =
UISettings::values.game_list_icon_size == UISettings::GameListIconSize::LargeIcon;
2018-01-23 21:32:27 -06:00
if (!Loader::IsValidSMDH(smdh_data)) {
2016-04-14 00:04:05 +03:00
// SMDH is not valid, set a default icon
if (UISettings::values.game_list_icon_size != UISettings::GameListIconSize::NoIcon)
setData(GetDefaultIcon(large), Qt::DecorationRole);
2016-04-14 00:04:05 +03:00
return;
}
2018-01-24 10:17:04 -06:00
2018-01-24 10:16:40 -06:00
Loader::SMDH smdh;
2018-01-23 21:32:27 -06:00
memcpy(&smdh, smdh_data.data(), sizeof(Loader::SMDH));
2016-04-14 00:04:05 +03:00
// Get icon from SMDH
if (UISettings::values.game_list_icon_size != UISettings::GameListIconSize::NoIcon) {
setData(GetQPixmapFromSMDH(smdh, large), Qt::DecorationRole);
}
2016-04-14 00:04:05 +03:00
// Get title from SMDH
setData(GetQStringShortTitleFromSMDH(smdh, Loader::SMDH::TitleLanguage::English),
TitleRole);
// Get long title from SMDH
setData(GetQStringLongTitleFromSMDH(smdh, Loader::SMDH::TitleLanguage::English),
LongTitleRole);
2015-08-31 21:35:33 -07:00
}
int type() const override {
return static_cast<int>(GameListItemType::Game);
}
2016-04-14 00:04:05 +03:00
QVariant data(int role) const override {
if (role == Qt::DisplayRole || role == SortRole) {
std::string path, filename, extension;
Common::SplitPath(data(FullPathRole).toString().toStdString(), &path, &filename,
&extension);
const std::unordered_map<UISettings::GameListText, QString> display_texts{
{UISettings::GameListText::FileName, QString::fromStdString(filename + extension)},
{UISettings::GameListText::FullPath, data(FullPathRole).toString()},
{UISettings::GameListText::TitleName, data(TitleRole).toString()},
{UISettings::GameListText::LongTitleName, data(LongTitleRole).toString()},
{UISettings::GameListText::TitleID,
QString::fromStdString(fmt::format("{:016X}", data(ProgramIdRole).toULongLong()))},
};
const QString& row1 = display_texts.at(UISettings::values.game_list_row_1).simplified();
2019-10-25 18:49:13 -03:00
if (role == SortRole)
return row1.toLower();
QString row2;
auto row_2_id = UISettings::values.game_list_row_2;
if (row_2_id != UISettings::GameListText::NoText) {
if (!row1.isEmpty()) {
row2 = UISettings::values.game_list_single_line_mode
? QStringLiteral(" ")
: QStringLiteral("\n ");
}
row2 += display_texts.at(row_2_id).simplified();
}
return QString(row1 + row2);
2015-08-31 21:35:33 -07:00
} else {
2016-04-14 00:04:05 +03:00
return GameListItem::data(role);
2015-08-31 21:35:33 -07:00
}
}
};
2018-04-16 00:42:58 +02:00
class GameListItemCompat : public GameListItem {
2018-09-01 01:42:07 +02:00
Q_DECLARE_TR_FUNCTIONS(GameListItemCompat)
2018-04-16 00:42:58 +02:00
public:
static constexpr int CompatNumberRole = SortRole;
2018-04-16 00:42:58 +02:00
GameListItemCompat() = default;
explicit GameListItemCompat(const QString& compatibility) {
setData(type(), TypeRole);
2018-09-01 01:42:07 +02:00
struct CompatStatus {
QString color;
const char* text;
const char* tooltip;
};
// clang-format off
static const std::map<QString, CompatStatus> status_data = {
{QStringLiteral("0"), {QStringLiteral("#5c93ed"), QT_TR_NOOP("Perfect"), QT_TR_NOOP("Game functions flawless with no audio or graphical glitches, all tested functionality works as intended without\nany workarounds needed.")}},
{QStringLiteral("1"), {QStringLiteral("#47d35c"), QT_TR_NOOP("Great"), QT_TR_NOOP("Game functions with minor graphical or audio glitches and is playable from start to finish. May require some\nworkarounds.")}},
{QStringLiteral("2"), {QStringLiteral("#94b242"), QT_TR_NOOP("Okay"), QT_TR_NOOP("Game functions with major graphical or audio glitches, but game is playable from start to finish with\nworkarounds.")}},
{QStringLiteral("3"), {QStringLiteral("#f2d624"), QT_TR_NOOP("Bad"), QT_TR_NOOP("Game functions, but with major graphical or audio glitches. Unable to progress in specific areas due to glitches\neven with workarounds.")}},
{QStringLiteral("4"), {QStringLiteral("#ff0000"), QT_TR_NOOP("Intro/Menu"), QT_TR_NOOP("Game is completely unplayable due to major graphical or audio glitches. Unable to progress past the Start\nScreen.")}},
{QStringLiteral("5"), {QStringLiteral("#828282"), QT_TR_NOOP("Won't Boot"), QT_TR_NOOP("The game crashes when attempting to startup.")}},
{QStringLiteral("99"), {QStringLiteral("#000000"), QT_TR_NOOP("Not Tested"), QT_TR_NOOP("The game has not yet been tested.")}}};
2018-09-01 01:42:07 +02:00
// clang-format on
auto iterator = status_data.find(compatibility);
2018-04-16 00:42:58 +02:00
if (iterator == status_data.end()) {
LOG_WARNING(Frontend, "Invalid compatibility number {}", compatibility.toStdString());
2018-04-16 00:42:58 +02:00
return;
}
const CompatStatus& status = iterator->second;
setData(compatibility, CompatNumberRole);
2018-09-01 01:42:07 +02:00
setText(QObject::tr(status.text));
setToolTip(QObject::tr(status.tooltip));
2018-04-16 00:42:58 +02:00
setData(CreateCirclePixmapFromColor(status.color), Qt::DecorationRole);
}
int type() const override {
return static_cast<int>(GameListItemType::Game);
}
2018-04-16 00:42:58 +02:00
bool operator<(const QStandardItem& other) const override {
return data(CompatNumberRole) < other.data(CompatNumberRole);
}
};
2018-05-01 19:57:01 +02:00
class GameListItemRegion : public GameListItem {
public:
GameListItemRegion() = default;
explicit GameListItemRegion(const std::vector<u8>& smdh_data) {
setData(type(), TypeRole);
2018-05-01 19:57:01 +02:00
if (!Loader::IsValidSMDH(smdh_data)) {
setText(QObject::tr("Invalid region"));
return;
}
Loader::SMDH smdh;
memcpy(&smdh, smdh_data.data(), sizeof(Loader::SMDH));
setText(GetRegionFromSMDH(smdh));
setData(GetRegionFromSMDH(smdh), SortRole);
}
int type() const override {
return static_cast<int>(GameListItemType::Game);
2018-05-01 19:57:01 +02:00
}
};
2015-08-31 21:35:33 -07:00
/**
* A specialization of GameListItem for size values.
* This class ensures that for every numerical size value it holds (in bytes), a correct
* human-readable string representation will be displayed to the user.
*/
class GameListItemSize : public GameListItem {
public:
static constexpr int SizeRole = SortRole;
2015-08-31 21:35:33 -07:00
GameListItemSize() = default;
explicit GameListItemSize(const qulonglong size_bytes) {
setData(type(), TypeRole);
2015-08-31 21:35:33 -07:00
setData(size_bytes, SizeRole);
}
void setData(const QVariant& value, int role) override {
2015-08-31 21:35:33 -07:00
// By specializing setData for SizeRole, we can ensure that the numerical and string
// representations of the data are always accurate and in the correct format.
if (role == SizeRole) {
qulonglong size_bytes = value.toULongLong();
GameListItem::setData(ReadableByteSize(size_bytes), Qt::DisplayRole);
GameListItem::setData(value, SizeRole);
} else {
GameListItem::setData(value, role);
}
}
int type() const override {
return static_cast<int>(GameListItemType::Game);
}
2015-08-31 21:35:33 -07:00
/**
* This operator is, in practice, only used by the TreeView sorting systems.
* Override it so that it will correctly sort by numerical value instead of by string
* representation.
2015-08-31 21:35:33 -07:00
*/
bool operator<(const QStandardItem& other) const override {
2015-08-31 21:35:33 -07:00
return data(SizeRole).toULongLong() < other.data(SizeRole).toULongLong();
}
};
class GameListDir : public GameListItem {
public:
static constexpr int GameDirRole = Qt::UserRole + 2;
explicit GameListDir(UISettings::GameDir& directory,
GameListItemType dir_type = GameListItemType::CustomDir)
: dir_type{dir_type} {
setData(type(), TypeRole);
UISettings::GameDir* game_dir = &directory;
setData(QVariant(UISettings::values.game_dirs.indexOf(directory)), GameDirRole);
const int icon_size = IconSizes.at(UISettings::values.game_list_icon_size);
switch (dir_type) {
case GameListItemType::InstalledDir:
setData(QIcon::fromTheme(QStringLiteral("sd_card")).pixmap(icon_size),
Qt::DecorationRole);
setData(QObject::tr("Installed Titles"), Qt::DisplayRole);
break;
case GameListItemType::SystemDir:
setData(QIcon::fromTheme(QStringLiteral("chip")).pixmap(icon_size), Qt::DecorationRole);
setData(QObject::tr("System Titles"), Qt::DisplayRole);
break;
case GameListItemType::CustomDir: {
QString icon_name = QFileInfo::exists(game_dir->path) ? QStringLiteral("folder")
: QStringLiteral("bad_folder");
setData(QIcon::fromTheme(icon_name).pixmap(icon_size), Qt::DecorationRole);
setData(game_dir->path, Qt::DisplayRole);
break;
}
default:
break;
}
}
int type() const override {
return static_cast<int>(dir_type);
}
/**
* Override to prevent automatic sorting.
*/
bool operator<(const QStandardItem& other) const override {
return false;
}
private:
GameListItemType dir_type;
};
class GameListAddDir : public GameListItem {
public:
explicit GameListAddDir() {
setData(type(), TypeRole);
int icon_size = IconSizes.at(UISettings::values.game_list_icon_size);
setData(QIcon::fromTheme(QStringLiteral("plus")).pixmap(icon_size), Qt::DecorationRole);
setData(QObject::tr("Add New Game Directory"), Qt::DisplayRole);
}
int type() const override {
return static_cast<int>(GameListItemType::AddDir);
}
bool operator<(const QStandardItem& other) const override {
return false;
}
};
class GameList;
class QHBoxLayout;
class QTreeView;
class QLabel;
class QLineEdit;
class QToolButton;
class GameListSearchField : public QWidget {
Q_OBJECT
public:
explicit GameListSearchField(GameList* parent = nullptr);
void setFilterResult(int visible, int total);
void clear();
void setFocus();
private:
class KeyReleaseEater : public QObject {
public:
explicit KeyReleaseEater(GameList* gamelist);
private:
GameList* gamelist = nullptr;
QString edit_filter_text_old;
protected:
// EventFilter in order to process systemkeys while editing the searchfield
bool eventFilter(QObject* obj, QEvent* event) override;
};
int visible;
int total;
QHBoxLayout* layout_filter = nullptr;
QTreeView* tree_view = nullptr;
QLabel* label_filter = nullptr;
QLineEdit* edit_filter = nullptr;
QLabel* label_filter_result = nullptr;
QToolButton* button_filter_close = nullptr;
};