/* SPDX-FileCopyrightText: 2026 Project Tick
* SPDX-FileContributor: Project Tick
* SPDX-License-Identifier: GPL-3.0-or-later
*
* MeshMC - A Custom Launcher for Minecraft
* Copyright (C) 2026 Project Tick
*
* 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, either version 3 of the License, or
* (at your option) any later version.
*
* 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 .
*/
#include "ModrinthPage.h"
#include "ui_ModrinthPage.h"
#include
#include "Application.h"
#include "Json.h"
#include "ui/dialogs/NewInstanceDialog.h"
#include "InstanceImportTask.h"
#include "ModrinthModel.h"
ModrinthPage::ModrinthPage(NewInstanceDialog* dialog, QWidget* parent)
: QWidget(parent), ui(new Ui::ModrinthPage), dialog(dialog)
{
ui->setupUi(this);
connect(ui->searchButton, &QPushButton::clicked, this,
&ModrinthPage::triggerSearch);
ui->searchEdit->installEventFilter(this);
listModel = new Modrinth::ListModel(this);
ui->packView->setModel(listModel);
ui->versionSelectionBox->view()->setVerticalScrollBarPolicy(
Qt::ScrollBarAsNeeded);
ui->versionSelectionBox->view()->parentWidget()->setMaximumHeight(300);
ui->sortByBox->addItem(tr("Sort by relevance"));
ui->sortByBox->addItem(tr("Sort by downloads"));
ui->sortByBox->addItem(tr("Sort by last updated"));
ui->sortByBox->addItem(tr("Sort by newest"));
ui->sortByBox->addItem(tr("Sort by follows"));
connect(ui->sortByBox, QOverload::of(&QComboBox::currentIndexChanged),
this, &ModrinthPage::triggerSearch);
connect(ui->packView->selectionModel(),
&QItemSelectionModel::currentChanged, this,
&ModrinthPage::onSelectionChanged);
connect(ui->versionSelectionBox, &QComboBox::currentTextChanged, this,
&ModrinthPage::onVersionSelectionChanged);
}
ModrinthPage::~ModrinthPage()
{
delete ui;
}
bool ModrinthPage::eventFilter(QObject* watched, QEvent* event)
{
if (watched == ui->searchEdit && event->type() == QEvent::KeyPress) {
QKeyEvent* keyEvent = static_cast(event);
if (keyEvent->key() == Qt::Key_Return) {
triggerSearch();
keyEvent->accept();
return true;
}
}
return QWidget::eventFilter(watched, event);
}
bool ModrinthPage::shouldDisplay() const
{
return true;
}
void ModrinthPage::openedImpl()
{
suggestCurrent();
triggerSearch();
}
void ModrinthPage::triggerSearch()
{
listModel->searchWithTerm(ui->searchEdit->text(),
ui->sortByBox->currentIndex());
}
void ModrinthPage::onSelectionChanged(QModelIndex first, QModelIndex second)
{
ui->versionSelectionBox->clear();
if (!first.isValid()) {
if (isOpened) {
dialog->setSuggestedPack();
}
return;
}
current =
listModel->data(first, Qt::UserRole).value();
QString text = "";
QString name = current.name;
if (current.slug.isEmpty()) {
text = name;
} else {
text = "" + name + "";
}
if (!current.author.isEmpty()) {
text += "
" + tr(" by ") + current.author;
}
text += "
";
ui->packDescription->setHtml(text + current.description);
if (isOpened) {
dialog->setSuggestedPack(current.name);
}
if (!current.versionsLoaded) {
qDebug() << "Loading Modrinth modpack versions";
NetJob* netJob =
new NetJob(QString("Modrinth::PackVersions(%1)").arg(current.name),
APPLICATION->network());
std::shared_ptr versionResponse =
std::make_shared();
QString projectId = current.projectId;
netJob->addNetAction(Net::Download::makeByteArray(
QString("https://api.modrinth.com/v2/project/%1/version?"
"loaders=[\"forge\",\"fabric\",\"quilt\",\"neoforge\"]")
.arg(projectId),
versionResponse.get()));
QObject::connect(
netJob, &NetJob::succeeded, this, [this, netJob, versionResponse] {
netJob->deleteLater();
QJsonParseError parse_error;
QJsonDocument doc =
QJsonDocument::fromJson(*versionResponse, &parse_error);
if (parse_error.error != QJsonParseError::NoError) {
qWarning()
<< "Error while parsing JSON response from Modrinth at "
<< parse_error.offset
<< " reason: " << parse_error.errorString();
qWarning() << *versionResponse;
return;
}
QJsonArray arr = doc.array();
try {
Modrinth::loadIndexedPackVersions(current, arr);
} catch (const JSONValidationError& e) {
qDebug() << *versionResponse;
qWarning()
<< "Error while reading Modrinth modpack version: "
<< e.cause();
}
for (auto version : current.versions) {
QString label = version.versionNumber;
if (!version.mcVersion.isEmpty()) {
label += " [" + version.mcVersion + "]";
}
if (!version.loaders.isEmpty()) {
label += " (" + version.loaders + ")";
}
ui->versionSelectionBox->addItem(
label, QVariant(version.downloadUrl));
}
suggestCurrent();
});
QObject::connect(netJob, &NetJob::failed, this,
[netJob] { netJob->deleteLater(); });
netJob->start();
} else {
for (auto version : current.versions) {
QString label = version.versionNumber;
if (!version.mcVersion.isEmpty()) {
label += " [" + version.mcVersion + "]";
}
if (!version.loaders.isEmpty()) {
label += " (" + version.loaders + ")";
}
ui->versionSelectionBox->addItem(label,
QVariant(version.downloadUrl));
}
suggestCurrent();
}
}
void ModrinthPage::suggestCurrent()
{
if (!isOpened) {
return;
}
if (selectedVersion.isEmpty()) {
dialog->setSuggestedPack();
return;
}
dialog->setSuggestedPack(current.name,
new InstanceImportTask(selectedVersion));
QString editedLogoName;
editedLogoName = "modrinth_" + current.slug;
listModel->getLogo(
current.slug, current.iconUrl, [this, editedLogoName](QString logo) {
dialog->setSuggestedIconFromFile(logo, editedLogoName);
});
}
void ModrinthPage::onVersionSelectionChanged(QString data)
{
if (data.isNull() || data.isEmpty()) {
selectedVersion = "";
return;
}
selectedVersion = ui->versionSelectionBox->currentData().toString();
suggestCurrent();
}