mirror of
https://github.com/PrismLauncher/PrismLauncher.git
synced 2026-06-29 01:54:20 +03:00
423 lines
No EOL
16 KiB
C++
423 lines
No EOL
16 KiB
C++
// SPDX-License-Identifier: GPL-3.0-only
|
|
/*
|
|
* Prism Launcher - Minecraft Launcher
|
|
* Copyright (c) 2023 Trial97 <alexandru.tripon97@gmail.com>
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, version 3.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*
|
|
* This file incorporates work covered by the following copyright and
|
|
* permission notice:
|
|
*
|
|
* Copyright 2013-2021 MultiMC Contributors
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#include "ExternalResourcesPage.h"
|
|
#include "ui/dialogs/CustomMessageBox.h"
|
|
#include "ui_ExternalResourcesPage.h"
|
|
|
|
#include "DesktopServices.h"
|
|
#include "minecraft/mod/ResourceFolderModel.h"
|
|
#include "ui/GuiUtil.h"
|
|
|
|
#include <QHeaderView>
|
|
#include <QKeyEvent>
|
|
#include <QMenu>
|
|
#include <QStyledItemDelegate>
|
|
#include <algorithm>
|
|
|
|
namespace {
|
|
class LockDelegate : public QStyledItemDelegate {
|
|
public:
|
|
explicit LockDelegate(QObject* parent = nullptr) : QStyledItemDelegate(parent) {}
|
|
|
|
void paint(QPainter* painter, const QStyleOptionViewItem& opt, const QModelIndex& index) const override
|
|
{
|
|
QStyleOptionViewItem option(opt);
|
|
initStyleOption(&option, index);
|
|
|
|
bool locked = index.data(Qt::UserRole).toBool();
|
|
|
|
const QIcon& icon = QIcon::fromTheme(locked ? "lock" : "unlock");
|
|
|
|
// Draw default background / selection
|
|
option.text.clear();
|
|
option.icon = QIcon();
|
|
|
|
option.widget->style()->drawControl(QStyle::CE_ItemViewItem, &option, painter);
|
|
|
|
int size = qMin(option.rect.width(), option.rect.height()) * 3 / 4;
|
|
QRect iconRect(option.rect.center().x() - (size / 2), option.rect.center().y() - (size / 2), size, size);
|
|
|
|
icon.paint(painter, iconRect);
|
|
}
|
|
|
|
bool editorEvent(QEvent* event, QAbstractItemModel* model, const QStyleOptionViewItem& /*option*/, const QModelIndex& index) override
|
|
{
|
|
if (event->type() == QEvent::MouseButtonRelease) {
|
|
bool locked = index.data(Qt::UserRole).toBool();
|
|
model->setData(index, !locked, Qt::UserRole);
|
|
return true;
|
|
}
|
|
return event->type() == QEvent::MouseButtonDblClick; // if double click ignore it
|
|
}
|
|
};
|
|
} // namespace
|
|
|
|
ExternalResourcesPage::ExternalResourcesPage(BaseInstance* instance, ResourceFolderModel* model, QWidget* parent)
|
|
: QMainWindow(parent)
|
|
, m_instance(instance)
|
|
, m_ui(new Ui::ExternalResourcesPage)
|
|
, m_model(model)
|
|
, m_filterModel(ResourceFolderModel::createFilterProxyModel(this))
|
|
{
|
|
m_ui->setupUi(this);
|
|
|
|
m_ui->actionsToolbar->insertSpacer(m_ui->actionViewFolder);
|
|
|
|
m_filterModel->setDynamicSortFilter(true);
|
|
m_filterModel->setFilterCaseSensitivity(Qt::CaseInsensitive);
|
|
m_filterModel->setSortCaseSensitivity(Qt::CaseInsensitive);
|
|
m_filterModel->setSourceModel(m_model);
|
|
m_filterModel->setFilterKeyColumn(-1);
|
|
m_ui->treeView->setModel(m_filterModel);
|
|
|
|
// keep the Update at the end of the list(otherwise there will be a need to iterate over the columns)
|
|
auto lockColumn = static_cast<int>(model->columnNames(false).size()) - 1;
|
|
m_ui->treeView->setItemDelegateForColumn(lockColumn, new LockDelegate(m_ui->treeView));
|
|
// must come after setModel
|
|
m_ui->treeView->setResizeModes(m_model->columnResizeModes());
|
|
|
|
m_ui->treeView->installEventFilter(this);
|
|
m_ui->treeView->sortByColumn(1, Qt::AscendingOrder);
|
|
m_ui->treeView->setContextMenuPolicy(Qt::CustomContextMenu);
|
|
|
|
// The default function names by Qt are pretty ugly, so let's just connect the actions manually,
|
|
// to make it easier to read :)
|
|
connect(m_ui->actionAddItem, &QAction::triggered, this, &ExternalResourcesPage::addItem);
|
|
connect(m_ui->actionRemoveItem, &QAction::triggered, this, &ExternalResourcesPage::removeItem);
|
|
connect(m_ui->actionEnableItem, &QAction::triggered, this, &ExternalResourcesPage::enableItem);
|
|
connect(m_ui->actionDisableItem, &QAction::triggered, this, &ExternalResourcesPage::disableItem);
|
|
connect(m_ui->actionViewHomepage, &QAction::triggered, this, &ExternalResourcesPage::viewHomepage);
|
|
connect(m_ui->actionViewConfigs, &QAction::triggered, this, &ExternalResourcesPage::viewConfigs);
|
|
connect(m_ui->actionViewFolder, &QAction::triggered, this, &ExternalResourcesPage::viewFolder);
|
|
|
|
connect(m_ui->treeView, &ModListView::customContextMenuRequested, this, &ExternalResourcesPage::showContextMenu);
|
|
connect(m_ui->treeView, &ModListView::activated, this, &ExternalResourcesPage::itemActivated);
|
|
|
|
connect(m_ui->actionEnableUpdates, &QAction::triggered, this, &ExternalResourcesPage::enableUpdates);
|
|
connect(m_ui->actionDisableUpdates, &QAction::triggered, this, &ExternalResourcesPage::disableUpdates);
|
|
|
|
auto* selectionModel = m_ui->treeView->selectionModel();
|
|
|
|
connect(selectionModel, &QItemSelectionModel::currentChanged, this, [this](const QModelIndex& current, const QModelIndex& previous) {
|
|
if (!current.isValid()) {
|
|
m_ui->frame->clear();
|
|
return;
|
|
}
|
|
|
|
updateFrame(current, previous);
|
|
});
|
|
|
|
auto updateExtra = [this]() {
|
|
if (updateExtraInfo) {
|
|
updateExtraInfo(id(), extraHeaderInfoString());
|
|
}
|
|
};
|
|
|
|
connect(selectionModel, &QItemSelectionModel::selectionChanged, this, updateExtra);
|
|
connect(model, &ResourceFolderModel::updateFinished, this, updateExtra);
|
|
connect(model, &ResourceFolderModel::parseFinished, this, updateExtra);
|
|
|
|
connect(selectionModel, &QItemSelectionModel::selectionChanged, this, [this] { updateActions(); });
|
|
connect(m_model, &ResourceFolderModel::rowsInserted, this, [this] { updateActions(); });
|
|
connect(m_model, &ResourceFolderModel::rowsRemoved, this, [this] { updateActions(); });
|
|
|
|
auto* viewHeader = m_ui->treeView->header();
|
|
viewHeader->setContextMenuPolicy(Qt::CustomContextMenu);
|
|
|
|
connect(viewHeader, &QHeaderView::customContextMenuRequested, this, &ExternalResourcesPage::showHeaderContextMenu);
|
|
|
|
m_model->loadColumns(m_ui->treeView);
|
|
connect(m_ui->treeView->header(), &QHeaderView::sectionResized, this, [this] { m_model->saveColumns(m_ui->treeView); });
|
|
connect(m_ui->filterEdit, &QLineEdit::textChanged, this, &ExternalResourcesPage::filterTextChanged);
|
|
updateActions();
|
|
}
|
|
|
|
ExternalResourcesPage::~ExternalResourcesPage()
|
|
{
|
|
delete m_ui;
|
|
}
|
|
|
|
QMenu* ExternalResourcesPage::createPopupMenu()
|
|
{
|
|
QMenu* filteredMenu = QMainWindow::createPopupMenu();
|
|
filteredMenu->removeAction(m_ui->actionsToolbar->toggleViewAction());
|
|
return filteredMenu;
|
|
}
|
|
|
|
void ExternalResourcesPage::showContextMenu(const QPoint& pos)
|
|
{
|
|
auto* menu = m_ui->actionsToolbar->createContextMenu(this, tr("Context menu"));
|
|
menu->exec(m_ui->treeView->mapToGlobal(pos));
|
|
delete menu;
|
|
}
|
|
|
|
void ExternalResourcesPage::showHeaderContextMenu(const QPoint& pos)
|
|
{
|
|
auto* menu = m_model->createHeaderContextMenu(m_ui->treeView);
|
|
menu->exec(m_ui->treeView->mapToGlobal(pos));
|
|
menu->deleteLater();
|
|
}
|
|
|
|
void ExternalResourcesPage::openedImpl()
|
|
{
|
|
m_model->startWatching();
|
|
|
|
const auto settingName = QString("WideBarVisibility_%1").arg(id());
|
|
m_wideBarSetting = APPLICATION->settings()->getOrRegisterSetting(settingName);
|
|
|
|
m_ui->actionsToolbar->setVisibilityState(QByteArray::fromBase64(m_wideBarSetting->get().toString().toUtf8()));
|
|
}
|
|
|
|
void ExternalResourcesPage::closedImpl()
|
|
{
|
|
m_model->stopWatching();
|
|
|
|
m_wideBarSetting->set(QString::fromUtf8(m_ui->actionsToolbar->getVisibilityState().toBase64()));
|
|
}
|
|
|
|
void ExternalResourcesPage::retranslate()
|
|
{
|
|
m_ui->retranslateUi(this);
|
|
}
|
|
|
|
void ExternalResourcesPage::itemActivated(const QModelIndex& /*unused*/)
|
|
{
|
|
auto selection = m_filterModel->mapSelectionToSource(m_ui->treeView->selectionModel()->selection());
|
|
m_model->setResourceEnabled(selection.indexes(), EnableAction::TOGGLE);
|
|
}
|
|
|
|
void ExternalResourcesPage::filterTextChanged(const QString& newContents)
|
|
{
|
|
m_viewFilter = newContents;
|
|
m_filterModel->setFilterRegularExpression(m_viewFilter);
|
|
}
|
|
|
|
bool ExternalResourcesPage::shouldDisplay() const
|
|
{
|
|
return true;
|
|
}
|
|
|
|
bool ExternalResourcesPage::listFilter(QKeyEvent* keyEvent)
|
|
{
|
|
switch (keyEvent->key()) {
|
|
case Qt::Key_Delete:
|
|
removeItem();
|
|
return true;
|
|
case Qt::Key_Plus:
|
|
addItem();
|
|
return true;
|
|
default:
|
|
break;
|
|
}
|
|
return QWidget::eventFilter(m_ui->treeView, keyEvent);
|
|
}
|
|
|
|
bool ExternalResourcesPage::eventFilter(QObject* obj, QEvent* ev)
|
|
{
|
|
if (ev->type() != QEvent::KeyPress) {
|
|
return QWidget::eventFilter(obj, ev);
|
|
}
|
|
|
|
auto* keyEvent = static_cast<QKeyEvent*>(ev);
|
|
if (obj == m_ui->treeView) {
|
|
return listFilter(keyEvent);
|
|
}
|
|
|
|
return QWidget::eventFilter(obj, ev);
|
|
}
|
|
|
|
void ExternalResourcesPage::addItem()
|
|
{
|
|
auto list = GuiUtil::BrowseForFiles(
|
|
helpPage(), tr("Select %1", "Select whatever type of files the page contains. Example: 'Loader Mods'").arg(displayName()),
|
|
m_fileSelectionFilter.arg(displayName()), APPLICATION->settings()->get("CentralModsDir").toString(), this->parentWidget());
|
|
|
|
if (!list.isEmpty()) {
|
|
for (const auto& filename : list) {
|
|
m_model->installResource(filename);
|
|
}
|
|
}
|
|
}
|
|
|
|
void ExternalResourcesPage::removeItem()
|
|
{
|
|
auto selection = m_filterModel->mapSelectionToSource(m_ui->treeView->selectionModel()->selection());
|
|
|
|
int count = 0;
|
|
bool folder = false;
|
|
for (auto& i : selection.indexes()) {
|
|
if (i.column() == 0) {
|
|
count++;
|
|
|
|
// if a folder is selected, show the confirmation dialog
|
|
if (m_model->at(i.row()).fileinfo().isDir()) {
|
|
folder = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
QString text;
|
|
bool multiple = count > 1;
|
|
|
|
if (multiple) {
|
|
text = tr("You are about to remove %1 items.\n"
|
|
"This may be permanent and they will be gone from the folder.\n\n"
|
|
"Are you sure?")
|
|
.arg(count);
|
|
} else if (folder) {
|
|
text = tr("You are about to remove the folder \"%1\".\n"
|
|
"This may be permanent and it will be gone from the parent folder.\n\n"
|
|
"Are you sure?")
|
|
.arg(m_model->at(selection.indexes().at(0).row()).fileinfo().fileName());
|
|
}
|
|
|
|
if (!text.isEmpty()) {
|
|
auto response = CustomMessageBox::selectable(this, tr("Confirm Removal"), text, QMessageBox::Warning,
|
|
QMessageBox::Yes | QMessageBox::No, QMessageBox::No)
|
|
->exec();
|
|
|
|
if (response != QMessageBox::Yes) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
removeItems(selection);
|
|
}
|
|
|
|
void ExternalResourcesPage::removeItems(const QItemSelection& selection)
|
|
{
|
|
if (m_instance != nullptr && m_instance->isRunning()) {
|
|
auto response = CustomMessageBox::selectable(this, tr("Confirm Delete"),
|
|
tr("If you remove this resource while the game is running it may crash your game.\n"
|
|
"Are you sure you want to do this?"),
|
|
QMessageBox::Warning, QMessageBox::Yes | QMessageBox::No, QMessageBox::No)
|
|
->exec();
|
|
|
|
if (response != QMessageBox::Yes) {
|
|
return;
|
|
}
|
|
}
|
|
m_model->deleteResources(selection.indexes());
|
|
}
|
|
|
|
void ExternalResourcesPage::enableItem()
|
|
{
|
|
auto selection = m_filterModel->mapSelectionToSource(m_ui->treeView->selectionModel()->selection());
|
|
m_model->setResourceEnabled(selection.indexes(), EnableAction::ENABLE);
|
|
}
|
|
|
|
void ExternalResourcesPage::disableItem()
|
|
{
|
|
auto selection = m_filterModel->mapSelectionToSource(m_ui->treeView->selectionModel()->selection());
|
|
m_model->setResourceEnabled(selection.indexes(), EnableAction::DISABLE);
|
|
}
|
|
|
|
void ExternalResourcesPage::viewHomepage()
|
|
{
|
|
auto selection = m_filterModel->mapSelectionToSource(m_ui->treeView->selectionModel()->selection()).indexes();
|
|
for (auto* resource : m_model->selectedResources(selection)) {
|
|
auto url = resource->homepage();
|
|
if (!url.isEmpty()) {
|
|
DesktopServices::openUrl(url);
|
|
}
|
|
}
|
|
}
|
|
|
|
void ExternalResourcesPage::viewConfigs()
|
|
{
|
|
DesktopServices::openPath(m_instance->instanceConfigFolder(), true);
|
|
}
|
|
|
|
void ExternalResourcesPage::viewFolder()
|
|
{
|
|
DesktopServices::openPath(m_model->dir().absolutePath(), true);
|
|
}
|
|
|
|
void ExternalResourcesPage::updateActions()
|
|
{
|
|
const bool hasSelection = m_ui->treeView->selectionModel()->hasSelection();
|
|
const QModelIndexList selection = m_filterModel->mapSelectionToSource(m_ui->treeView->selectionModel()->selection()).indexes();
|
|
const QList<Resource*> selectedResources = m_model->selectedResources(selection);
|
|
const bool hasMeta = hasSelection && std::any_of(selectedResources.begin(), selectedResources.end(),
|
|
[](Resource* resource) { return resource->metadata(); });
|
|
|
|
m_ui->actionUpdateItem->setEnabled(!m_model->empty());
|
|
m_ui->actionResetItemMetadata->setEnabled(hasSelection);
|
|
|
|
m_ui->actionChangeVersion->setEnabled(selectedResources.size() == 1 && selectedResources[0]->metadata() != nullptr);
|
|
|
|
m_ui->actionRemoveItem->setEnabled(hasSelection);
|
|
m_ui->actionEnableItem->setEnabled(hasSelection);
|
|
m_ui->actionDisableItem->setEnabled(hasSelection);
|
|
|
|
m_ui->actionViewHomepage->setEnabled(hasSelection && std::any_of(selectedResources.begin(), selectedResources.end(),
|
|
[](Resource* resource) { return !resource->homepage().isEmpty(); }));
|
|
|
|
m_ui->actionEnableUpdates->setEnabled(hasMeta);
|
|
m_ui->actionDisableUpdates->setEnabled(hasMeta);
|
|
m_ui->actionExportMetadata->setEnabled(!m_model->empty());
|
|
}
|
|
|
|
void ExternalResourcesPage::updateFrame(const QModelIndex& current, [[maybe_unused]] const QModelIndex& previous)
|
|
{
|
|
auto sourceCurrent = m_filterModel->mapToSource(current);
|
|
int row = sourceCurrent.row();
|
|
const Resource& resource = m_model->at(row);
|
|
m_ui->frame->updateWithResource(resource);
|
|
}
|
|
|
|
QString ExternalResourcesPage::extraHeaderInfoString()
|
|
{
|
|
if (m_ui && m_ui->treeView && m_ui->treeView->selectionModel()) {
|
|
auto selection = m_filterModel->mapSelectionToSource(m_ui->treeView->selectionModel()->selection()).indexes();
|
|
if (auto count = std::count_if(selection.cbegin(), selection.cend(), [](auto v) { return v.column() == 0; }); count != 0) {
|
|
return tr(" (%1 installed, %2 selected)").arg(m_model->size()).arg(count);
|
|
}
|
|
}
|
|
return tr(" (%1 installed)").arg(m_model->size());
|
|
}
|
|
|
|
void ExternalResourcesPage::enableUpdates()
|
|
{
|
|
auto selection = m_filterModel->mapSelectionToSource(m_ui->treeView->selectionModel()->selection());
|
|
m_model->setModUpdate(selection.indexes(), EnableAction::ENABLE);
|
|
}
|
|
|
|
void ExternalResourcesPage::disableUpdates()
|
|
{
|
|
auto selection = m_filterModel->mapSelectionToSource(m_ui->treeView->selectionModel()->selection());
|
|
m_model->setModUpdate(selection.indexes(), EnableAction::DISABLE);
|
|
} |