9a8667e99c
Signed-off-by: Trial97 <alexandru.tripon97@gmail.com>
342 lines
14 KiB
C++
342 lines
14 KiB
C++
// SPDX-License-Identifier: GPL-3.0-only
|
|
/*
|
|
* Prism Launcher - Minecraft Launcher
|
|
* Copyright (c) 2022 Jamie Mansfield <jmansfield@cadixdev.org>
|
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
|
* Copyright (C) 2022 TheKodeToad <TheKodeToad@proton.me>
|
|
* 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 "ModFolderPage.h"
|
|
#include "ui_ExternalResourcesPage.h"
|
|
|
|
#include <QAbstractItemModel>
|
|
#include <QEvent>
|
|
#include <QKeyEvent>
|
|
#include <QMenu>
|
|
#include <QMessageBox>
|
|
#include <QSortFilterProxyModel>
|
|
#include <algorithm>
|
|
|
|
#include "Application.h"
|
|
|
|
#include "ui/GuiUtil.h"
|
|
#include "ui/dialogs/CustomMessageBox.h"
|
|
#include "ui/dialogs/ModUpdateDialog.h"
|
|
#include "ui/dialogs/ResourceDownloadDialog.h"
|
|
|
|
#include "DesktopServices.h"
|
|
|
|
#include "minecraft/PackProfile.h"
|
|
#include "minecraft/VersionFilterData.h"
|
|
#include "minecraft/mod/Mod.h"
|
|
#include "minecraft/mod/ModFolderModel.h"
|
|
|
|
#include "modplatform/ModIndex.h"
|
|
#include "modplatform/ResourceAPI.h"
|
|
|
|
#include "Version.h"
|
|
#include "tasks/ConcurrentTask.h"
|
|
#include "ui/dialogs/ProgressDialog.h"
|
|
|
|
ModFolderPage::ModFolderPage(BaseInstance* inst, std::shared_ptr<ModFolderModel> mods, QWidget* parent)
|
|
: ExternalResourcesPage(inst, mods, parent), m_model(mods)
|
|
{
|
|
// This is structured like that so that these changes
|
|
// do not affect the Resource pack and Shader pack tabs
|
|
{
|
|
ui->actionDownloadItem->setText(tr("Download mods"));
|
|
ui->actionDownloadItem->setToolTip(tr("Download mods from online mod platforms"));
|
|
ui->actionDownloadItem->setEnabled(true);
|
|
ui->actionAddItem->setText(tr("Add file"));
|
|
ui->actionAddItem->setToolTip(tr("Add a locally downloaded file"));
|
|
|
|
ui->actionsToolbar->insertActionBefore(ui->actionAddItem, ui->actionDownloadItem);
|
|
|
|
connect(ui->actionDownloadItem, &QAction::triggered, this, &ModFolderPage::installMods);
|
|
|
|
ui->actionUpdateItem->setToolTip(tr("Try to check or update all selected mods (all mods if none are selected)"));
|
|
ui->actionsToolbar->insertActionAfter(ui->actionAddItem, ui->actionUpdateItem);
|
|
connect(ui->actionUpdateItem, &QAction::triggered, this, &ModFolderPage::updateMods);
|
|
|
|
ui->actionVisitItemPage->setToolTip(tr("Go to mod's home page"));
|
|
ui->actionsToolbar->addAction(ui->actionVisitItemPage);
|
|
connect(ui->actionVisitItemPage, &QAction::triggered, this, &ModFolderPage::visitModPages);
|
|
|
|
ui->actionRemoveItemMetadata->setToolTip(tr("Remove mod's metadata"));
|
|
ui->actionsToolbar->insertActionAfter(ui->actionRemoveItem, ui->actionRemoveItemMetadata);
|
|
connect(ui->actionRemoveItemMetadata, &QAction::triggered, this, &ModFolderPage::deleteModMetadata);
|
|
|
|
auto check_allow_update = [this] { return ui->treeView->selectionModel()->hasSelection() || !m_model->empty(); };
|
|
|
|
connect(ui->treeView->selectionModel(), &QItemSelectionModel::selectionChanged, this, [this, check_allow_update] {
|
|
ui->actionUpdateItem->setEnabled(check_allow_update());
|
|
|
|
auto selection = m_filterModel->mapSelectionToSource(ui->treeView->selectionModel()->selection()).indexes();
|
|
auto mods_list = m_model->selectedMods(selection);
|
|
auto selected = std::count_if(mods_list.cbegin(), mods_list.cend(),
|
|
[](Mod* v) { return v->metadata() != nullptr || v->homeurl().size() != 0; });
|
|
if (selected <= 1) {
|
|
ui->actionVisitItemPage->setText(tr("Visit mod's page"));
|
|
ui->actionVisitItemPage->setToolTip(tr("Go to mod's home page"));
|
|
|
|
ui->actionRemoveItemMetadata->setToolTip(tr("Remove mod's metadata"));
|
|
} else {
|
|
ui->actionVisitItemPage->setText(tr("Visit mods' pages"));
|
|
ui->actionVisitItemPage->setToolTip(tr("Go to the pages of the selected mods"));
|
|
|
|
ui->actionRemoveItemMetadata->setToolTip(tr("Remove mods' metadata"));
|
|
}
|
|
ui->actionVisitItemPage->setEnabled(selected != 0);
|
|
ui->actionRemoveItemMetadata->setEnabled(selected != 0);
|
|
});
|
|
|
|
connect(mods.get(), &ModFolderModel::rowsInserted, this,
|
|
[this, check_allow_update] { ui->actionUpdateItem->setEnabled(check_allow_update()); });
|
|
|
|
connect(mods.get(), &ModFolderModel::rowsRemoved, this,
|
|
[this, check_allow_update] { ui->actionUpdateItem->setEnabled(check_allow_update()); });
|
|
|
|
connect(mods.get(), &ModFolderModel::updateFinished, this,
|
|
[this, check_allow_update] { ui->actionUpdateItem->setEnabled(check_allow_update()); });
|
|
}
|
|
}
|
|
|
|
bool ModFolderPage::shouldDisplay() const
|
|
{
|
|
return true;
|
|
}
|
|
|
|
bool ModFolderPage::onSelectionChanged(const QModelIndex& current, [[maybe_unused]] const QModelIndex& previous)
|
|
{
|
|
auto sourceCurrent = m_filterModel->mapToSource(current);
|
|
int row = sourceCurrent.row();
|
|
Mod const* m = m_model->at(row);
|
|
if (m)
|
|
ui->frame->updateWithMod(*m);
|
|
|
|
return true;
|
|
}
|
|
|
|
void ModFolderPage::removeItems(const QItemSelection& selection)
|
|
{
|
|
if (m_instance != nullptr && m_instance->isRunning()) {
|
|
auto response = CustomMessageBox::selectable(this, "Confirm Delete",
|
|
"If you remove mods 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->deleteMods(selection.indexes());
|
|
}
|
|
|
|
void ModFolderPage::installMods()
|
|
{
|
|
if (m_instance->typeName() != "Minecraft")
|
|
return; // this is a null instance or a legacy instance
|
|
|
|
auto profile = static_cast<MinecraftInstance*>(m_instance)->getPackProfile();
|
|
if (!profile->getModLoaders().has_value()) {
|
|
QMessageBox::critical(this, tr("Error"), tr("Please install a mod loader first!"));
|
|
return;
|
|
}
|
|
|
|
ResourceDownload::ModDownloadDialog mdownload(this, m_model, m_instance);
|
|
if (mdownload.exec()) {
|
|
auto tasks = new ConcurrentTask(this, "Download Mods", APPLICATION->settings()->get("NumberOfConcurrentDownloads").toInt());
|
|
connect(tasks, &Task::failed, [this, tasks](QString reason) {
|
|
CustomMessageBox::selectable(this, tr("Error"), reason, QMessageBox::Critical)->show();
|
|
tasks->deleteLater();
|
|
});
|
|
connect(tasks, &Task::aborted, [this, tasks]() {
|
|
CustomMessageBox::selectable(this, tr("Aborted"), tr("Download stopped by user."), QMessageBox::Information)->show();
|
|
tasks->deleteLater();
|
|
});
|
|
connect(tasks, &Task::succeeded, [this, tasks]() {
|
|
QStringList warnings = tasks->warnings();
|
|
if (warnings.count())
|
|
CustomMessageBox::selectable(this, tr("Warnings"), warnings.join('\n'), QMessageBox::Warning)->show();
|
|
|
|
tasks->deleteLater();
|
|
});
|
|
|
|
for (auto& task : mdownload.getTasks()) {
|
|
tasks->addTask(task);
|
|
}
|
|
|
|
ProgressDialog loadDialog(this);
|
|
loadDialog.setSkipButton(true, tr("Abort"));
|
|
loadDialog.execWithTask(tasks);
|
|
|
|
m_model->update();
|
|
}
|
|
}
|
|
|
|
void ModFolderPage::updateMods()
|
|
{
|
|
if (m_instance->typeName() != "Minecraft")
|
|
return; // this is a null instance or a legacy instance
|
|
|
|
auto profile = static_cast<MinecraftInstance*>(m_instance)->getPackProfile();
|
|
if (!profile->getModLoaders().has_value()) {
|
|
QMessageBox::critical(this, tr("Error"), tr("Please install a mod loader first!"));
|
|
return;
|
|
}
|
|
if (APPLICATION->settings()->get("ModMetadataDisabled").toBool()) {
|
|
QMessageBox::critical(this, tr("Error"), tr("The mod update is disabled when the metadata is disabled!"));
|
|
return;
|
|
}
|
|
auto selection = m_filterModel->mapSelectionToSource(ui->treeView->selectionModel()->selection()).indexes();
|
|
|
|
auto mods_list = m_model->selectedMods(selection);
|
|
bool use_all = mods_list.empty();
|
|
if (use_all)
|
|
mods_list = m_model->allMods();
|
|
|
|
ModUpdateDialog update_dialog(this, m_instance, m_model, mods_list);
|
|
update_dialog.checkCandidates();
|
|
|
|
if (update_dialog.aborted()) {
|
|
CustomMessageBox::selectable(this, tr("Aborted"), tr("The mod updater was aborted!"), QMessageBox::Warning)->show();
|
|
return;
|
|
}
|
|
if (update_dialog.noUpdates()) {
|
|
QString message{ tr("'%1' is up-to-date! :)").arg(mods_list.front()->name()) };
|
|
if (mods_list.size() > 1) {
|
|
if (use_all) {
|
|
message = tr("All mods are up-to-date! :)");
|
|
} else {
|
|
message = tr("All selected mods are up-to-date! :)");
|
|
}
|
|
}
|
|
CustomMessageBox::selectable(this, tr("Update checker"), message)->exec();
|
|
return;
|
|
}
|
|
|
|
if (update_dialog.exec()) {
|
|
auto tasks = new ConcurrentTask(this, "Download Mods", APPLICATION->settings()->get("NumberOfConcurrentDownloads").toInt());
|
|
connect(tasks, &Task::failed, [this, tasks](QString reason) {
|
|
CustomMessageBox::selectable(this, tr("Error"), reason, QMessageBox::Critical)->show();
|
|
tasks->deleteLater();
|
|
});
|
|
connect(tasks, &Task::aborted, [this, tasks]() {
|
|
CustomMessageBox::selectable(this, tr("Aborted"), tr("Download stopped by user."), QMessageBox::Information)->show();
|
|
tasks->deleteLater();
|
|
});
|
|
connect(tasks, &Task::succeeded, [this, tasks]() {
|
|
QStringList warnings = tasks->warnings();
|
|
if (warnings.count()) {
|
|
CustomMessageBox::selectable(this, tr("Warnings"), warnings.join('\n'), QMessageBox::Warning)->show();
|
|
}
|
|
tasks->deleteLater();
|
|
});
|
|
|
|
for (auto task : update_dialog.getTasks()) {
|
|
tasks->addTask(task);
|
|
}
|
|
|
|
ProgressDialog loadDialog(this);
|
|
loadDialog.setSkipButton(true, tr("Abort"));
|
|
loadDialog.execWithTask(tasks);
|
|
|
|
m_model->update();
|
|
}
|
|
}
|
|
|
|
CoreModFolderPage::CoreModFolderPage(BaseInstance* inst, std::shared_ptr<ModFolderModel> mods, QWidget* parent)
|
|
: ModFolderPage(inst, mods, parent)
|
|
{}
|
|
|
|
bool CoreModFolderPage::shouldDisplay() const
|
|
{
|
|
if (ModFolderPage::shouldDisplay()) {
|
|
auto inst = dynamic_cast<MinecraftInstance*>(m_instance);
|
|
if (!inst)
|
|
return true;
|
|
|
|
auto version = inst->getPackProfile();
|
|
|
|
if (!version)
|
|
return true;
|
|
if (!version->getComponent("net.minecraftforge"))
|
|
return false;
|
|
if (!version->getComponent("net.minecraft"))
|
|
return false;
|
|
if (version->getComponent("net.minecraft")->getReleaseDateTime() < g_VersionFilterData.legacyCutoffDate)
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
NilModFolderPage::NilModFolderPage(BaseInstance* inst, std::shared_ptr<ModFolderModel> mods, QWidget* parent)
|
|
: ModFolderPage(inst, mods, parent)
|
|
{}
|
|
|
|
bool NilModFolderPage::shouldDisplay() const
|
|
{
|
|
return m_model->dir().exists();
|
|
}
|
|
|
|
void ModFolderPage::visitModPages()
|
|
{
|
|
auto selection = m_filterModel->mapSelectionToSource(ui->treeView->selectionModel()->selection()).indexes();
|
|
for (auto mod : m_model->selectedMods(selection)) {
|
|
auto url = mod->metaurl();
|
|
if (!url.isEmpty())
|
|
DesktopServices::openUrl(url);
|
|
}
|
|
}
|
|
|
|
void ModFolderPage::deleteModMetadata()
|
|
{
|
|
auto selection = m_filterModel->mapSelectionToSource(ui->treeView->selectionModel()->selection()).indexes();
|
|
auto selectionCount = m_model->selectedMods(selection).length();
|
|
if (selectionCount == 0)
|
|
return;
|
|
if (selectionCount > 1) {
|
|
auto response = CustomMessageBox::selectable(this, tr("Confirm Removal"),
|
|
tr("You are about to remove the metadata for %1 mods.\n"
|
|
"Are you sure?")
|
|
.arg(selectionCount),
|
|
QMessageBox::Warning, QMessageBox::Yes | QMessageBox::No, QMessageBox::No)
|
|
->exec();
|
|
|
|
if (response != QMessageBox::Yes)
|
|
return;
|
|
}
|
|
|
|
m_model->deleteModsMetadata(selection);
|
|
}
|