rpcs3/rpcs3/rpcs3qt/pkg_install_dialog.cpp

269 lines
8.3 KiB
C++

#include "pkg_install_dialog.h"
#include "game_compatibility.h"
#include "numbered_widget_item.h"
#include "richtext_item_delegate.h"
#include "qt_utils.h"
#include "Emu/system_utils.hpp"
#include "Utilities/File.h"
#include <QPushButton>
#include <QFileInfo>
#include <QHBoxLayout>
#include <QToolButton>
#include <QCheckBox>
enum Roles
{
FullPathRole = Qt::UserRole + 0,
ChangelogRole = Qt::UserRole + 1,
TitleRole = Qt::UserRole + 2,
TitleIdRole = Qt::UserRole + 3,
VersionRole = Qt::UserRole + 4,
DataSizeRole = Qt::UserRole + 5,
};
pkg_install_dialog::pkg_install_dialog(const QStringList& paths, game_compatibility* compat, QWidget* parent)
: QDialog(parent)
{
m_dir_list = new QListWidget(this);
m_dir_list->setItemDelegate(new richtext_item_delegate(m_dir_list->itemDelegate()));
for (const QString& path : paths)
{
const compat::package_info info = game_compatibility::GetPkgInfo(path, compat);
if (!info.is_valid)
{
continue;
}
const QFileInfo file_info(path);
// We have to build our complicated localized string in some annoying manner
QString accumulated_info;
QString tooltip;
const auto append_comma = [&accumulated_info]()
{
if (!accumulated_info.isEmpty())
{
accumulated_info += ", ";
}
};
if (!info.title_id.isEmpty())
{
accumulated_info = info.title_id;
}
if (info.type != compat::package_type::other)
{
append_comma();
if (info.type == compat::package_type::dlc)
{
accumulated_info += tr("DLC", "Package type info (DLC)");
}
else
{
accumulated_info += tr("Update", "Package type info (Update)");
}
}
else if (!info.local_cat.isEmpty())
{
append_comma();
accumulated_info += info.local_cat;
}
if (!info.version.isEmpty())
{
append_comma();
accumulated_info += tr("v.%0", "Version info").arg(info.version);
}
if (info.changelog.isEmpty())
{
tooltip = tr("No info", "Changelog info placeholder");
}
else
{
tooltip = tr("Changelog:\n\n%0", "Changelog info").arg(info.changelog);
}
append_comma();
accumulated_info += file_info.fileName();
const QString text = tr("<b>%0</b> (%1) - %2", "Package text").arg(info.title.simplified())
.arg(accumulated_info).arg(gui::utils::format_byte_size(info.data_size));
QListWidgetItem* item = new numbered_widget_item(text, m_dir_list);
item->setData(Roles::FullPathRole, info.path);
item->setData(Roles::ChangelogRole, info.changelog);
item->setData(Roles::TitleRole, info.title);
item->setData(Roles::TitleIdRole, info.title_id);
item->setData(Roles::VersionRole, info.version);
item->setData(Roles::DataSizeRole, static_cast<qulonglong>(info.data_size));
item->setToolTip(tooltip);
}
m_dir_list->sortItems();
m_dir_list->setCurrentRow(0);
m_dir_list->setMinimumWidth((m_dir_list->sizeHintForColumn(0) * 125) / 100);
// Create contextual label (updated in connect(m_dir_list, &QListWidget::itemChanged ...))
QLabel* installation_info = new QLabel();
installation_info->setTextFormat(Qt::RichText); // Support HTML tags
// Create buttons
QDialogButtonBox* buttons = new QDialogButtonBox(QDialogButtonBox::Cancel | QDialogButtonBox::Ok);
buttons->button(QDialogButtonBox::Ok)->setText(tr("Install"));
buttons->button(QDialogButtonBox::Ok)->setDefault(true);
connect(buttons, &QDialogButtonBox::clicked, this, [this, buttons](QAbstractButton* button)
{
if (button == buttons->button(QDialogButtonBox::Ok))
{
accept();
}
else if (button == buttons->button(QDialogButtonBox::Cancel))
{
reject();
}
});
QHBoxLayout* hbox = nullptr;
if (m_dir_list->count() > 1)
{
for (int i = 0; i < m_dir_list->count(); i++)
{
if (QListWidgetItem* item = m_dir_list->item(i))
{
item->setFlags(item->flags() | Qt::ItemIsUserCheckable);
item->setCheckState(Qt::Checked);
}
}
connect(m_dir_list, &QListWidget::itemChanged, this, [this, installation_info, buttons](QListWidgetItem*)
{
update_info(installation_info, buttons);
});
QToolButton* move_up = new QToolButton;
move_up->setArrowType(Qt::UpArrow);
move_up->setToolTip(tr("Move selected item up"));
connect(move_up, &QToolButton::clicked, this, [this]() { move_item(-1); });
QToolButton* move_down = new QToolButton;
move_down->setArrowType(Qt::DownArrow);
move_down->setToolTip(tr("Move selected item down"));
connect(move_down, &QToolButton::clicked, this, [this]() { move_item(1); });
hbox = new QHBoxLayout;
hbox->addStretch();
hbox->addWidget(move_up);
hbox->addWidget(move_down);
}
QLabel* description = new QLabel(m_dir_list->count() == 1
? tr("Do you want to install this package?")
: tr("You are about to install multiple packages.\nReorder and/or exclude them if needed, then click \"Install\" to proceed.")
);
QLabel* label = new QLabel(tr("Would you like to precompile caches and install shortcuts to the installed software?"));
QCheckBox* precompile_check = new QCheckBox(tr("Precompile caches"));
connect(precompile_check, &QCheckBox::checkStateChanged, this, [this](Qt::CheckState state){ m_precompile_caches = state != Qt::CheckState::Unchecked; });
precompile_check->setChecked(true);
QCheckBox* desk_check = new QCheckBox(tr("Add desktop shortcut(s)"));
connect(desk_check, &QCheckBox::checkStateChanged, this, [this](Qt::CheckState state){ m_create_desktop_shortcuts = state != Qt::CheckState::Unchecked; });
#ifdef _WIN32
QCheckBox* quick_check = new QCheckBox(tr("Add Start menu shortcut(s)"));
#elif defined(__APPLE__)
QCheckBox* quick_check = new QCheckBox(tr("Add dock shortcut(s)"));
#else
QCheckBox* quick_check = new QCheckBox(tr("Add launcher shortcut(s)"));
#endif
connect(quick_check, &QCheckBox::checkStateChanged, this, [this](Qt::CheckState state){ m_create_app_shortcut = state != Qt::CheckState::Unchecked; });
QVBoxLayout* vbox = new QVBoxLayout;
vbox->addWidget(description);
if (hbox) vbox->addLayout(hbox);
vbox->addWidget(m_dir_list);
vbox->addWidget(label);
vbox->addWidget(precompile_check);
vbox->addWidget(desk_check);
vbox->addWidget(quick_check);
vbox->addWidget(installation_info);
vbox->addWidget(buttons);
setLayout(vbox);
setWindowTitle(tr("PKG Installation"));
setObjectName("pkg_install_dialog");
update_info(installation_info, buttons); // Just to show and check available and required size
}
void pkg_install_dialog::update_info(QLabel* installation_info, QDialogButtonBox* buttons) const
{
u64 data_size = 0;
u64 free_space = 0;
// Retrieve disk space info on data path's drive
if (fs::device_stat stat{}; fs::statfs(rpcs3::utils::get_hdd0_game_dir(), stat))
{
free_space = stat.avail_free;
}
for (int i = 0; i < m_dir_list->count(); i++)
{
if (m_dir_list->count() == 1 || m_dir_list->item(i)->checkState() == Qt::Checked)
{
data_size += m_dir_list->item(i)->data(Roles::DataSizeRole).toULongLong();
}
}
installation_info->setText(gui::utils::make_paragraph(
tr("Installation path: %0\nAvailable disk space: %1%2\nRequired disk space: %3")
.arg(QString::fromStdString(rpcs3::utils::get_hdd0_game_dir()))
.arg(gui::utils::format_byte_size(free_space))
.arg(data_size <= free_space ? QString() : tr(" - <b>NOT ENOUGH SPACE</b>"))
.arg(gui::utils::format_byte_size(data_size))));
buttons->button(QDialogButtonBox::Ok)->setEnabled(data_size && (data_size <= free_space));
}
void pkg_install_dialog::move_item(int offset) const
{
const int src_index = m_dir_list->currentRow();
const int dest_index = src_index + offset;
if (src_index >= 0 && src_index < m_dir_list->count() &&
dest_index >= 0 && dest_index < m_dir_list->count())
{
QListWidgetItem* item = m_dir_list->takeItem(src_index);
m_dir_list->insertItem(dest_index, item);
m_dir_list->setCurrentItem(item);
}
}
std::vector<compat::package_info> pkg_install_dialog::get_paths_to_install() const
{
std::vector<compat::package_info> result;
for (int i = 0; i < m_dir_list->count(); i++)
{
const QListWidgetItem* item = m_dir_list->item(i);
if (item && (m_dir_list->count() == 1 || item->checkState() == Qt::Checked))
{
compat::package_info info;
info.path = item->data(Roles::FullPathRole).toString();
info.title = item->data(Roles::TitleRole).toString();
info.title_id = item->data(Roles::TitleIdRole).toString();
info.changelog = item->data(Roles::ChangelogRole).toString();
info.version = item->data(Roles::VersionRole).toString();
result.push_back(info);
}
}
return result;
}