2015-09-01 01:35:33 -03:00
|
|
|
// Copyright 2015 Citra Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2018-08-09 20:53:30 -04:00
|
|
|
#include <regex>
|
2017-04-17 23:53:40 -03:00
|
|
|
#include <QApplication>
|
2018-06-14 12:02:32 -04:00
|
|
|
#include <QDir>
|
2017-02-12 17:28:56 -03:00
|
|
|
#include <QFileInfo>
|
2015-09-01 01:35:33 -03:00
|
|
|
#include <QHeaderView>
|
2018-08-29 10:42:53 -03:00
|
|
|
#include <QJsonArray>
|
|
|
|
#include <QJsonDocument>
|
|
|
|
#include <QJsonObject>
|
2017-04-29 23:04:39 -03:00
|
|
|
#include <QKeyEvent>
|
2016-12-15 06:55:03 -03:00
|
|
|
#include <QMenu>
|
2015-09-01 01:35:33 -03:00
|
|
|
#include <QThreadPool>
|
2018-08-29 10:42:53 -03:00
|
|
|
#include <fmt/format.h>
|
2015-09-01 01:35:33 -03:00
|
|
|
#include "common/common_paths.h"
|
2018-09-02 11:53:06 -03:00
|
|
|
#include "common/common_types.h"
|
|
|
|
#include "common/file_util.h"
|
2015-09-01 01:35:33 -03:00
|
|
|
#include "common/logging/log.h"
|
2018-07-28 12:32:16 -04:00
|
|
|
#include "core/file_sys/content_archive.h"
|
|
|
|
#include "core/file_sys/control_metadata.h"
|
2018-09-03 22:58:19 -03:00
|
|
|
#include "core/file_sys/nca_metadata.h"
|
2018-08-20 21:36:36 -03:00
|
|
|
#include "core/file_sys/registered_cache.h"
|
2018-08-08 14:34:06 -04:00
|
|
|
#include "core/file_sys/romfs.h"
|
2018-07-18 21:07:11 -04:00
|
|
|
#include "core/file_sys/vfs_real.h"
|
2018-08-31 13:21:34 -03:00
|
|
|
#include "core/hle/service/filesystem/filesystem.h"
|
2016-09-20 12:21:23 -03:00
|
|
|
#include "core/loader/loader.h"
|
2018-08-31 13:21:34 -03:00
|
|
|
#include "yuzu/game_list.h"
|
|
|
|
#include "yuzu/game_list_p.h"
|
|
|
|
#include "yuzu/main.h"
|
|
|
|
#include "yuzu/ui_settings.h"
|
2015-09-01 01:35:33 -03:00
|
|
|
|
2018-07-18 00:04:33 -04:00
|
|
|
GameList::SearchField::KeyReleaseEater::KeyReleaseEater(GameList* gamelist) : gamelist{gamelist} {}
|
2017-04-29 23:04:39 -03:00
|
|
|
|
|
|
|
// EventFilter in order to process systemkeys while editing the searchfield
|
|
|
|
bool GameList::SearchField::KeyReleaseEater::eventFilter(QObject* obj, QEvent* event) {
|
|
|
|
// If it isn't a KeyRelease event then continue with standard event processing
|
|
|
|
if (event->type() != QEvent::KeyRelease)
|
|
|
|
return QObject::eventFilter(obj, event);
|
|
|
|
|
|
|
|
QKeyEvent* keyEvent = static_cast<QKeyEvent*>(event);
|
|
|
|
int rowCount = gamelist->tree_view->model()->rowCount();
|
|
|
|
QString edit_filter_text = gamelist->search_field->edit_filter->text().toLower();
|
|
|
|
|
|
|
|
// If the searchfield's text hasn't changed special function keys get checked
|
|
|
|
// If no function key changes the searchfield's text the filter doesn't need to get reloaded
|
|
|
|
if (edit_filter_text == edit_filter_text_old) {
|
|
|
|
switch (keyEvent->key()) {
|
|
|
|
// Escape: Resets the searchfield
|
|
|
|
case Qt::Key_Escape: {
|
|
|
|
if (edit_filter_text_old.isEmpty()) {
|
|
|
|
return QObject::eventFilter(obj, event);
|
|
|
|
} else {
|
|
|
|
gamelist->search_field->edit_filter->clear();
|
|
|
|
edit_filter_text = "";
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// Return and Enter
|
2017-05-02 18:55:27 -03:00
|
|
|
// If the enter key gets pressed first checks how many and which entry is visible
|
2017-04-29 23:04:39 -03:00
|
|
|
// If there is only one result launch this game
|
|
|
|
case Qt::Key_Return:
|
|
|
|
case Qt::Key_Enter: {
|
|
|
|
QStandardItemModel* item_model = new QStandardItemModel(gamelist->tree_view);
|
|
|
|
QModelIndex root_index = item_model->invisibleRootItem()->index();
|
|
|
|
QStandardItem* child_file;
|
|
|
|
QString file_path;
|
|
|
|
int resultCount = 0;
|
|
|
|
for (int i = 0; i < rowCount; ++i) {
|
|
|
|
if (!gamelist->tree_view->isRowHidden(i, root_index)) {
|
|
|
|
++resultCount;
|
|
|
|
child_file = gamelist->item_model->item(i, 0);
|
|
|
|
file_path = child_file->data(GameListItemPath::FullPathRole).toString();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (resultCount == 1) {
|
|
|
|
// To avoid loading error dialog loops while confirming them using enter
|
|
|
|
// Also users usually want to run a diffrent game after closing one
|
|
|
|
gamelist->search_field->edit_filter->setText("");
|
|
|
|
edit_filter_text = "";
|
|
|
|
emit gamelist->GameChosen(file_path);
|
|
|
|
} else {
|
|
|
|
return QObject::eventFilter(obj, event);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return QObject::eventFilter(obj, event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
edit_filter_text_old = edit_filter_text;
|
|
|
|
return QObject::eventFilter(obj, event);
|
|
|
|
}
|
|
|
|
|
2017-05-02 18:55:27 -03:00
|
|
|
void GameList::SearchField::setFilterResult(int visible, int total) {
|
2017-04-29 23:04:39 -03:00
|
|
|
QString result_of_text = tr("of");
|
|
|
|
QString result_text;
|
|
|
|
if (total == 1) {
|
|
|
|
result_text = tr("result");
|
|
|
|
} else {
|
|
|
|
result_text = tr("results");
|
|
|
|
}
|
|
|
|
label_filter_result->setText(
|
2017-05-02 18:55:27 -03:00
|
|
|
QString("%1 %2 %3 %4").arg(visible).arg(result_of_text).arg(total).arg(result_text));
|
2017-04-29 23:04:39 -03:00
|
|
|
}
|
2015-09-01 01:35:33 -03:00
|
|
|
|
2017-04-29 23:04:39 -03:00
|
|
|
void GameList::SearchField::clear() {
|
|
|
|
edit_filter->setText("");
|
|
|
|
}
|
|
|
|
|
|
|
|
void GameList::SearchField::setFocus() {
|
|
|
|
if (edit_filter->isVisible()) {
|
|
|
|
edit_filter->setFocus();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
GameList::SearchField::SearchField(GameList* parent) : QWidget{parent} {
|
|
|
|
KeyReleaseEater* keyReleaseEater = new KeyReleaseEater(parent);
|
|
|
|
layout_filter = new QHBoxLayout;
|
|
|
|
layout_filter->setMargin(8);
|
|
|
|
label_filter = new QLabel;
|
|
|
|
label_filter->setText(tr("Filter:"));
|
|
|
|
edit_filter = new QLineEdit;
|
|
|
|
edit_filter->setText("");
|
|
|
|
edit_filter->setPlaceholderText(tr("Enter pattern to filter"));
|
|
|
|
edit_filter->installEventFilter(keyReleaseEater);
|
|
|
|
edit_filter->setClearButtonEnabled(true);
|
2018-01-18 22:03:13 -03:00
|
|
|
connect(edit_filter, &QLineEdit::textChanged, parent, &GameList::onTextChanged);
|
2017-04-29 23:04:39 -03:00
|
|
|
label_filter_result = new QLabel;
|
|
|
|
button_filter_close = new QToolButton(this);
|
|
|
|
button_filter_close->setText("X");
|
|
|
|
button_filter_close->setCursor(Qt::ArrowCursor);
|
|
|
|
button_filter_close->setStyleSheet("QToolButton{ border: none; padding: 0px; color: "
|
|
|
|
"#000000; font-weight: bold; background: #F0F0F0; }"
|
|
|
|
"QToolButton:hover{ border: none; padding: 0px; color: "
|
|
|
|
"#EEEEEE; font-weight: bold; background: #E81123}");
|
2018-01-18 22:03:13 -03:00
|
|
|
connect(button_filter_close, &QToolButton::clicked, parent, &GameList::onFilterCloseClicked);
|
2017-04-29 23:04:39 -03:00
|
|
|
layout_filter->setSpacing(10);
|
|
|
|
layout_filter->addWidget(label_filter);
|
|
|
|
layout_filter->addWidget(edit_filter);
|
|
|
|
layout_filter->addWidget(label_filter_result);
|
|
|
|
layout_filter->addWidget(button_filter_close);
|
|
|
|
setLayout(layout_filter);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-05-02 18:55:27 -03:00
|
|
|
* Checks if all words separated by spaces are contained in another string
|
|
|
|
* This offers a word order insensitive search function
|
|
|
|
*
|
2018-01-17 21:51:14 -03:00
|
|
|
* @param haystack String that gets checked if it contains all words of the userinput string
|
|
|
|
* @param userinput String containing all words getting checked
|
2017-05-02 18:55:27 -03:00
|
|
|
* @return true if the haystack contains all words of userinput
|
|
|
|
*/
|
2018-07-18 16:52:12 -04:00
|
|
|
static bool ContainsAllWords(const QString& haystack, const QString& userinput) {
|
2018-07-18 00:12:39 -04:00
|
|
|
const QStringList userinput_split =
|
|
|
|
userinput.split(' ', QString::SplitBehavior::SkipEmptyParts);
|
|
|
|
|
2017-04-29 23:04:39 -03:00
|
|
|
return std::all_of(userinput_split.begin(), userinput_split.end(),
|
2018-07-18 00:12:39 -04:00
|
|
|
[&haystack](const QString& s) { return haystack.contains(s); });
|
2017-04-29 23:04:39 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Event in order to filter the gamelist after editing the searchfield
|
|
|
|
void GameList::onTextChanged(const QString& newText) {
|
|
|
|
int rowCount = tree_view->model()->rowCount();
|
|
|
|
QString edit_filter_text = newText.toLower();
|
|
|
|
|
|
|
|
QModelIndex root_index = item_model->invisibleRootItem()->index();
|
|
|
|
|
|
|
|
// If the searchfield is empty every item is visible
|
|
|
|
// Otherwise the filter gets applied
|
|
|
|
if (edit_filter_text.isEmpty()) {
|
|
|
|
for (int i = 0; i < rowCount; ++i) {
|
|
|
|
tree_view->setRowHidden(i, root_index, false);
|
|
|
|
}
|
|
|
|
search_field->setFilterResult(rowCount, rowCount);
|
|
|
|
} else {
|
|
|
|
int result_count = 0;
|
|
|
|
for (int i = 0; i < rowCount; ++i) {
|
2018-08-06 13:48:14 -04:00
|
|
|
const QStandardItem* child_file = item_model->item(i, 0);
|
|
|
|
const QString file_path =
|
|
|
|
child_file->data(GameListItemPath::FullPathRole).toString().toLower();
|
|
|
|
QString file_name = file_path.mid(file_path.lastIndexOf('/') + 1);
|
|
|
|
const QString file_title =
|
|
|
|
child_file->data(GameListItemPath::TitleRole).toString().toLower();
|
|
|
|
const QString file_programmid =
|
2017-04-29 23:04:39 -03:00
|
|
|
child_file->data(GameListItemPath::ProgramIdRole).toString().toLower();
|
|
|
|
|
|
|
|
// Only items which filename in combination with its title contains all words
|
2018-01-16 13:55:06 -03:00
|
|
|
// that are in the searchfield will be visible in the gamelist
|
2017-04-29 23:04:39 -03:00
|
|
|
// The search is case insensitive because of toLower()
|
|
|
|
// I decided not to use Qt::CaseInsensitive in containsAllWords to prevent
|
|
|
|
// multiple conversions of edit_filter_text for each game in the gamelist
|
2018-07-18 00:15:46 -04:00
|
|
|
if (ContainsAllWords(file_name.append(' ').append(file_title), edit_filter_text) ||
|
2017-04-29 23:04:39 -03:00
|
|
|
(file_programmid.count() == 16 && edit_filter_text.contains(file_programmid))) {
|
|
|
|
tree_view->setRowHidden(i, root_index, false);
|
|
|
|
++result_count;
|
|
|
|
} else {
|
|
|
|
tree_view->setRowHidden(i, root_index, true);
|
|
|
|
}
|
|
|
|
search_field->setFilterResult(result_count, rowCount);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void GameList::onFilterCloseClicked() {
|
|
|
|
main_window->filterBarSetChecked(false);
|
|
|
|
}
|
|
|
|
|
2018-08-03 11:51:48 -04:00
|
|
|
GameList::GameList(FileSys::VirtualFilesystem vfs, GMainWindow* parent)
|
|
|
|
: QWidget{parent}, vfs(std::move(vfs)) {
|
2017-04-17 23:53:40 -03:00
|
|
|
watcher = new QFileSystemWatcher(this);
|
|
|
|
connect(watcher, &QFileSystemWatcher::directoryChanged, this, &GameList::RefreshGameDirectory);
|
|
|
|
|
2017-04-29 23:04:39 -03:00
|
|
|
this->main_window = parent;
|
|
|
|
layout = new QVBoxLayout;
|
2015-09-01 01:35:33 -03:00
|
|
|
tree_view = new QTreeView;
|
2017-04-29 23:04:39 -03:00
|
|
|
search_field = new SearchField(this);
|
2015-09-01 01:35:33 -03:00
|
|
|
item_model = new QStandardItemModel(tree_view);
|
|
|
|
tree_view->setModel(item_model);
|
|
|
|
|
|
|
|
tree_view->setAlternatingRowColors(true);
|
|
|
|
tree_view->setSelectionMode(QHeaderView::SingleSelection);
|
|
|
|
tree_view->setSelectionBehavior(QHeaderView::SelectRows);
|
|
|
|
tree_view->setVerticalScrollMode(QHeaderView::ScrollPerPixel);
|
|
|
|
tree_view->setHorizontalScrollMode(QHeaderView::ScrollPerPixel);
|
|
|
|
tree_view->setSortingEnabled(true);
|
|
|
|
tree_view->setEditTriggers(QHeaderView::NoEditTriggers);
|
|
|
|
tree_view->setUniformRowHeights(true);
|
2016-12-15 06:55:03 -03:00
|
|
|
tree_view->setContextMenuPolicy(Qt::CustomContextMenu);
|
2015-09-01 01:35:33 -03:00
|
|
|
|
|
|
|
item_model->insertColumns(0, COLUMN_COUNT);
|
|
|
|
item_model->setHeaderData(COLUMN_NAME, Qt::Horizontal, "Name");
|
2018-08-29 10:42:53 -03:00
|
|
|
item_model->setHeaderData(COLUMN_COMPATIBILITY, Qt::Horizontal, "Compatibility");
|
2016-04-13 18:04:05 -03:00
|
|
|
item_model->setHeaderData(COLUMN_FILE_TYPE, Qt::Horizontal, "File type");
|
2015-09-01 01:35:33 -03:00
|
|
|
item_model->setHeaderData(COLUMN_SIZE, Qt::Horizontal, "Size");
|
|
|
|
|
2016-12-10 23:20:34 -03:00
|
|
|
connect(tree_view, &QTreeView::activated, this, &GameList::ValidateEntry);
|
2016-12-15 06:55:03 -03:00
|
|
|
connect(tree_view, &QTreeView::customContextMenuRequested, this, &GameList::PopupContextMenu);
|
2015-09-01 01:35:33 -03:00
|
|
|
|
2016-09-17 21:38:01 -03:00
|
|
|
// We must register all custom types with the Qt Automoc system so that we are able to use it
|
2016-12-15 06:55:03 -03:00
|
|
|
// with signals/slots. In this case, QList falls under the umbrells of custom types.
|
2015-09-01 01:35:33 -03:00
|
|
|
qRegisterMetaType<QList<QStandardItem*>>("QList<QStandardItem*>");
|
|
|
|
|
2017-02-18 17:09:14 -03:00
|
|
|
layout->setContentsMargins(0, 0, 0, 0);
|
2017-04-29 23:04:39 -03:00
|
|
|
layout->setSpacing(0);
|
2015-09-01 01:35:33 -03:00
|
|
|
layout->addWidget(tree_view);
|
2017-04-29 23:04:39 -03:00
|
|
|
layout->addWidget(search_field);
|
2015-09-01 01:35:33 -03:00
|
|
|
setLayout(layout);
|
|
|
|
}
|
|
|
|
|
2016-09-17 21:38:01 -03:00
|
|
|
GameList::~GameList() {
|
2015-09-01 01:35:33 -03:00
|
|
|
emit ShouldCancelWorker();
|
|
|
|
}
|
|
|
|
|
2017-04-29 23:04:39 -03:00
|
|
|
void GameList::setFilterFocus() {
|
2017-05-06 07:08:28 -03:00
|
|
|
if (tree_view->model()->rowCount() > 0) {
|
|
|
|
search_field->setFocus();
|
|
|
|
}
|
2017-04-29 23:04:39 -03:00
|
|
|
}
|
|
|
|
|
2017-05-02 19:23:20 -03:00
|
|
|
void GameList::setFilterVisible(bool visibility) {
|
|
|
|
search_field->setVisible(visibility);
|
2017-04-29 23:04:39 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
void GameList::clearFilter() {
|
|
|
|
search_field->clear();
|
|
|
|
}
|
|
|
|
|
2016-12-10 23:27:38 -03:00
|
|
|
void GameList::AddEntry(const QList<QStandardItem*>& entry_items) {
|
2015-09-01 01:35:33 -03:00
|
|
|
item_model->invisibleRootItem()->appendRow(entry_items);
|
|
|
|
}
|
|
|
|
|
2016-09-17 21:38:01 -03:00
|
|
|
void GameList::ValidateEntry(const QModelIndex& item) {
|
2015-09-01 01:35:33 -03:00
|
|
|
// We don't care about the individual QStandardItem that was selected, but its row.
|
2018-08-06 14:57:14 -04:00
|
|
|
const int row = item_model->itemFromIndex(item)->row();
|
|
|
|
const QStandardItem* child_file = item_model->invisibleRootItem()->child(row, COLUMN_NAME);
|
|
|
|
const QString file_path = child_file->data(GameListItemPath::FullPathRole).toString();
|
2015-09-01 01:35:33 -03:00
|
|
|
|
|
|
|
if (file_path.isEmpty())
|
|
|
|
return;
|
2018-08-06 14:57:14 -04:00
|
|
|
|
|
|
|
if (!QFileInfo::exists(file_path))
|
2015-09-01 01:35:33 -03:00
|
|
|
return;
|
2018-08-06 14:57:14 -04:00
|
|
|
|
|
|
|
const QFileInfo file_info{file_path};
|
|
|
|
if (file_info.isDir()) {
|
|
|
|
const QDir dir{file_path};
|
|
|
|
const QStringList matching_main = dir.entryList(QStringList("main"), QDir::Files);
|
2018-06-14 17:25:40 -04:00
|
|
|
if (matching_main.size() == 1) {
|
|
|
|
emit GameChosen(dir.path() + DIR_SEP + matching_main[0]);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-04-29 23:04:39 -03:00
|
|
|
// Users usually want to run a diffrent game after closing one
|
|
|
|
search_field->clear();
|
2015-09-01 01:35:33 -03:00
|
|
|
emit GameChosen(file_path);
|
|
|
|
}
|
|
|
|
|
2017-04-17 23:53:40 -03:00
|
|
|
void GameList::DonePopulating(QStringList watch_list) {
|
|
|
|
// Clear out the old directories to watch for changes and add the new ones
|
|
|
|
auto watch_dirs = watcher->directories();
|
|
|
|
if (!watch_dirs.isEmpty()) {
|
|
|
|
watcher->removePaths(watch_dirs);
|
|
|
|
}
|
|
|
|
// Workaround: Add the watch paths in chunks to allow the gui to refresh
|
|
|
|
// This prevents the UI from stalling when a large number of watch paths are added
|
|
|
|
// Also artificially caps the watcher to a certain number of directories
|
|
|
|
constexpr int LIMIT_WATCH_DIRECTORIES = 5000;
|
|
|
|
constexpr int SLICE_SIZE = 25;
|
|
|
|
int len = std::min(watch_list.length(), LIMIT_WATCH_DIRECTORIES);
|
|
|
|
for (int i = 0; i < len; i += SLICE_SIZE) {
|
|
|
|
watcher->addPaths(watch_list.mid(i, i + SLICE_SIZE));
|
|
|
|
QCoreApplication::processEvents();
|
|
|
|
}
|
2015-09-01 01:35:33 -03:00
|
|
|
tree_view->setEnabled(true);
|
2017-04-29 23:04:39 -03:00
|
|
|
int rowCount = tree_view->model()->rowCount();
|
|
|
|
search_field->setFilterResult(rowCount, rowCount);
|
2017-05-06 07:08:28 -03:00
|
|
|
if (rowCount > 0) {
|
|
|
|
search_field->setFocus();
|
|
|
|
}
|
2015-09-01 01:35:33 -03:00
|
|
|
}
|
|
|
|
|
2016-12-15 06:55:03 -03:00
|
|
|
void GameList::PopupContextMenu(const QPoint& menu_location) {
|
|
|
|
QModelIndex item = tree_view->indexAt(menu_location);
|
|
|
|
if (!item.isValid())
|
|
|
|
return;
|
|
|
|
|
|
|
|
int row = item_model->itemFromIndex(item)->row();
|
|
|
|
QStandardItem* child_file = item_model->invisibleRootItem()->child(row, COLUMN_NAME);
|
|
|
|
u64 program_id = child_file->data(GameListItemPath::ProgramIdRole).toULongLong();
|
|
|
|
|
|
|
|
QMenu context_menu;
|
|
|
|
QAction* open_save_location = context_menu.addAction(tr("Open Save Data Location"));
|
2018-08-29 10:42:53 -03:00
|
|
|
QAction* navigate_to_gamedb_entry = context_menu.addAction(tr("Navigate to GameDB entry"));
|
|
|
|
|
2016-12-15 06:55:03 -03:00
|
|
|
open_save_location->setEnabled(program_id != 0);
|
2018-08-29 10:42:53 -03:00
|
|
|
auto it = FindMatchingCompatibilityEntry(compatibility_list, program_id);
|
|
|
|
navigate_to_gamedb_entry->setVisible(it != compatibility_list.end() && program_id != 0);
|
|
|
|
|
2016-12-15 06:55:03 -03:00
|
|
|
connect(open_save_location, &QAction::triggered,
|
2018-08-21 01:46:40 -03:00
|
|
|
[&]() { emit OpenFolderRequested(program_id, GameListOpenTarget::SaveData); });
|
2018-08-29 10:42:53 -03:00
|
|
|
connect(navigate_to_gamedb_entry, &QAction::triggered,
|
|
|
|
[&]() { emit NavigateToGamedbEntryRequested(program_id, compatibility_list); });
|
|
|
|
|
2016-12-15 06:55:03 -03:00
|
|
|
context_menu.exec(tree_view->viewport()->mapToGlobal(menu_location));
|
|
|
|
}
|
|
|
|
|
2018-08-29 10:42:53 -03:00
|
|
|
void GameList::LoadCompatibilityList() {
|
|
|
|
QFile compat_list{":compatibility_list/compatibility_list.json"};
|
|
|
|
|
|
|
|
if (!compat_list.open(QFile::ReadOnly | QFile::Text)) {
|
|
|
|
LOG_ERROR(Frontend, "Unable to open game compatibility list");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (compat_list.size() == 0) {
|
|
|
|
LOG_WARNING(Frontend, "Game compatibility list is empty");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const QByteArray content = compat_list.readAll();
|
|
|
|
if (content.isEmpty()) {
|
|
|
|
LOG_ERROR(Frontend, "Unable to completely read game compatibility list");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const QString string_content = content;
|
|
|
|
QJsonDocument json = QJsonDocument::fromJson(string_content.toUtf8());
|
|
|
|
QJsonArray arr = json.array();
|
|
|
|
|
|
|
|
for (const QJsonValue& value : arr) {
|
|
|
|
QJsonObject game = value.toObject();
|
|
|
|
|
|
|
|
if (game.contains("compatibility") && game["compatibility"].isDouble()) {
|
|
|
|
int compatibility = game["compatibility"].toInt();
|
|
|
|
QString directory = game["directory"].toString();
|
|
|
|
QJsonArray ids = game["releases"].toArray();
|
|
|
|
|
|
|
|
for (const QJsonValue& value : ids) {
|
|
|
|
QJsonObject object = value.toObject();
|
|
|
|
QString id = object["id"].toString();
|
|
|
|
compatibility_list.emplace(
|
|
|
|
id.toUpper().toStdString(),
|
|
|
|
std::make_pair(QString::number(compatibility), directory));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-17 21:38:01 -03:00
|
|
|
void GameList::PopulateAsync(const QString& dir_path, bool deep_scan) {
|
|
|
|
if (!FileUtil::Exists(dir_path.toStdString()) ||
|
|
|
|
!FileUtil::IsDirectory(dir_path.toStdString())) {
|
2018-07-02 12:20:50 -04:00
|
|
|
LOG_ERROR(Frontend, "Could not find game list folder at {}", dir_path.toLocal8Bit().data());
|
2017-05-02 18:55:27 -03:00
|
|
|
search_field->setFilterResult(0, 0);
|
2015-09-01 01:35:33 -03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
tree_view->setEnabled(false);
|
|
|
|
// Delete any rows that might already exist if we're repopulating
|
|
|
|
item_model->removeRows(0, item_model->rowCount());
|
|
|
|
|
|
|
|
emit ShouldCancelWorker();
|
2017-02-23 18:29:00 -03:00
|
|
|
|
2018-08-29 10:42:53 -03:00
|
|
|
GameListWorker* worker = new GameListWorker(vfs, dir_path, deep_scan, compatibility_list);
|
2015-09-01 01:35:33 -03:00
|
|
|
|
2016-12-10 23:20:34 -03:00
|
|
|
connect(worker, &GameListWorker::EntryReady, this, &GameList::AddEntry, Qt::QueuedConnection);
|
|
|
|
connect(worker, &GameListWorker::Finished, this, &GameList::DonePopulating,
|
|
|
|
Qt::QueuedConnection);
|
2016-09-17 21:38:01 -03:00
|
|
|
// Use DirectConnection here because worker->Cancel() is thread-safe and we want it to cancel
|
|
|
|
// without delay.
|
2016-12-10 23:20:34 -03:00
|
|
|
connect(this, &GameList::ShouldCancelWorker, worker, &GameListWorker::Cancel,
|
|
|
|
Qt::DirectConnection);
|
2015-09-01 01:35:33 -03:00
|
|
|
|
|
|
|
QThreadPool::globalInstance()->start(worker);
|
|
|
|
current_worker = std::move(worker);
|
2015-09-07 03:51:57 -03:00
|
|
|
}
|
|
|
|
|
2016-09-17 21:38:01 -03:00
|
|
|
void GameList::SaveInterfaceLayout() {
|
2016-01-24 17:23:55 -03:00
|
|
|
UISettings::values.gamelist_header_state = tree_view->header()->saveState();
|
2015-09-07 03:51:57 -03:00
|
|
|
}
|
|
|
|
|
2016-09-17 21:38:01 -03:00
|
|
|
void GameList::LoadInterfaceLayout() {
|
2015-09-07 03:51:57 -03:00
|
|
|
auto header = tree_view->header();
|
2016-04-29 20:40:54 -03:00
|
|
|
if (!header->restoreState(UISettings::values.gamelist_header_state)) {
|
|
|
|
// We are using the name column to display icons and titles
|
|
|
|
// so make it as large as possible as default.
|
|
|
|
header->resizeSection(COLUMN_NAME, header->width());
|
|
|
|
}
|
2015-09-07 03:51:57 -03:00
|
|
|
|
|
|
|
item_model->sort(header->sortIndicatorSection(), header->sortIndicatorOrder());
|
2015-09-01 01:35:33 -03:00
|
|
|
}
|
|
|
|
|
2018-08-25 12:50:15 -03:00
|
|
|
const QStringList GameList::supported_file_extensions = {"nso", "nro", "nca", "xci", "nsp"};
|
2017-02-12 17:28:56 -03:00
|
|
|
|
|
|
|
static bool HasSupportedFileExtension(const std::string& file_name) {
|
2018-08-06 14:49:21 -04:00
|
|
|
const QFileInfo file = QFileInfo(QString::fromStdString(file_name));
|
2017-02-14 00:02:53 -03:00
|
|
|
return GameList::supported_file_extensions.contains(file.suffix(), Qt::CaseInsensitive);
|
2017-02-12 17:28:56 -03:00
|
|
|
}
|
|
|
|
|
2018-06-14 12:02:32 -04:00
|
|
|
static bool IsExtractedNCAMain(const std::string& file_name) {
|
2018-08-06 14:49:21 -04:00
|
|
|
return QFileInfo(QString::fromStdString(file_name)).fileName() == "main";
|
2018-06-14 12:02:32 -04:00
|
|
|
}
|
|
|
|
|
2018-06-14 17:25:40 -04:00
|
|
|
static QString FormatGameName(const std::string& physical_name) {
|
2018-08-06 14:49:21 -04:00
|
|
|
const QString physical_name_as_qstring = QString::fromStdString(physical_name);
|
|
|
|
const QFileInfo file_info(physical_name_as_qstring);
|
|
|
|
|
2018-06-14 12:02:32 -04:00
|
|
|
if (IsExtractedNCAMain(physical_name)) {
|
2018-06-14 17:25:40 -04:00
|
|
|
return file_info.dir().path();
|
2018-06-14 12:02:32 -04:00
|
|
|
}
|
2018-08-06 14:49:21 -04:00
|
|
|
|
|
|
|
return physical_name_as_qstring;
|
2018-06-14 12:02:32 -04:00
|
|
|
}
|
|
|
|
|
2017-02-23 18:29:00 -03:00
|
|
|
void GameList::RefreshGameDirectory() {
|
|
|
|
if (!UISettings::values.gamedir.isEmpty() && current_worker != nullptr) {
|
2018-07-02 12:13:26 -04:00
|
|
|
LOG_INFO(Frontend, "Change detected in the games directory. Reloading game list.");
|
2017-04-29 23:04:39 -03:00
|
|
|
search_field->clear();
|
2017-02-23 18:29:00 -03:00
|
|
|
PopulateAsync(UISettings::values.gamedir, UISettings::values.gamedir_deepscan);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-10 11:11:33 -04:00
|
|
|
static void GetMetadataFromControlNCA(const std::shared_ptr<FileSys::NCA>& nca,
|
|
|
|
std::vector<u8>& icon, std::string& name) {
|
|
|
|
const auto control_dir = FileSys::ExtractRomFS(nca->GetRomFS());
|
|
|
|
if (control_dir == nullptr)
|
|
|
|
return;
|
|
|
|
|
|
|
|
const auto nacp_file = control_dir->GetFile("control.nacp");
|
|
|
|
if (nacp_file == nullptr)
|
|
|
|
return;
|
|
|
|
FileSys::NACP nacp(nacp_file);
|
|
|
|
name = nacp.GetApplicationName();
|
|
|
|
|
|
|
|
FileSys::VirtualFile icon_file = nullptr;
|
|
|
|
for (const auto& language : FileSys::LANGUAGE_NAMES) {
|
|
|
|
icon_file = control_dir->GetFile("icon_" + std::string(language) + ".dat");
|
|
|
|
if (icon_file != nullptr) {
|
|
|
|
icon = icon_file->ReadAllBytes();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-31 13:21:34 -03:00
|
|
|
GameListWorker::GameListWorker(
|
|
|
|
FileSys::VirtualFilesystem vfs, QString dir_path, bool deep_scan,
|
|
|
|
const std::unordered_map<std::string, std::pair<QString, QString>>& compatibility_list)
|
|
|
|
: vfs(std::move(vfs)), dir_path(std::move(dir_path)), deep_scan(deep_scan),
|
|
|
|
compatibility_list(compatibility_list) {}
|
|
|
|
|
|
|
|
GameListWorker::~GameListWorker() = default;
|
|
|
|
|
2018-08-16 18:08:44 -03:00
|
|
|
void GameListWorker::AddInstalledTitlesToGameList(std::shared_ptr<FileSys::RegisteredCache> cache) {
|
|
|
|
const auto installed_games = cache->ListEntriesFilter(FileSys::TitleType::Application,
|
|
|
|
FileSys::ContentRecordType::Program);
|
2018-08-09 20:53:30 -04:00
|
|
|
|
|
|
|
for (const auto& game : installed_games) {
|
2018-08-16 18:08:44 -03:00
|
|
|
const auto& file = cache->GetEntryUnparsed(game);
|
2018-08-09 20:53:30 -04:00
|
|
|
std::unique_ptr<Loader::AppLoader> loader = Loader::GetLoader(file);
|
|
|
|
if (!loader)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
std::vector<u8> icon;
|
|
|
|
std::string name;
|
2018-08-20 05:21:50 -03:00
|
|
|
u64 program_id = 0;
|
2018-08-09 20:53:30 -04:00
|
|
|
loader->ReadProgramId(program_id);
|
|
|
|
|
2018-08-16 18:08:44 -03:00
|
|
|
const auto& control = cache->GetEntry(game.title_id, FileSys::ContentRecordType::Control);
|
2018-08-10 11:11:33 -04:00
|
|
|
if (control != nullptr)
|
|
|
|
GetMetadataFromControlNCA(control, icon, name);
|
2018-08-09 20:53:30 -04:00
|
|
|
emit EntryReady({
|
|
|
|
new GameListItemPath(
|
|
|
|
FormatGameName(file->GetFullPath()), icon, QString::fromStdString(name),
|
|
|
|
QString::fromStdString(Loader::GetFileTypeString(loader->GetFileType())),
|
|
|
|
program_id),
|
|
|
|
new GameListItem(
|
|
|
|
QString::fromStdString(Loader::GetFileTypeString(loader->GetFileType()))),
|
|
|
|
new GameListItemSize(file->GetSize()),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2018-08-16 18:08:44 -03:00
|
|
|
const auto control_data = cache->ListEntriesFilter(FileSys::TitleType::Application,
|
|
|
|
FileSys::ContentRecordType::Control);
|
2018-08-10 11:11:33 -04:00
|
|
|
|
|
|
|
for (const auto& entry : control_data) {
|
2018-08-16 18:08:44 -03:00
|
|
|
const auto nca = cache->GetEntry(entry);
|
2018-08-10 11:11:33 -04:00
|
|
|
if (nca != nullptr)
|
|
|
|
nca_control_map.insert_or_assign(entry.title_id, nca);
|
|
|
|
}
|
2018-08-11 22:48:27 -04:00
|
|
|
}
|
2018-08-10 11:11:33 -04:00
|
|
|
|
2018-08-11 22:48:27 -04:00
|
|
|
void GameListWorker::FillControlMap(const std::string& dir_path) {
|
|
|
|
const auto nca_control_callback = [this](u64* num_entries_out, const std::string& directory,
|
|
|
|
const std::string& virtual_name) -> bool {
|
2018-07-28 12:32:16 -04:00
|
|
|
std::string physical_name = directory + DIR_SEP + virtual_name;
|
|
|
|
|
|
|
|
if (stop_processing)
|
|
|
|
return false; // Breaks the callback loop.
|
|
|
|
|
|
|
|
bool is_dir = FileUtil::IsDirectory(physical_name);
|
|
|
|
QFileInfo file_info(physical_name.c_str());
|
|
|
|
if (!is_dir && file_info.suffix().toStdString() == "nca") {
|
2018-08-08 14:34:06 -04:00
|
|
|
auto nca =
|
|
|
|
std::make_shared<FileSys::NCA>(vfs->OpenFile(physical_name, FileSys::Mode::Read));
|
2018-07-28 12:32:16 -04:00
|
|
|
if (nca->GetType() == FileSys::NCAContentType::Control)
|
|
|
|
nca_control_map.insert_or_assign(nca->GetTitleId(), nca);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
FileUtil::ForeachDirectoryEntry(nullptr, dir_path, nca_control_callback);
|
2018-08-11 22:48:27 -04:00
|
|
|
}
|
2018-07-28 12:32:16 -04:00
|
|
|
|
2018-08-11 22:48:27 -04:00
|
|
|
void GameListWorker::AddFstEntriesToGameList(const std::string& dir_path, unsigned int recursion) {
|
|
|
|
const auto callback = [this, recursion](u64* num_entries_out, const std::string& directory,
|
|
|
|
const std::string& virtual_name) -> bool {
|
2015-09-01 01:35:33 -03:00
|
|
|
std::string physical_name = directory + DIR_SEP + virtual_name;
|
|
|
|
|
|
|
|
if (stop_processing)
|
2015-11-26 05:34:26 -03:00
|
|
|
return false; // Breaks the callback loop.
|
2015-09-01 01:35:33 -03:00
|
|
|
|
2017-04-17 23:53:40 -03:00
|
|
|
bool is_dir = FileUtil::IsDirectory(physical_name);
|
2018-06-14 12:02:32 -04:00
|
|
|
if (!is_dir &&
|
|
|
|
(HasSupportedFileExtension(physical_name) || IsExtractedNCAMain(physical_name))) {
|
2018-07-18 21:07:11 -04:00
|
|
|
std::unique_ptr<Loader::AppLoader> loader =
|
2018-08-03 11:51:48 -04:00
|
|
|
Loader::GetLoader(vfs->OpenFile(physical_name, FileSys::Mode::Read));
|
2018-07-28 12:32:16 -04:00
|
|
|
if (!loader || ((loader->GetFileType() == Loader::FileType::Unknown ||
|
|
|
|
loader->GetFileType() == Loader::FileType::Error) &&
|
|
|
|
!UISettings::values.show_unknown))
|
2015-11-26 05:34:26 -03:00
|
|
|
return true;
|
2015-09-01 01:35:33 -03:00
|
|
|
|
2018-07-28 12:32:16 -04:00
|
|
|
std::vector<u8> icon;
|
|
|
|
const auto res1 = loader->ReadIcon(icon);
|
|
|
|
|
2018-08-20 05:21:50 -03:00
|
|
|
u64 program_id = 0;
|
2018-07-28 12:32:16 -04:00
|
|
|
const auto res2 = loader->ReadProgramId(program_id);
|
|
|
|
|
|
|
|
std::string name = " ";
|
|
|
|
const auto res3 = loader->ReadTitle(name);
|
|
|
|
|
2018-08-09 21:37:35 -04:00
|
|
|
if (res1 != Loader::ResultStatus::Success && res3 != Loader::ResultStatus::Success &&
|
2018-07-28 12:32:16 -04:00
|
|
|
res2 == Loader::ResultStatus::Success) {
|
|
|
|
// Use from metadata pool.
|
|
|
|
if (nca_control_map.find(program_id) != nca_control_map.end()) {
|
|
|
|
const auto nca = nca_control_map[program_id];
|
2018-08-10 11:11:33 -04:00
|
|
|
GetMetadataFromControlNCA(nca, icon, name);
|
2018-07-28 12:32:16 -04:00
|
|
|
}
|
|
|
|
}
|
2016-12-15 06:55:03 -03:00
|
|
|
|
2018-08-29 10:42:53 -03:00
|
|
|
auto it = FindMatchingCompatibilityEntry(compatibility_list, program_id);
|
|
|
|
|
|
|
|
// The game list uses this as compatibility number for untested games
|
|
|
|
QString compatibility("99");
|
|
|
|
if (it != compatibility_list.end())
|
|
|
|
compatibility = it->second.first;
|
|
|
|
|
2015-09-01 01:35:33 -03:00
|
|
|
emit EntryReady({
|
2018-07-28 12:32:16 -04:00
|
|
|
new GameListItemPath(
|
|
|
|
FormatGameName(physical_name), icon, QString::fromStdString(name),
|
|
|
|
QString::fromStdString(Loader::GetFileTypeString(loader->GetFileType())),
|
|
|
|
program_id),
|
2018-08-29 10:42:53 -03:00
|
|
|
new GameListItemCompat(compatibility),
|
2016-09-17 21:38:01 -03:00
|
|
|
new GameListItem(
|
|
|
|
QString::fromStdString(Loader::GetFileTypeString(loader->GetFileType()))),
|
2015-09-01 01:35:33 -03:00
|
|
|
new GameListItemSize(FileUtil::GetSize(physical_name)),
|
|
|
|
});
|
2017-04-17 23:53:40 -03:00
|
|
|
} else if (is_dir && recursion > 0) {
|
|
|
|
watch_list.append(QString::fromStdString(physical_name));
|
2016-06-01 02:04:58 -04:00
|
|
|
AddFstEntriesToGameList(physical_name, recursion - 1);
|
2015-09-01 01:35:33 -03:00
|
|
|
}
|
|
|
|
|
2015-11-26 05:34:26 -03:00
|
|
|
return true;
|
2015-09-01 01:35:33 -03:00
|
|
|
};
|
2015-11-26 05:34:26 -03:00
|
|
|
|
|
|
|
FileUtil::ForeachDirectoryEntry(nullptr, dir_path, callback);
|
2015-09-01 01:35:33 -03:00
|
|
|
}
|
|
|
|
|
2016-09-17 21:38:01 -03:00
|
|
|
void GameListWorker::run() {
|
2015-09-01 01:35:33 -03:00
|
|
|
stop_processing = false;
|
2017-04-17 23:53:40 -03:00
|
|
|
watch_list.append(dir_path);
|
2018-08-11 22:48:27 -04:00
|
|
|
FillControlMap(dir_path.toStdString());
|
2018-08-25 19:59:19 -03:00
|
|
|
AddInstalledTitlesToGameList();
|
2015-09-06 03:59:04 -03:00
|
|
|
AddFstEntriesToGameList(dir_path.toStdString(), deep_scan ? 256 : 0);
|
2018-08-11 22:48:27 -04:00
|
|
|
nca_control_map.clear();
|
2017-04-17 23:53:40 -03:00
|
|
|
emit Finished(watch_list);
|
2015-09-01 01:35:33 -03:00
|
|
|
}
|
|
|
|
|
2016-09-17 21:38:01 -03:00
|
|
|
void GameListWorker::Cancel() {
|
2017-02-23 18:29:00 -03:00
|
|
|
this->disconnect();
|
2015-09-01 01:35:33 -03:00
|
|
|
stop_processing = true;
|
|
|
|
}
|