2022-03-20 20:01:08 +01:00
|
|
|
// SPDX-License-Identifier: GPL-3.0-only
|
|
|
|
/*
|
|
|
|
* PolyMC - Minecraft Launcher
|
|
|
|
* Copyright (c) 2022 Jamie Mansfield <jmansfield@cadixdev.org>
|
2014-06-02 00:49:53 +02:00
|
|
|
*
|
2022-03-20 20:01:08 +01:00
|
|
|
* 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.
|
2014-06-02 00:49:53 +02:00
|
|
|
*
|
2022-03-20 20:01:08 +01:00
|
|
|
* 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.
|
2014-06-02 00:49:53 +02:00
|
|
|
*
|
2022-03-20 20:01:08 +01:00
|
|
|
* 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.
|
2014-06-02 00:49:53 +02:00
|
|
|
*/
|
|
|
|
|
2014-07-12 23:02:52 +02:00
|
|
|
#include "ModFolderPage.h"
|
|
|
|
#include "ui_ModFolderPage.h"
|
2014-06-02 00:49:53 +02:00
|
|
|
|
|
|
|
#include <QMessageBox>
|
|
|
|
#include <QEvent>
|
|
|
|
#include <QKeyEvent>
|
|
|
|
#include <QAbstractItemModel>
|
2019-07-16 01:30:53 +02:00
|
|
|
#include <QMenu>
|
2021-11-22 03:55:16 +01:00
|
|
|
#include <QSortFilterProxyModel>
|
2014-06-02 00:49:53 +02:00
|
|
|
|
2021-11-20 16:22:22 +01:00
|
|
|
#include "Application.h"
|
2021-11-22 03:55:16 +01:00
|
|
|
|
|
|
|
#include "ui/dialogs/CustomMessageBox.h"
|
2022-01-14 09:43:42 +01:00
|
|
|
#include "ui/dialogs/ModDownloadDialog.h"
|
2021-11-22 03:55:16 +01:00
|
|
|
#include "ui/GuiUtil.h"
|
|
|
|
|
|
|
|
#include "DesktopServices.h"
|
|
|
|
|
2019-08-04 03:27:53 +02:00
|
|
|
#include "minecraft/mod/ModFolderModel.h"
|
|
|
|
#include "minecraft/mod/Mod.h"
|
2015-02-09 01:51:14 +01:00
|
|
|
#include "minecraft/VersionFilterData.h"
|
2020-06-27 12:02:31 +02:00
|
|
|
#include "minecraft/PackProfile.h"
|
2014-06-02 00:49:53 +02:00
|
|
|
|
2022-02-20 20:55:26 +01:00
|
|
|
#include "modplatform/ModAPI.h"
|
|
|
|
|
2019-07-30 01:25:37 +02:00
|
|
|
#include "Version.h"
|
2022-01-14 09:43:42 +01:00
|
|
|
#include "ui/dialogs/ProgressDialog.h"
|
2022-04-01 09:10:51 -03:00
|
|
|
#include "tasks/SequentialTask.h"
|
2019-07-30 01:16:56 +02:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
// FIXME: wasteful
|
|
|
|
void RemoveThePrefix(QString & string) {
|
|
|
|
QRegularExpression regex(QStringLiteral("^(([Tt][Hh][eE])|([Tt][eE][Hh])) +"));
|
|
|
|
string.remove(regex);
|
|
|
|
string = string.trimmed();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class ModSortProxy : public QSortFilterProxyModel
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
explicit ModSortProxy(QObject *parent = 0) : QSortFilterProxyModel(parent)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
2020-01-08 21:12:45 +01:00
|
|
|
bool filterAcceptsRow(int source_row, const QModelIndex & source_parent) const override {
|
|
|
|
ModFolderModel *model = qobject_cast<ModFolderModel *>(sourceModel());
|
|
|
|
if(!model) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
const auto &mod = model->at(source_row);
|
|
|
|
if(mod.name().contains(filterRegExp())) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if(mod.description().contains(filterRegExp())) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
for(auto & author: mod.authors()) {
|
|
|
|
if (author.contains(filterRegExp())) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-07-30 01:16:56 +02:00
|
|
|
bool lessThan(const QModelIndex & source_left, const QModelIndex & source_right) const override
|
|
|
|
{
|
2019-08-04 03:27:53 +02:00
|
|
|
ModFolderModel *model = qobject_cast<ModFolderModel *>(sourceModel());
|
2019-07-30 01:16:56 +02:00
|
|
|
if(
|
|
|
|
!model ||
|
|
|
|
!source_left.isValid() ||
|
|
|
|
!source_right.isValid() ||
|
|
|
|
source_left.column() != source_right.column()
|
|
|
|
) {
|
|
|
|
return QSortFilterProxyModel::lessThan(source_left, source_right);
|
|
|
|
}
|
|
|
|
|
|
|
|
// we are now guaranteed to have two valid indexes in the same column... we love the provided invariants unconditionally and proceed.
|
|
|
|
|
2019-08-04 03:27:53 +02:00
|
|
|
auto column = (ModFolderModel::Columns) source_left.column();
|
2019-07-31 01:28:55 +02:00
|
|
|
bool invert = false;
|
2019-07-30 01:16:56 +02:00
|
|
|
switch(column) {
|
2019-07-31 01:28:55 +02:00
|
|
|
// GH-2550 - sort by enabled/disabled
|
2019-08-04 03:27:53 +02:00
|
|
|
case ModFolderModel::ActiveColumn: {
|
2019-07-31 01:28:55 +02:00
|
|
|
auto dataL = source_left.data(Qt::CheckStateRole).toBool();
|
|
|
|
auto dataR = source_right.data(Qt::CheckStateRole).toBool();
|
|
|
|
if(dataL != dataR) {
|
|
|
|
return dataL > dataR;
|
|
|
|
}
|
|
|
|
// fallthrough
|
|
|
|
invert = sortOrder() == Qt::DescendingOrder;
|
|
|
|
}
|
2019-07-30 01:16:56 +02:00
|
|
|
// GH-2722 - sort mod names in a way that discards "The" prefixes
|
2019-08-04 03:27:53 +02:00
|
|
|
case ModFolderModel::NameColumn: {
|
|
|
|
auto dataL = model->data(model->index(source_left.row(), ModFolderModel::NameColumn)).toString();
|
2019-07-30 01:16:56 +02:00
|
|
|
RemoveThePrefix(dataL);
|
2019-08-04 03:27:53 +02:00
|
|
|
auto dataR = model->data(model->index(source_right.row(), ModFolderModel::NameColumn)).toString();
|
2019-07-30 01:16:56 +02:00
|
|
|
RemoveThePrefix(dataR);
|
|
|
|
|
2019-07-31 01:28:55 +02:00
|
|
|
auto less = dataL.compare(dataR, sortCaseSensitivity());
|
|
|
|
if(less != 0) {
|
|
|
|
return invert ? (less > 0) : (less < 0);
|
|
|
|
}
|
|
|
|
// fallthrough
|
|
|
|
invert = sortOrder() == Qt::DescendingOrder;
|
2019-07-30 01:16:56 +02:00
|
|
|
}
|
|
|
|
// GH-2762 - sort versions by parsing them as versions
|
2019-08-04 03:27:53 +02:00
|
|
|
case ModFolderModel::VersionColumn: {
|
|
|
|
auto dataL = Version(model->data(model->index(source_left.row(), ModFolderModel::VersionColumn)).toString());
|
|
|
|
auto dataR = Version(model->data(model->index(source_right.row(), ModFolderModel::VersionColumn)).toString());
|
2019-07-31 01:28:55 +02:00
|
|
|
return invert ? (dataL > dataR) : (dataL < dataR);
|
2019-07-30 01:16:56 +02:00
|
|
|
}
|
|
|
|
default: {
|
|
|
|
return QSortFilterProxyModel::lessThan(source_left, source_right);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-07-25 01:02:30 +02:00
|
|
|
ModFolderPage::ModFolderPage(
|
|
|
|
BaseInstance *inst,
|
2019-08-04 03:27:53 +02:00
|
|
|
std::shared_ptr<ModFolderModel> mods,
|
2019-07-25 01:02:30 +02:00
|
|
|
QString id,
|
|
|
|
QString iconName,
|
|
|
|
QString displayName,
|
|
|
|
QString helpPage,
|
|
|
|
QWidget *parent
|
|
|
|
) :
|
|
|
|
QMainWindow(parent),
|
|
|
|
ui(new Ui::ModFolderPage)
|
2018-07-15 14:51:05 +02:00
|
|
|
{
|
|
|
|
ui->setupUi(this);
|
2022-03-03 09:51:46 -03:00
|
|
|
|
|
|
|
// This is structured like that so that these changes
|
|
|
|
// do not affect the Resouce pack and Shader pack tabs
|
2022-01-28 19:32:42 +01:00
|
|
|
if(id == "mods") {
|
2022-03-03 09:51:46 -03:00
|
|
|
auto act = new QAction(tr("Download mods"), this);
|
|
|
|
act->setToolTip(tr("Download mods from online mod platforms"));
|
|
|
|
ui->actionsToolbar->insertActionBefore(ui->actionAdd, act);
|
2022-01-28 19:32:42 +01:00
|
|
|
connect(act, &QAction::triggered, this, &ModFolderPage::on_actionInstall_mods_triggered);
|
2022-03-03 09:51:46 -03:00
|
|
|
|
2022-03-11 18:43:17 -03:00
|
|
|
ui->actionAdd->setText(tr("Add .jar"));
|
|
|
|
ui->actionAdd->setToolTip(tr("Add mods via local file"));
|
2022-03-03 04:02:22 -03:00
|
|
|
}
|
2022-03-03 09:51:46 -03:00
|
|
|
|
2019-07-23 00:48:14 +02:00
|
|
|
ui->actionsToolbar->insertSpacer(ui->actionView_configs);
|
2019-07-17 02:01:29 +02:00
|
|
|
|
2018-07-15 14:51:05 +02:00
|
|
|
m_inst = inst;
|
2019-07-15 01:07:21 +02:00
|
|
|
on_RunningState_changed(m_inst && m_inst->isRunning());
|
2018-07-15 14:51:05 +02:00
|
|
|
m_mods = mods;
|
|
|
|
m_id = id;
|
|
|
|
m_displayName = displayName;
|
|
|
|
m_iconName = iconName;
|
|
|
|
m_helpName = helpPage;
|
|
|
|
m_fileSelectionFilter = "%1 (*.zip *.jar)";
|
2019-07-30 01:16:56 +02:00
|
|
|
m_filterModel = new ModSortProxy(this);
|
2018-07-15 14:51:05 +02:00
|
|
|
m_filterModel->setDynamicSortFilter(true);
|
|
|
|
m_filterModel->setFilterCaseSensitivity(Qt::CaseInsensitive);
|
|
|
|
m_filterModel->setSortCaseSensitivity(Qt::CaseInsensitive);
|
|
|
|
m_filterModel->setSourceModel(m_mods.get());
|
|
|
|
m_filterModel->setFilterKeyColumn(-1);
|
|
|
|
ui->modTreeView->setModel(m_filterModel);
|
|
|
|
ui->modTreeView->installEventFilter(this);
|
|
|
|
ui->modTreeView->sortByColumn(1, Qt::AscendingOrder);
|
2019-07-25 01:02:30 +02:00
|
|
|
ui->modTreeView->setContextMenuPolicy(Qt::CustomContextMenu);
|
|
|
|
connect(ui->modTreeView, &ModListView::customContextMenuRequested, this, &ModFolderPage::ShowContextMenu);
|
2019-08-04 21:13:50 +02:00
|
|
|
connect(ui->modTreeView, &ModListView::activated, this, &ModFolderPage::modItemActivated);
|
2019-07-25 01:02:30 +02:00
|
|
|
|
2018-07-15 14:51:05 +02:00
|
|
|
auto smodel = ui->modTreeView->selectionModel();
|
|
|
|
connect(smodel, &QItemSelectionModel::currentChanged, this, &ModFolderPage::modCurrent);
|
2021-06-18 23:21:12 +01:00
|
|
|
connect(ui->filterEdit, &QLineEdit::textChanged, this, &ModFolderPage::on_filterTextChanged);
|
2019-07-15 01:07:21 +02:00
|
|
|
connect(m_inst, &BaseInstance::runningStatusChanged, this, &ModFolderPage::on_RunningState_changed);
|
2014-06-02 00:49:53 +02:00
|
|
|
}
|
|
|
|
|
2019-08-05 00:44:56 +02:00
|
|
|
void ModFolderPage::modItemActivated(const QModelIndex&)
|
2019-08-04 21:13:50 +02:00
|
|
|
{
|
2019-08-05 00:44:56 +02:00
|
|
|
if(!m_controlsEnabled) {
|
|
|
|
return;
|
2019-08-04 21:13:50 +02:00
|
|
|
}
|
2019-08-05 00:44:56 +02:00
|
|
|
auto selection = m_filterModel->mapSelectionToSource(ui->modTreeView->selectionModel()->selection());
|
|
|
|
m_mods->setModStatus(selection.indexes(), ModFolderModel::Toggle);
|
2019-08-04 21:13:50 +02:00
|
|
|
}
|
|
|
|
|
2019-07-16 01:30:53 +02:00
|
|
|
QMenu * ModFolderPage::createPopupMenu()
|
|
|
|
{
|
|
|
|
QMenu* filteredMenu = QMainWindow::createPopupMenu();
|
|
|
|
filteredMenu->removeAction(ui->actionsToolbar->toggleViewAction() );
|
|
|
|
return filteredMenu;
|
|
|
|
}
|
|
|
|
|
2019-07-25 01:02:30 +02:00
|
|
|
void ModFolderPage::ShowContextMenu(const QPoint& pos)
|
|
|
|
{
|
|
|
|
auto menu = ui->actionsToolbar->createContextMenu(this, tr("Context menu"));
|
|
|
|
menu->exec(ui->modTreeView->mapToGlobal(pos));
|
|
|
|
delete menu;
|
|
|
|
}
|
|
|
|
|
2018-03-19 02:36:12 +01:00
|
|
|
void ModFolderPage::openedImpl()
|
2015-05-11 22:50:35 +02:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
m_mods->startWatching();
|
2015-05-11 22:50:35 +02:00
|
|
|
}
|
|
|
|
|
2018-03-19 02:36:12 +01:00
|
|
|
void ModFolderPage::closedImpl()
|
2015-05-11 22:50:35 +02:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
m_mods->stopWatching();
|
2015-05-11 22:50:35 +02:00
|
|
|
}
|
|
|
|
|
2016-08-04 21:54:25 +02:00
|
|
|
void ModFolderPage::on_filterTextChanged(const QString& newContents)
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
m_viewFilter = newContents;
|
|
|
|
m_filterModel->setFilterFixedString(m_viewFilter);
|
2016-08-04 21:54:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-08-04 03:27:53 +02:00
|
|
|
CoreModFolderPage::CoreModFolderPage(BaseInstance *inst, std::shared_ptr<ModFolderModel> mods,
|
2018-07-15 14:51:05 +02:00
|
|
|
QString id, QString iconName, QString displayName,
|
|
|
|
QString helpPage, QWidget *parent)
|
|
|
|
: ModFolderPage(inst, mods, id, iconName, displayName, helpPage, parent)
|
2014-07-10 00:47:08 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2014-06-02 00:49:53 +02:00
|
|
|
ModFolderPage::~ModFolderPage()
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
m_mods->stopWatching();
|
|
|
|
delete ui;
|
2014-06-02 00:49:53 +02:00
|
|
|
}
|
|
|
|
|
2019-07-15 01:07:21 +02:00
|
|
|
void ModFolderPage::on_RunningState_changed(bool running)
|
|
|
|
{
|
|
|
|
if(m_controlsEnabled == !running) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
m_controlsEnabled = !running;
|
2022-03-20 15:34:13 +01:00
|
|
|
ui->actionsToolbar->setEnabled(m_controlsEnabled);
|
2019-07-15 01:07:21 +02:00
|
|
|
}
|
|
|
|
|
2014-07-12 17:58:23 +02:00
|
|
|
bool ModFolderPage::shouldDisplay() const
|
2014-06-30 02:02:57 +02:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
return true;
|
2014-06-30 02:02:57 +02:00
|
|
|
}
|
|
|
|
|
2022-02-22 18:23:53 +00:00
|
|
|
void ModFolderPage::retranslate()
|
|
|
|
{
|
|
|
|
ui->retranslateUi(this);
|
|
|
|
}
|
|
|
|
|
2014-07-12 17:58:23 +02:00
|
|
|
bool CoreModFolderPage::shouldDisplay() const
|
2014-07-10 00:47:08 +02:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
if (ModFolderPage::shouldDisplay())
|
|
|
|
{
|
|
|
|
auto inst = dynamic_cast<MinecraftInstance *>(m_inst);
|
|
|
|
if (!inst)
|
|
|
|
return true;
|
2020-06-27 12:02:31 +02:00
|
|
|
auto version = inst->getPackProfile();
|
2018-07-15 14:51:05 +02:00
|
|
|
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;
|
2014-07-10 00:47:08 +02:00
|
|
|
}
|
|
|
|
|
2014-06-02 00:49:53 +02:00
|
|
|
bool ModFolderPage::modListFilter(QKeyEvent *keyEvent)
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
switch (keyEvent->key())
|
|
|
|
{
|
|
|
|
case Qt::Key_Delete:
|
2019-07-16 01:30:53 +02:00
|
|
|
on_actionRemove_triggered();
|
2018-07-15 14:51:05 +02:00
|
|
|
return true;
|
|
|
|
case Qt::Key_Plus:
|
2019-07-16 01:30:53 +02:00
|
|
|
on_actionAdd_triggered();
|
2018-07-15 14:51:05 +02:00
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return QWidget::eventFilter(ui->modTreeView, keyEvent);
|
2014-06-02 00:49:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ModFolderPage::eventFilter(QObject *obj, QEvent *ev)
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
if (ev->type() != QEvent::KeyPress)
|
|
|
|
{
|
|
|
|
return QWidget::eventFilter(obj, ev);
|
|
|
|
}
|
|
|
|
QKeyEvent *keyEvent = static_cast<QKeyEvent *>(ev);
|
|
|
|
if (obj == ui->modTreeView)
|
|
|
|
return modListFilter(keyEvent);
|
|
|
|
return QWidget::eventFilter(obj, ev);
|
2014-06-02 00:49:53 +02:00
|
|
|
}
|
|
|
|
|
2019-07-16 01:30:53 +02:00
|
|
|
void ModFolderPage::on_actionAdd_triggered()
|
2014-06-02 00:49:53 +02:00
|
|
|
{
|
2019-07-15 01:07:21 +02:00
|
|
|
if(!m_controlsEnabled) {
|
|
|
|
return;
|
|
|
|
}
|
2018-07-15 14:51:05 +02:00
|
|
|
auto list = GuiUtil::BrowseForFiles(
|
|
|
|
m_helpName,
|
|
|
|
tr("Select %1",
|
|
|
|
"Select whatever type of files the page contains. Example: 'Loader Mods'")
|
|
|
|
.arg(m_displayName),
|
2021-11-20 16:22:22 +01:00
|
|
|
m_fileSelectionFilter.arg(m_displayName), APPLICATION->settings()->get("CentralModsDir").toString(),
|
2018-07-15 14:51:05 +02:00
|
|
|
this->parentWidget());
|
|
|
|
if (!list.empty())
|
|
|
|
{
|
|
|
|
for (auto filename : list)
|
|
|
|
{
|
|
|
|
m_mods->installMod(filename);
|
|
|
|
}
|
|
|
|
}
|
2014-06-02 00:49:53 +02:00
|
|
|
}
|
2015-09-09 23:53:33 +02:00
|
|
|
|
2019-07-16 01:30:53 +02:00
|
|
|
void ModFolderPage::on_actionEnable_triggered()
|
2016-08-04 23:16:03 +02:00
|
|
|
{
|
2019-07-15 01:07:21 +02:00
|
|
|
if(!m_controlsEnabled) {
|
|
|
|
return;
|
|
|
|
}
|
2018-07-15 14:51:05 +02:00
|
|
|
auto selection = m_filterModel->mapSelectionToSource(ui->modTreeView->selectionModel()->selection());
|
2019-08-05 00:44:56 +02:00
|
|
|
m_mods->setModStatus(selection.indexes(), ModFolderModel::Enable);
|
2016-08-04 23:16:03 +02:00
|
|
|
}
|
|
|
|
|
2019-07-16 01:30:53 +02:00
|
|
|
void ModFolderPage::on_actionDisable_triggered()
|
2016-08-04 23:16:03 +02:00
|
|
|
{
|
2019-07-15 01:07:21 +02:00
|
|
|
if(!m_controlsEnabled) {
|
|
|
|
return;
|
|
|
|
}
|
2018-07-15 14:51:05 +02:00
|
|
|
auto selection = m_filterModel->mapSelectionToSource(ui->modTreeView->selectionModel()->selection());
|
2019-08-05 00:44:56 +02:00
|
|
|
m_mods->setModStatus(selection.indexes(), ModFolderModel::Disable);
|
2016-08-04 23:16:03 +02:00
|
|
|
}
|
|
|
|
|
2019-07-16 01:30:53 +02:00
|
|
|
void ModFolderPage::on_actionRemove_triggered()
|
2014-06-02 00:49:53 +02:00
|
|
|
{
|
2019-07-15 01:07:21 +02:00
|
|
|
if(!m_controlsEnabled) {
|
|
|
|
return;
|
|
|
|
}
|
2018-07-15 14:51:05 +02:00
|
|
|
auto selection = m_filterModel->mapSelectionToSource(ui->modTreeView->selectionModel()->selection());
|
|
|
|
m_mods->deleteMods(selection.indexes());
|
2014-06-02 00:49:53 +02:00
|
|
|
}
|
|
|
|
|
2022-01-14 09:43:42 +01:00
|
|
|
void ModFolderPage::on_actionInstall_mods_triggered()
|
|
|
|
{
|
|
|
|
if(!m_controlsEnabled) {
|
|
|
|
return;
|
|
|
|
}
|
2022-01-14 20:22:15 +01:00
|
|
|
if(m_inst->typeName() != "Minecraft"){
|
|
|
|
return; //this is a null instance or a legacy instance
|
|
|
|
}
|
2022-02-20 20:55:26 +01:00
|
|
|
auto profile = ((MinecraftInstance *)m_inst)->getPackProfile();
|
2022-04-14 21:55:03 +02:00
|
|
|
if (profile->getModLoader() == ModAPI::Unspecified) {
|
2022-01-24 07:12:19 +01:00
|
|
|
QMessageBox::critical(this,tr("Error"),tr("Please install a mod loader first!"));
|
2022-01-14 20:22:15 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
ModDownloadDialog mdownload(m_mods, this, m_inst);
|
2022-04-01 09:10:51 -03:00
|
|
|
if (mdownload.exec()) {
|
|
|
|
SequentialTask* tasks = new SequentialTask(this);
|
|
|
|
connect(tasks, &Task::failed, [this, tasks](QString reason) {
|
|
|
|
CustomMessageBox::selectable(this, tr("Error"), reason, QMessageBox::Critical)->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);
|
2022-01-14 20:22:15 +01:00
|
|
|
}
|
2022-04-01 09:10:51 -03:00
|
|
|
ProgressDialog loadDialog(this);
|
|
|
|
loadDialog.setSkipButton(true, tr("Abort"));
|
|
|
|
loadDialog.execWithTask(tasks);
|
|
|
|
m_mods->update();
|
2022-01-14 09:43:42 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-16 01:30:53 +02:00
|
|
|
void ModFolderPage::on_actionView_configs_triggered()
|
2016-11-26 14:59:27 +01:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
DesktopServices::openDirectory(m_inst->instanceConfigFolder(), true);
|
2016-11-26 14:59:27 +01:00
|
|
|
}
|
|
|
|
|
2019-07-16 01:30:53 +02:00
|
|
|
void ModFolderPage::on_actionView_Folder_triggered()
|
2014-06-02 00:49:53 +02:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
DesktopServices::openDirectory(m_mods->dir().absolutePath(), true);
|
2014-06-02 00:49:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void ModFolderPage::modCurrent(const QModelIndex ¤t, const QModelIndex &previous)
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
if (!current.isValid())
|
|
|
|
{
|
|
|
|
ui->frame->clear();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
auto sourceCurrent = m_filterModel->mapToSource(current);
|
|
|
|
int row = sourceCurrent.row();
|
|
|
|
Mod &m = m_mods->operator[](row);
|
|
|
|
ui->frame->updateWithMod(m);
|
2014-06-02 00:49:53 +02:00
|
|
|
}
|