2024-07-30 11:42:36 +00:00
|
|
|
// SPDX-FileCopyrightText: 2002-2024 PCSX2 Dev Team
|
|
|
|
// SPDX-License-Identifier: GPL-3.0+
|
2021-12-13 12:12:54 +00:00
|
|
|
|
2023-05-13 04:30:41 +00:00
|
|
|
#include "GameListModel.h"
|
|
|
|
#include "GameListRefreshThread.h"
|
|
|
|
#include "GameListWidget.h"
|
|
|
|
#include "QtHost.h"
|
|
|
|
#include "QtUtils.h"
|
|
|
|
|
|
|
|
#include "pcsx2/GameList.h"
|
|
|
|
#include "pcsx2/Host.h"
|
|
|
|
|
2022-05-18 13:27:23 +00:00
|
|
|
#include "common/Assertions.h"
|
2023-12-22 11:17:46 +00:00
|
|
|
#include "common/Console.h"
|
2021-12-13 12:12:54 +00:00
|
|
|
#include "common/StringUtil.h"
|
|
|
|
|
2023-05-13 04:30:41 +00:00
|
|
|
#include "fmt/format.h"
|
2021-12-13 12:12:54 +00:00
|
|
|
|
|
|
|
#include <QtCore/QSortFilterProxyModel>
|
|
|
|
#include <QtGui/QPixmap>
|
|
|
|
#include <QtGui/QWheelEvent>
|
|
|
|
#include <QtWidgets/QApplication>
|
|
|
|
#include <QtWidgets/QHeaderView>
|
|
|
|
#include <QtWidgets/QMenu>
|
2022-08-31 19:05:05 +00:00
|
|
|
#include <QtWidgets/QScrollBar>
|
2021-12-13 12:12:54 +00:00
|
|
|
|
2022-05-07 06:32:44 +00:00
|
|
|
static const char* SUPPORTED_FORMATS_STRING = QT_TRANSLATE_NOOP(GameListWidget,
|
|
|
|
".bin/.iso (ISO Disc Images)\n"
|
2023-05-13 03:31:24 +00:00
|
|
|
".mdf (Media Descriptor File)\n"
|
2022-05-07 06:32:44 +00:00
|
|
|
".chd (Compressed Hunks of Data)\n"
|
|
|
|
".cso (Compressed ISO)\n"
|
2024-02-01 15:14:24 +00:00
|
|
|
".zso (Compressed ISO)\n"
|
2022-05-07 06:32:44 +00:00
|
|
|
".gz (Gzip Compressed ISO)");
|
|
|
|
|
2022-07-09 08:52:33 +00:00
|
|
|
static constexpr float MIN_SCALE = 0.1f;
|
|
|
|
static constexpr float MAX_SCALE = 2.0f;
|
|
|
|
|
2021-12-13 12:12:54 +00:00
|
|
|
class GameListSortModel final : public QSortFilterProxyModel
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
explicit GameListSortModel(GameListModel* parent)
|
|
|
|
: QSortFilterProxyModel(parent)
|
|
|
|
, m_model(parent)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2022-07-09 08:52:33 +00:00
|
|
|
void setFilterType(GameList::EntryType type)
|
|
|
|
{
|
|
|
|
m_filter_type = type;
|
|
|
|
invalidateRowsFilter();
|
|
|
|
}
|
|
|
|
void setFilterRegion(GameList::Region region)
|
|
|
|
{
|
|
|
|
m_filter_region = region;
|
|
|
|
invalidateRowsFilter();
|
|
|
|
}
|
|
|
|
void setFilterName(const QString& name)
|
|
|
|
{
|
|
|
|
m_filter_name = name;
|
|
|
|
invalidateRowsFilter();
|
|
|
|
}
|
|
|
|
|
2021-12-13 12:12:54 +00:00
|
|
|
bool filterAcceptsRow(int source_row, const QModelIndex& source_parent) const override
|
|
|
|
{
|
2022-07-09 08:52:33 +00:00
|
|
|
if (m_filter_type != GameList::EntryType::Count ||
|
|
|
|
m_filter_region != GameList::Region::Count ||
|
|
|
|
!m_filter_name.isEmpty())
|
|
|
|
{
|
|
|
|
const auto lock = GameList::GetLock();
|
|
|
|
const GameList::Entry* entry = GameList::GetEntryByIndex(source_row);
|
|
|
|
if (m_filter_type != GameList::EntryType::Count && entry->type != m_filter_type)
|
|
|
|
return false;
|
|
|
|
if (m_filter_region != GameList::Region::Count && entry->region != m_filter_region)
|
|
|
|
return false;
|
2022-09-08 18:19:35 +00:00
|
|
|
if (!m_filter_name.isEmpty() &&
|
|
|
|
!QString::fromStdString(entry->path).contains(m_filter_name, Qt::CaseInsensitive) &&
|
|
|
|
!QString::fromStdString(entry->serial).contains(m_filter_name, Qt::CaseInsensitive) &&
|
2023-09-09 22:36:10 +00:00
|
|
|
!QString::fromStdString(entry->title).contains(m_filter_name, Qt::CaseInsensitive) &&
|
|
|
|
!QString::fromStdString(entry->title_en).contains(m_filter_name, Qt::CaseInsensitive))
|
2022-07-09 08:52:33 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-12-13 12:12:54 +00:00
|
|
|
return QSortFilterProxyModel::filterAcceptsRow(source_row, source_parent);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool lessThan(const QModelIndex& source_left, const QModelIndex& source_right) const override
|
|
|
|
{
|
|
|
|
return m_model->lessThan(source_left, source_right, source_left.column());
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
GameListModel* m_model;
|
2022-07-09 08:52:33 +00:00
|
|
|
GameList::EntryType m_filter_type = GameList::EntryType::Count;
|
|
|
|
GameList::Region m_filter_region = GameList::Region::Count;
|
|
|
|
QString m_filter_name;
|
2021-12-13 12:12:54 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
GameListWidget::GameListWidget(QWidget* parent /* = nullptr */)
|
2022-07-09 08:52:33 +00:00
|
|
|
: QWidget(parent)
|
2021-12-13 12:12:54 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
GameListWidget::~GameListWidget() = default;
|
|
|
|
|
|
|
|
void GameListWidget::initialize()
|
|
|
|
{
|
2023-09-09 04:02:43 +00:00
|
|
|
const float cover_scale = Host::GetBaseFloatSettingValue("UI", "GameListCoverArtScale", 0.45f);
|
|
|
|
const bool show_cover_titles = Host::GetBaseBoolSettingValue("UI", "GameListShowCoverTitles", true);
|
|
|
|
m_model = new GameListModel(cover_scale, show_cover_titles, this);
|
2022-07-17 05:24:59 +00:00
|
|
|
m_model->updateCacheSize(width(), height());
|
2021-12-13 12:12:54 +00:00
|
|
|
|
|
|
|
m_sort_model = new GameListSortModel(m_model);
|
|
|
|
m_sort_model->setSourceModel(m_model);
|
2022-07-09 08:52:33 +00:00
|
|
|
|
|
|
|
m_ui.setupUi(this);
|
|
|
|
for (u32 type = 0; type < static_cast<u32>(GameList::EntryType::Count); type++)
|
|
|
|
{
|
|
|
|
m_ui.filterType->addItem(GameListModel::getIconForType(static_cast<GameList::EntryType>(type)),
|
|
|
|
qApp->translate("GameList", GameList::EntryTypeToDisplayString(static_cast<GameList::EntryType>(type))));
|
|
|
|
}
|
|
|
|
for (u32 region = 0; region < static_cast<u32>(GameList::Region::Count); region++)
|
|
|
|
{
|
|
|
|
m_ui.filterRegion->addItem(GameListModel::getIconForRegion(static_cast<GameList::Region>(region)),
|
|
|
|
qApp->translate("GameList", GameList::RegionToString(static_cast<GameList::Region>(region))));
|
|
|
|
}
|
|
|
|
|
|
|
|
connect(m_ui.viewGameList, &QPushButton::clicked, this, &GameListWidget::showGameList);
|
|
|
|
connect(m_ui.viewGameGrid, &QPushButton::clicked, this, &GameListWidget::showGameGrid);
|
|
|
|
connect(m_ui.gridScale, &QSlider::valueChanged, this, &GameListWidget::gridIntScale);
|
|
|
|
connect(m_ui.viewGridTitles, &QPushButton::toggled, this, &GameListWidget::setShowCoverTitles);
|
|
|
|
connect(m_ui.filterType, &QComboBox::currentIndexChanged, this, [this](int index) {
|
|
|
|
m_sort_model->setFilterType((index == 0) ? GameList::EntryType::Count : static_cast<GameList::EntryType>(index - 1));
|
|
|
|
});
|
|
|
|
connect(m_ui.filterRegion, &QComboBox::currentIndexChanged, this, [this](int index) {
|
|
|
|
m_sort_model->setFilterRegion((index == 0) ? GameList::Region::Count : static_cast<GameList::Region>(index - 1));
|
|
|
|
});
|
|
|
|
connect(m_ui.searchText, &QLineEdit::textChanged, this, [this](const QString& text) {
|
|
|
|
m_sort_model->setFilterName(text);
|
|
|
|
});
|
|
|
|
|
|
|
|
m_table_view = new QTableView(m_ui.stack);
|
2021-12-13 12:12:54 +00:00
|
|
|
m_table_view->setModel(m_sort_model);
|
|
|
|
m_table_view->setSortingEnabled(true);
|
|
|
|
m_table_view->setSelectionMode(QAbstractItemView::SingleSelection);
|
|
|
|
m_table_view->setSelectionBehavior(QAbstractItemView::SelectRows);
|
|
|
|
m_table_view->setContextMenuPolicy(Qt::CustomContextMenu);
|
|
|
|
m_table_view->setAlternatingRowColors(true);
|
|
|
|
m_table_view->setShowGrid(false);
|
|
|
|
m_table_view->setCurrentIndex({});
|
|
|
|
m_table_view->horizontalHeader()->setHighlightSections(false);
|
|
|
|
m_table_view->horizontalHeader()->setContextMenuPolicy(Qt::CustomContextMenu);
|
|
|
|
m_table_view->verticalHeader()->hide();
|
|
|
|
m_table_view->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOn);
|
2022-06-09 17:04:17 +00:00
|
|
|
m_table_view->setVerticalScrollMode(QAbstractItemView::ScrollMode::ScrollPerPixel);
|
2021-12-13 12:12:54 +00:00
|
|
|
|
|
|
|
loadTableViewColumnVisibilitySettings();
|
|
|
|
loadTableViewColumnSortSettings();
|
|
|
|
|
|
|
|
connect(m_table_view->selectionModel(), &QItemSelectionModel::currentChanged, this,
|
|
|
|
&GameListWidget::onSelectionModelCurrentChanged);
|
|
|
|
connect(m_table_view, &QTableView::activated, this, &GameListWidget::onTableViewItemActivated);
|
|
|
|
connect(m_table_view, &QTableView::customContextMenuRequested, this,
|
|
|
|
&GameListWidget::onTableViewContextMenuRequested);
|
|
|
|
connect(m_table_view->horizontalHeader(), &QHeaderView::customContextMenuRequested, this,
|
|
|
|
&GameListWidget::onTableViewHeaderContextMenuRequested);
|
|
|
|
connect(m_table_view->horizontalHeader(), &QHeaderView::sortIndicatorChanged, this,
|
|
|
|
&GameListWidget::onTableViewHeaderSortIndicatorChanged);
|
|
|
|
|
2022-07-09 08:52:33 +00:00
|
|
|
m_ui.stack->insertWidget(0, m_table_view);
|
2021-12-13 12:12:54 +00:00
|
|
|
|
2022-07-09 08:52:33 +00:00
|
|
|
m_list_view = new GameListGridListView(m_ui.stack);
|
2021-12-13 12:12:54 +00:00
|
|
|
m_list_view->setModel(m_sort_model);
|
|
|
|
m_list_view->setModelColumn(GameListModel::Column_Cover);
|
2023-04-25 10:02:11 +00:00
|
|
|
m_list_view->setSelectionMode(QAbstractItemView::SingleSelection);
|
2021-12-13 12:12:54 +00:00
|
|
|
m_list_view->setViewMode(QListView::IconMode);
|
|
|
|
m_list_view->setResizeMode(QListView::Adjust);
|
|
|
|
m_list_view->setUniformItemSizes(true);
|
|
|
|
m_list_view->setItemAlignment(Qt::AlignHCenter);
|
|
|
|
m_list_view->setContextMenuPolicy(Qt::CustomContextMenu);
|
|
|
|
m_list_view->setFrameStyle(QFrame::NoFrame);
|
2022-06-09 17:04:17 +00:00
|
|
|
m_list_view->setVerticalScrollMode(QAbstractItemView::ScrollMode::ScrollPerPixel);
|
2022-08-31 19:05:05 +00:00
|
|
|
m_list_view->verticalScrollBar()->setSingleStep(15);
|
2023-09-09 04:02:43 +00:00
|
|
|
onCoverScaleChanged();
|
2021-12-13 12:12:54 +00:00
|
|
|
|
|
|
|
connect(m_list_view->selectionModel(), &QItemSelectionModel::currentChanged, this,
|
|
|
|
&GameListWidget::onSelectionModelCurrentChanged);
|
|
|
|
connect(m_list_view, &GameListGridListView::zoomIn, this, &GameListWidget::gridZoomIn);
|
|
|
|
connect(m_list_view, &GameListGridListView::zoomOut, this, &GameListWidget::gridZoomOut);
|
|
|
|
connect(m_list_view, &QListView::activated, this, &GameListWidget::onListViewItemActivated);
|
|
|
|
connect(m_list_view, &QListView::customContextMenuRequested, this, &GameListWidget::onListViewContextMenuRequested);
|
2023-09-09 04:02:43 +00:00
|
|
|
connect(m_model, &GameListModel::coverScaleChanged, this, &GameListWidget::onCoverScaleChanged);
|
2021-12-13 12:12:54 +00:00
|
|
|
|
2022-07-09 08:52:33 +00:00
|
|
|
m_ui.stack->insertWidget(1, m_list_view);
|
2021-12-13 12:12:54 +00:00
|
|
|
|
2022-07-09 08:52:33 +00:00
|
|
|
m_empty_widget = new QWidget(m_ui.stack);
|
2022-05-07 06:32:44 +00:00
|
|
|
m_empty_ui.setupUi(m_empty_widget);
|
|
|
|
m_empty_ui.supportedFormats->setText(qApp->translate("GameListWidget", SUPPORTED_FORMATS_STRING));
|
|
|
|
connect(m_empty_ui.addGameDirectory, &QPushButton::clicked, this, [this]() { emit addGameDirectoryRequested(); });
|
|
|
|
connect(m_empty_ui.scanForNewGames, &QPushButton::clicked, this, [this]() { refresh(false); });
|
2022-07-09 08:52:33 +00:00
|
|
|
m_ui.stack->insertWidget(2, m_empty_widget);
|
2022-05-07 06:32:44 +00:00
|
|
|
|
2022-05-24 12:37:44 +00:00
|
|
|
if (Host::GetBaseBoolSettingValue("UI", "GameListGridView", false))
|
2022-07-09 08:52:33 +00:00
|
|
|
m_ui.stack->setCurrentIndex(1);
|
2021-12-13 12:12:54 +00:00
|
|
|
else
|
2022-07-09 08:52:33 +00:00
|
|
|
m_ui.stack->setCurrentIndex(0);
|
2021-12-13 12:12:54 +00:00
|
|
|
|
2022-07-09 08:52:33 +00:00
|
|
|
updateToolbar();
|
2021-12-13 12:12:54 +00:00
|
|
|
resizeTableViewColumnsToFit();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GameListWidget::isShowingGameList() const
|
|
|
|
{
|
2022-07-09 08:52:33 +00:00
|
|
|
return m_ui.stack->currentIndex() == 0;
|
2021-12-13 12:12:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool GameListWidget::isShowingGameGrid() const
|
|
|
|
{
|
2022-07-09 08:52:33 +00:00
|
|
|
return m_ui.stack->currentIndex() == 1;
|
2021-12-13 12:12:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool GameListWidget::getShowGridCoverTitles() const
|
|
|
|
{
|
|
|
|
return m_model->getShowCoverTitles();
|
|
|
|
}
|
|
|
|
|
|
|
|
void GameListWidget::refresh(bool invalidate_cache)
|
|
|
|
{
|
2022-05-15 08:16:24 +00:00
|
|
|
cancelRefresh();
|
2021-12-13 12:12:54 +00:00
|
|
|
|
|
|
|
m_refresh_thread = new GameListRefreshThread(invalidate_cache);
|
|
|
|
connect(m_refresh_thread, &GameListRefreshThread::refreshProgress, this, &GameListWidget::onRefreshProgress,
|
|
|
|
Qt::QueuedConnection);
|
|
|
|
connect(m_refresh_thread, &GameListRefreshThread::refreshComplete, this, &GameListWidget::onRefreshComplete,
|
|
|
|
Qt::QueuedConnection);
|
|
|
|
m_refresh_thread->start();
|
|
|
|
}
|
|
|
|
|
2022-05-15 08:16:24 +00:00
|
|
|
void GameListWidget::cancelRefresh()
|
|
|
|
{
|
|
|
|
if (!m_refresh_thread)
|
|
|
|
return;
|
|
|
|
|
|
|
|
m_refresh_thread->cancel();
|
|
|
|
m_refresh_thread->wait();
|
2022-09-08 00:37:40 +00:00
|
|
|
|
|
|
|
// Cancelling might not be instant if we're say, scanning a gzip dump. Wait until it's done.
|
|
|
|
while (m_refresh_thread)
|
|
|
|
QApplication::processEvents(QEventLoop::ExcludeUserInputEvents, 1);
|
2022-05-15 08:16:24 +00:00
|
|
|
}
|
|
|
|
|
2023-09-09 03:30:53 +00:00
|
|
|
void GameListWidget::reloadThemeSpecificImages()
|
2022-07-24 13:52:22 +00:00
|
|
|
{
|
2023-09-09 03:30:53 +00:00
|
|
|
m_model->reloadThemeSpecificImages();
|
2022-07-24 13:52:22 +00:00
|
|
|
}
|
|
|
|
|
2021-12-13 12:12:54 +00:00
|
|
|
void GameListWidget::onRefreshProgress(const QString& status, int current, int total)
|
|
|
|
{
|
2022-05-07 06:32:44 +00:00
|
|
|
// switch away from the placeholder while we scan, in case we find anything
|
2022-07-09 08:52:33 +00:00
|
|
|
if (m_ui.stack->currentIndex() == 2)
|
|
|
|
m_ui.stack->setCurrentIndex(Host::GetBaseBoolSettingValue("UI", "GameListGridView", false) ? 1 : 0);
|
2022-05-07 06:32:44 +00:00
|
|
|
|
2021-12-13 12:12:54 +00:00
|
|
|
m_model->refresh();
|
|
|
|
emit refreshProgress(status, current, total);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GameListWidget::onRefreshComplete()
|
|
|
|
{
|
|
|
|
m_model->refresh();
|
|
|
|
emit refreshComplete();
|
|
|
|
|
|
|
|
pxAssertRel(m_refresh_thread, "Has a refresh thread");
|
|
|
|
m_refresh_thread->wait();
|
|
|
|
delete m_refresh_thread;
|
|
|
|
m_refresh_thread = nullptr;
|
2022-05-07 06:32:44 +00:00
|
|
|
|
|
|
|
// if we still had no games, switch to the helper widget
|
|
|
|
if (m_model->rowCount() == 0)
|
2022-07-09 08:52:33 +00:00
|
|
|
m_ui.stack->setCurrentIndex(2);
|
2021-12-13 12:12:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GameListWidget::onSelectionModelCurrentChanged(const QModelIndex& current, const QModelIndex& previous)
|
|
|
|
{
|
|
|
|
const QModelIndex source_index = m_sort_model->mapToSource(current);
|
|
|
|
if (!source_index.isValid() || source_index.row() >= static_cast<int>(GameList::GetEntryCount()))
|
|
|
|
return;
|
|
|
|
|
|
|
|
emit selectionChanged();
|
|
|
|
}
|
|
|
|
|
|
|
|
void GameListWidget::onTableViewItemActivated(const QModelIndex& index)
|
|
|
|
{
|
|
|
|
const QModelIndex source_index = m_sort_model->mapToSource(index);
|
|
|
|
if (!source_index.isValid() || source_index.row() >= static_cast<int>(GameList::GetEntryCount()))
|
|
|
|
return;
|
|
|
|
|
|
|
|
emit entryActivated();
|
|
|
|
}
|
|
|
|
|
|
|
|
void GameListWidget::onTableViewContextMenuRequested(const QPoint& point)
|
|
|
|
{
|
|
|
|
emit entryContextMenuRequested(m_table_view->mapToGlobal(point));
|
|
|
|
}
|
|
|
|
|
|
|
|
void GameListWidget::onListViewItemActivated(const QModelIndex& index)
|
|
|
|
{
|
|
|
|
const QModelIndex source_index = m_sort_model->mapToSource(index);
|
|
|
|
if (!source_index.isValid() || source_index.row() >= static_cast<int>(GameList::GetEntryCount()))
|
|
|
|
return;
|
|
|
|
|
|
|
|
emit entryActivated();
|
|
|
|
}
|
|
|
|
|
|
|
|
void GameListWidget::onListViewContextMenuRequested(const QPoint& point)
|
|
|
|
{
|
|
|
|
emit entryContextMenuRequested(m_list_view->mapToGlobal(point));
|
|
|
|
}
|
|
|
|
|
|
|
|
void GameListWidget::onTableViewHeaderContextMenuRequested(const QPoint& point)
|
|
|
|
{
|
|
|
|
QMenu menu;
|
|
|
|
|
|
|
|
for (int column = 0; column < GameListModel::Column_Count; column++)
|
|
|
|
{
|
|
|
|
if (column == GameListModel::Column_Cover)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
QAction* action = menu.addAction(m_model->getColumnDisplayName(column));
|
|
|
|
action->setCheckable(true);
|
|
|
|
action->setChecked(!m_table_view->isColumnHidden(column));
|
|
|
|
connect(action, &QAction::toggled, [this, column](bool enabled) {
|
|
|
|
m_table_view->setColumnHidden(column, !enabled);
|
|
|
|
saveTableViewColumnVisibilitySettings(column);
|
|
|
|
resizeTableViewColumnsToFit();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
menu.exec(m_table_view->mapToGlobal(point));
|
|
|
|
}
|
|
|
|
|
|
|
|
void GameListWidget::onTableViewHeaderSortIndicatorChanged(int, Qt::SortOrder)
|
|
|
|
{
|
|
|
|
saveTableViewColumnSortSettings();
|
|
|
|
}
|
|
|
|
|
2023-09-09 04:02:43 +00:00
|
|
|
void GameListWidget::onCoverScaleChanged()
|
|
|
|
{
|
|
|
|
m_model->updateCacheSize(width(), height());
|
|
|
|
|
|
|
|
m_list_view->setSpacing(m_model->getCoverArtSpacing());
|
|
|
|
|
|
|
|
QFont font;
|
2024-08-18 15:47:41 +00:00
|
|
|
font.setPointSizeF(20.0f * m_model->getCoverScale());
|
2023-09-09 04:02:43 +00:00
|
|
|
m_list_view->setFont(font);
|
|
|
|
}
|
|
|
|
|
2021-12-13 12:12:54 +00:00
|
|
|
void GameListWidget::listZoom(float delta)
|
|
|
|
{
|
|
|
|
const float new_scale = std::clamp(m_model->getCoverScale() + delta, MIN_SCALE, MAX_SCALE);
|
2022-09-07 07:44:10 +00:00
|
|
|
Host::SetBaseFloatSettingValue("UI", "GameListCoverArtScale", new_scale);
|
|
|
|
Host::CommitBaseSettingChanges();
|
2021-12-13 12:12:54 +00:00
|
|
|
m_model->setCoverScale(new_scale);
|
2022-07-09 08:52:33 +00:00
|
|
|
updateToolbar();
|
2021-12-13 12:12:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GameListWidget::gridZoomIn()
|
|
|
|
{
|
|
|
|
listZoom(0.05f);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GameListWidget::gridZoomOut()
|
|
|
|
{
|
|
|
|
listZoom(-0.05f);
|
|
|
|
}
|
|
|
|
|
2022-07-09 08:52:33 +00:00
|
|
|
void GameListWidget::gridIntScale(int int_scale)
|
|
|
|
{
|
|
|
|
const float new_scale = std::clamp(static_cast<float>(int_scale) / 100.0f, MIN_SCALE, MAX_SCALE);
|
|
|
|
|
2022-09-07 07:44:10 +00:00
|
|
|
Host::SetBaseFloatSettingValue("UI", "GameListCoverArtScale", new_scale);
|
|
|
|
Host::CommitBaseSettingChanges();
|
2022-07-09 08:52:33 +00:00
|
|
|
m_model->setCoverScale(new_scale);
|
|
|
|
updateToolbar();
|
|
|
|
}
|
|
|
|
|
2021-12-13 12:12:54 +00:00
|
|
|
void GameListWidget::refreshGridCovers()
|
|
|
|
{
|
|
|
|
m_model->refreshCovers();
|
|
|
|
}
|
|
|
|
|
|
|
|
void GameListWidget::showGameList()
|
|
|
|
{
|
2022-07-09 08:52:33 +00:00
|
|
|
if (m_ui.stack->currentIndex() == 0 || m_model->rowCount() == 0)
|
|
|
|
{
|
|
|
|
// We can click the toolbar multiple times, so keep it correct.
|
|
|
|
updateToolbar();
|
2021-12-13 12:12:54 +00:00
|
|
|
return;
|
2022-07-09 08:52:33 +00:00
|
|
|
}
|
2021-12-13 12:12:54 +00:00
|
|
|
|
2022-09-07 07:44:10 +00:00
|
|
|
Host::SetBaseBoolSettingValue("UI", "GameListGridView", false);
|
|
|
|
Host::CommitBaseSettingChanges();
|
2022-07-09 08:52:33 +00:00
|
|
|
m_ui.stack->setCurrentIndex(0);
|
2021-12-13 12:12:54 +00:00
|
|
|
resizeTableViewColumnsToFit();
|
2022-07-09 08:52:33 +00:00
|
|
|
updateToolbar();
|
|
|
|
emit layoutChange();
|
2021-12-13 12:12:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GameListWidget::showGameGrid()
|
|
|
|
{
|
2022-07-09 08:52:33 +00:00
|
|
|
if (m_ui.stack->currentIndex() == 1 || m_model->rowCount() == 0)
|
|
|
|
{
|
|
|
|
// We can click the toolbar multiple times, so keep it correct.
|
|
|
|
updateToolbar();
|
2021-12-13 12:12:54 +00:00
|
|
|
return;
|
2022-07-09 08:52:33 +00:00
|
|
|
}
|
2021-12-13 12:12:54 +00:00
|
|
|
|
2022-09-07 07:44:10 +00:00
|
|
|
Host::SetBaseBoolSettingValue("UI", "GameListGridView", true);
|
|
|
|
Host::CommitBaseSettingChanges();
|
2022-07-09 08:52:33 +00:00
|
|
|
m_ui.stack->setCurrentIndex(1);
|
|
|
|
updateToolbar();
|
|
|
|
emit layoutChange();
|
2021-12-13 12:12:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GameListWidget::setShowCoverTitles(bool enabled)
|
|
|
|
{
|
|
|
|
if (m_model->getShowCoverTitles() == enabled)
|
|
|
|
return;
|
|
|
|
|
2022-09-07 07:44:10 +00:00
|
|
|
Host::SetBaseBoolSettingValue("UI", "GameListShowCoverTitles", enabled);
|
|
|
|
Host::CommitBaseSettingChanges();
|
2021-12-13 12:12:54 +00:00
|
|
|
m_model->setShowCoverTitles(enabled);
|
|
|
|
if (isShowingGameGrid())
|
|
|
|
m_model->refresh();
|
2022-07-09 08:52:33 +00:00
|
|
|
updateToolbar();
|
|
|
|
emit layoutChange();
|
2021-12-13 12:12:54 +00:00
|
|
|
}
|
|
|
|
|
2022-07-09 08:52:33 +00:00
|
|
|
void GameListWidget::updateToolbar()
|
|
|
|
{
|
|
|
|
const bool grid_view = isShowingGameGrid();
|
|
|
|
{
|
|
|
|
QSignalBlocker sb(m_ui.viewGameGrid);
|
|
|
|
m_ui.viewGameGrid->setChecked(grid_view);
|
|
|
|
}
|
|
|
|
{
|
|
|
|
QSignalBlocker sb(m_ui.viewGameList);
|
|
|
|
m_ui.viewGameList->setChecked(!grid_view);
|
|
|
|
}
|
|
|
|
{
|
|
|
|
QSignalBlocker sb(m_ui.viewGridTitles);
|
|
|
|
m_ui.viewGridTitles->setChecked(m_model->getShowCoverTitles());
|
|
|
|
}
|
|
|
|
{
|
|
|
|
QSignalBlocker sb(m_ui.gridScale);
|
|
|
|
m_ui.gridScale->setValue(static_cast<int>(m_model->getCoverScale() * 100.0f));
|
|
|
|
}
|
|
|
|
|
|
|
|
m_ui.viewGridTitles->setEnabled(grid_view);
|
|
|
|
m_ui.gridScale->setEnabled(grid_view);
|
|
|
|
}
|
|
|
|
|
2021-12-13 12:12:54 +00:00
|
|
|
void GameListWidget::resizeEvent(QResizeEvent* event)
|
|
|
|
{
|
2022-07-09 08:52:33 +00:00
|
|
|
QWidget::resizeEvent(event);
|
2021-12-13 12:12:54 +00:00
|
|
|
resizeTableViewColumnsToFit();
|
2022-07-17 05:24:59 +00:00
|
|
|
m_model->updateCacheSize(width(), height());
|
2021-12-13 12:12:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GameListWidget::resizeTableViewColumnsToFit()
|
|
|
|
{
|
|
|
|
QtUtils::ResizeColumnsForTableView(m_table_view, {
|
2022-02-02 23:50:00 +00:00
|
|
|
45, // type
|
2021-12-13 12:12:54 +00:00
|
|
|
80, // code
|
|
|
|
-1, // title
|
|
|
|
-1, // file title
|
2022-07-17 13:39:01 +00:00
|
|
|
65, // crc
|
2022-10-22 04:52:01 +00:00
|
|
|
80, // time played
|
|
|
|
80, // last played
|
2021-12-13 12:12:54 +00:00
|
|
|
80, // size
|
2022-02-03 23:53:47 +00:00
|
|
|
60, // region
|
2023-02-03 04:34:04 +00:00
|
|
|
120 // compatibility
|
2021-12-13 12:12:54 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
static std::string getColumnVisibilitySettingsKeyName(int column)
|
|
|
|
{
|
|
|
|
return StringUtil::StdStringFromFormat("Show%s",
|
|
|
|
GameListModel::getColumnName(static_cast<GameListModel::Column>(column)));
|
|
|
|
}
|
|
|
|
|
|
|
|
void GameListWidget::loadTableViewColumnVisibilitySettings()
|
|
|
|
{
|
|
|
|
static constexpr std::array<bool, GameListModel::Column_Count> DEFAULT_VISIBILITY = {{
|
|
|
|
true, // type
|
|
|
|
true, // code
|
|
|
|
true, // title
|
|
|
|
false, // file title
|
|
|
|
false, // crc
|
2022-10-22 04:52:01 +00:00
|
|
|
true, // time played
|
|
|
|
true, // last played
|
2021-12-13 12:12:54 +00:00
|
|
|
true, // size
|
|
|
|
true, // region
|
|
|
|
true // compatibility
|
|
|
|
}};
|
|
|
|
|
|
|
|
for (int column = 0; column < GameListModel::Column_Count; column++)
|
|
|
|
{
|
2022-05-24 12:37:44 +00:00
|
|
|
const bool visible = Host::GetBaseBoolSettingValue(
|
2021-12-13 12:12:54 +00:00
|
|
|
"GameListTableView", getColumnVisibilitySettingsKeyName(column).c_str(), DEFAULT_VISIBILITY[column]);
|
|
|
|
m_table_view->setColumnHidden(column, !visible);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void GameListWidget::saveTableViewColumnVisibilitySettings()
|
|
|
|
{
|
|
|
|
for (int column = 0; column < GameListModel::Column_Count; column++)
|
|
|
|
{
|
|
|
|
const bool visible = !m_table_view->isColumnHidden(column);
|
2022-09-07 07:44:10 +00:00
|
|
|
Host::SetBaseBoolSettingValue("GameListTableView", getColumnVisibilitySettingsKeyName(column).c_str(), visible);
|
|
|
|
Host::CommitBaseSettingChanges();
|
2021-12-13 12:12:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void GameListWidget::saveTableViewColumnVisibilitySettings(int column)
|
|
|
|
{
|
|
|
|
const bool visible = !m_table_view->isColumnHidden(column);
|
2022-09-07 07:44:10 +00:00
|
|
|
Host::SetBaseBoolSettingValue("GameListTableView", getColumnVisibilitySettingsKeyName(column).c_str(), visible);
|
|
|
|
Host::CommitBaseSettingChanges();
|
2021-12-13 12:12:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GameListWidget::loadTableViewColumnSortSettings()
|
|
|
|
{
|
|
|
|
const GameListModel::Column DEFAULT_SORT_COLUMN = GameListModel::Column_Type;
|
|
|
|
const bool DEFAULT_SORT_DESCENDING = false;
|
|
|
|
|
|
|
|
const GameListModel::Column sort_column =
|
2022-05-24 12:37:44 +00:00
|
|
|
GameListModel::getColumnIdForName(Host::GetBaseStringSettingValue("GameListTableView", "SortColumn"))
|
2021-12-13 12:12:54 +00:00
|
|
|
.value_or(DEFAULT_SORT_COLUMN);
|
|
|
|
const bool sort_descending =
|
2022-05-24 12:37:44 +00:00
|
|
|
Host::GetBaseBoolSettingValue("GameListTableView", "SortDescending", DEFAULT_SORT_DESCENDING);
|
2024-08-22 17:37:55 +00:00
|
|
|
const Qt::SortOrder sort_order = sort_descending ? Qt::DescendingOrder : Qt::AscendingOrder;
|
|
|
|
m_sort_model->sort(sort_column, sort_order);
|
|
|
|
if (QHeaderView* hv = m_table_view->horizontalHeader())
|
|
|
|
hv->setSortIndicator(sort_column, sort_order);
|
2021-12-13 12:12:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GameListWidget::saveTableViewColumnSortSettings()
|
|
|
|
{
|
|
|
|
const int sort_column = m_table_view->horizontalHeader()->sortIndicatorSection();
|
|
|
|
const bool sort_descending = (m_table_view->horizontalHeader()->sortIndicatorOrder() == Qt::DescendingOrder);
|
|
|
|
|
|
|
|
if (sort_column >= 0 && sort_column < GameListModel::Column_Count)
|
|
|
|
{
|
2022-09-07 07:44:10 +00:00
|
|
|
Host::SetBaseStringSettingValue(
|
2021-12-13 12:12:54 +00:00
|
|
|
"GameListTableView", "SortColumn", GameListModel::getColumnName(static_cast<GameListModel::Column>(sort_column)));
|
|
|
|
}
|
|
|
|
|
2022-09-07 07:44:10 +00:00
|
|
|
Host::SetBaseBoolSettingValue("GameListTableView", "SortDescending", sort_descending);
|
|
|
|
Host::CommitBaseSettingChanges();
|
2021-12-13 12:12:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const GameList::Entry* GameListWidget::getSelectedEntry() const
|
|
|
|
{
|
2022-07-09 08:52:33 +00:00
|
|
|
if (m_ui.stack->currentIndex() == 0)
|
2021-12-13 12:12:54 +00:00
|
|
|
{
|
|
|
|
const QItemSelectionModel* selection_model = m_table_view->selectionModel();
|
|
|
|
if (!selection_model->hasSelection())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
const QModelIndexList selected_rows = selection_model->selectedRows();
|
|
|
|
if (selected_rows.empty())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
const QModelIndex source_index = m_sort_model->mapToSource(selected_rows[0]);
|
|
|
|
if (!source_index.isValid())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
return GameList::GetEntryByIndex(source_index.row());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
const QItemSelectionModel* selection_model = m_list_view->selectionModel();
|
|
|
|
if (!selection_model->hasSelection())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
const QModelIndex source_index = m_sort_model->mapToSource(selection_model->currentIndex());
|
|
|
|
if (!source_index.isValid())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
return GameList::GetEntryByIndex(source_index.row());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-04 08:03:30 +00:00
|
|
|
void GameListWidget::rescanFile(const std::string& path)
|
|
|
|
{
|
|
|
|
// We can't do this while there's a VM running, because of CDVD state... ugh.
|
|
|
|
if (QtHost::IsVMValid())
|
|
|
|
{
|
|
|
|
Console.Error(fmt::format("Can't re-scan ELF at '{}' because we have a VM running.", path));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
GameList::RescanPath(path);
|
|
|
|
m_model->refresh();
|
|
|
|
}
|
|
|
|
|
2021-12-13 12:12:54 +00:00
|
|
|
GameListGridListView::GameListGridListView(QWidget* parent /*= nullptr*/)
|
|
|
|
: QListView(parent)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void GameListGridListView::wheelEvent(QWheelEvent* e)
|
|
|
|
{
|
|
|
|
if (e->modifiers() & Qt::ControlModifier)
|
|
|
|
{
|
|
|
|
int dy = e->angleDelta().y();
|
|
|
|
if (dy != 0)
|
|
|
|
{
|
|
|
|
if (dy < 0)
|
|
|
|
zoomOut();
|
|
|
|
else
|
|
|
|
zoomIn();
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QListView::wheelEvent(e);
|
|
|
|
}
|