Merge pull request #392 from flowln/mod_filter
This commit is contained in:
commit
a42d2afcee
@ -825,7 +825,6 @@ SET(LAUNCHER_SOURCES
|
|||||||
ui/dialogs/ModDownloadDialog.cpp
|
ui/dialogs/ModDownloadDialog.cpp
|
||||||
ui/dialogs/ModDownloadDialog.h
|
ui/dialogs/ModDownloadDialog.h
|
||||||
|
|
||||||
|
|
||||||
# GUI - widgets
|
# GUI - widgets
|
||||||
ui/widgets/Common.cpp
|
ui/widgets/Common.cpp
|
||||||
ui/widgets/Common.h
|
ui/widgets/Common.h
|
||||||
@ -849,6 +848,8 @@ SET(LAUNCHER_SOURCES
|
|||||||
ui/widgets/LogView.h
|
ui/widgets/LogView.h
|
||||||
ui/widgets/MCModInfoFrame.cpp
|
ui/widgets/MCModInfoFrame.cpp
|
||||||
ui/widgets/MCModInfoFrame.h
|
ui/widgets/MCModInfoFrame.h
|
||||||
|
ui/widgets/ModFilterWidget.cpp
|
||||||
|
ui/widgets/ModFilterWidget.h
|
||||||
ui/widgets/ModListView.cpp
|
ui/widgets/ModListView.cpp
|
||||||
ui/widgets/ModListView.h
|
ui/widgets/ModListView.h
|
||||||
ui/widgets/PageContainer.cpp
|
ui/widgets/PageContainer.cpp
|
||||||
@ -907,6 +908,7 @@ qt5_wrap_ui(LAUNCHER_UI
|
|||||||
ui/widgets/InstanceCardWidget.ui
|
ui/widgets/InstanceCardWidget.ui
|
||||||
ui/widgets/CustomCommands.ui
|
ui/widgets/CustomCommands.ui
|
||||||
ui/widgets/MCModInfoFrame.ui
|
ui/widgets/MCModInfoFrame.ui
|
||||||
|
ui/widgets/ModFilterWidget.ui
|
||||||
ui/dialogs/CopyInstanceDialog.ui
|
ui/dialogs/CopyInstanceDialog.ui
|
||||||
ui/dialogs/ProfileSetupDialog.ui
|
ui/dialogs/ProfileSetupDialog.ui
|
||||||
ui/dialogs/ProgressDialog.ui
|
ui/dialogs/ProgressDialog.ui
|
||||||
|
@ -3,6 +3,8 @@
|
|||||||
#include <QString>
|
#include <QString>
|
||||||
#include <QList>
|
#include <QList>
|
||||||
|
|
||||||
|
#include "Version.h"
|
||||||
|
|
||||||
namespace ModPlatform {
|
namespace ModPlatform {
|
||||||
class ListModel;
|
class ListModel;
|
||||||
}
|
}
|
||||||
@ -22,7 +24,7 @@ class ModAPI {
|
|||||||
QString search;
|
QString search;
|
||||||
QString sorting;
|
QString sorting;
|
||||||
ModLoaderType mod_loader;
|
ModLoaderType mod_loader;
|
||||||
QString version;
|
std::list<Version> versions;
|
||||||
};
|
};
|
||||||
|
|
||||||
virtual void searchMods(CallerType* caller, SearchArgs&& args) const = 0;
|
virtual void searchMods(CallerType* caller, SearchArgs&& args) const = 0;
|
||||||
@ -30,7 +32,7 @@ class ModAPI {
|
|||||||
|
|
||||||
struct VersionSearchArgs {
|
struct VersionSearchArgs {
|
||||||
QString addonId;
|
QString addonId;
|
||||||
QList<QString> mcVersions;
|
std::list<Version> mcVersions;
|
||||||
ModLoaderType loader;
|
ModLoaderType loader;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -53,4 +55,15 @@ class ModAPI {
|
|||||||
}
|
}
|
||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
inline auto getGameVersionsString(std::list<Version> mcVersions) const -> QString
|
||||||
|
{
|
||||||
|
QString s;
|
||||||
|
for(auto& ver : mcVersions){
|
||||||
|
s += QString("%1,").arg(ver.toString());
|
||||||
|
}
|
||||||
|
s.remove(s.length() - 1, 1); //remove last comma
|
||||||
|
return s;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
@ -6,6 +6,8 @@ class FlameAPI : public NetworkModAPI {
|
|||||||
private:
|
private:
|
||||||
inline auto getModSearchURL(SearchArgs& args) const -> QString override
|
inline auto getModSearchURL(SearchArgs& args) const -> QString override
|
||||||
{
|
{
|
||||||
|
auto gameVersionStr = args.versions.size() != 0 ? QString("gameVersion=%1").arg(args.versions.front().toString()) : QString();
|
||||||
|
|
||||||
return QString(
|
return QString(
|
||||||
"https://addons-ecs.forgesvc.net/api/v2/addon/search?"
|
"https://addons-ecs.forgesvc.net/api/v2/addon/search?"
|
||||||
"gameId=432&"
|
"gameId=432&"
|
||||||
@ -17,12 +19,12 @@ class FlameAPI : public NetworkModAPI {
|
|||||||
"searchFilter=%2&"
|
"searchFilter=%2&"
|
||||||
"sort=%3&"
|
"sort=%3&"
|
||||||
"modLoaderType=%4&"
|
"modLoaderType=%4&"
|
||||||
"gameVersion=%5")
|
"%5")
|
||||||
.arg(args.offset)
|
.arg(args.offset)
|
||||||
.arg(args.search)
|
.arg(args.search)
|
||||||
.arg(args.sorting)
|
.arg(args.sorting)
|
||||||
.arg(args.mod_loader)
|
.arg(args.mod_loader)
|
||||||
.arg(args.version);
|
.arg(gameVersionStr);
|
||||||
};
|
};
|
||||||
|
|
||||||
inline auto getVersionsURL(VersionSearchArgs& args) const -> QString override
|
inline auto getVersionsURL(VersionSearchArgs& args) const -> QString override
|
||||||
|
@ -22,12 +22,12 @@ class ModrinthAPI : public NetworkModAPI {
|
|||||||
"limit=25&"
|
"limit=25&"
|
||||||
"query=%2&"
|
"query=%2&"
|
||||||
"index=%3&"
|
"index=%3&"
|
||||||
"facets=[[\"categories:%4\"],[\"versions:%5\"],[\"project_type:mod\"]]")
|
"facets=[[\"categories:%4\"],%5[\"project_type:mod\"]]")
|
||||||
.arg(args.offset)
|
.arg(args.offset)
|
||||||
.arg(args.search)
|
.arg(args.search)
|
||||||
.arg(args.sorting)
|
.arg(args.sorting)
|
||||||
.arg(getModLoaderString(args.mod_loader))
|
.arg(getModLoaderString(args.mod_loader))
|
||||||
.arg(args.version);
|
.arg(getGameVersionsArray(args.versions));
|
||||||
};
|
};
|
||||||
|
|
||||||
inline auto getVersionsURL(VersionSearchArgs& args) const -> QString override
|
inline auto getVersionsURL(VersionSearchArgs& args) const -> QString override
|
||||||
@ -40,15 +40,14 @@ class ModrinthAPI : public NetworkModAPI {
|
|||||||
.arg(getModLoaderString(args.loader));
|
.arg(getModLoaderString(args.loader));
|
||||||
};
|
};
|
||||||
|
|
||||||
inline auto getGameVersionsString(QList<QString> mcVersions) const -> QString
|
auto getGameVersionsArray(std::list<Version> mcVersions) const -> QString
|
||||||
{
|
{
|
||||||
QString s;
|
QString s;
|
||||||
for(int i = 0; i < mcVersions.count(); i++){
|
for(auto& ver : mcVersions){
|
||||||
s += mcVersions.at(i);
|
s += QString("\"versions:%1\",").arg(ver.toString());
|
||||||
if(i < mcVersions.count() - 1)
|
|
||||||
s += ",";
|
|
||||||
}
|
}
|
||||||
return s;
|
s.remove(s.length() - 1, 1); //remove last comma
|
||||||
|
return s.isEmpty() ? QString() : QString("[%1],").arg(s);
|
||||||
}
|
}
|
||||||
|
|
||||||
static auto getModLoaderString(ModLoaderType type) -> const QString
|
static auto getModLoaderString(ModLoaderType type) -> const QString
|
||||||
|
@ -22,7 +22,9 @@ ModDownloadDialog::ModDownloadDialog(const std::shared_ptr<ModFolderModel> &mods
|
|||||||
: QDialog(parent), mods(mods), m_instance(instance)
|
: QDialog(parent), mods(mods), m_instance(instance)
|
||||||
{
|
{
|
||||||
setObjectName(QStringLiteral("ModDownloadDialog"));
|
setObjectName(QStringLiteral("ModDownloadDialog"));
|
||||||
resize(400, 347);
|
|
||||||
|
resize(std::max(0.5*parent->width(), 400.0), std::max(0.75*parent->height(), 400.0));
|
||||||
|
|
||||||
m_verticalLayout = new QVBoxLayout(this);
|
m_verticalLayout = new QVBoxLayout(this);
|
||||||
m_verticalLayout->setObjectName(QStringLiteral("verticalLayout"));
|
m_verticalLayout->setObjectName(QStringLiteral("verticalLayout"));
|
||||||
|
|
||||||
@ -80,7 +82,7 @@ void ModDownloadDialog::confirm()
|
|||||||
tr("Confirm mods to download")
|
tr("Confirm mods to download")
|
||||||
);
|
);
|
||||||
|
|
||||||
for(auto task : keys){
|
for(auto& task : keys){
|
||||||
confirm_dialog->appendMod(task, modTask.find(task).value()->getFilename());
|
confirm_dialog->appendMod(task, modTask.find(task).value()->getFilename());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -72,14 +72,11 @@ void ListModel::performPaginatedSearch()
|
|||||||
auto profile = (dynamic_cast<MinecraftInstance*>((dynamic_cast<ModPage*>(parent()))->m_instance))->getPackProfile();
|
auto profile = (dynamic_cast<MinecraftInstance*>((dynamic_cast<ModPage*>(parent()))->m_instance))->getPackProfile();
|
||||||
|
|
||||||
m_parent->apiProvider()->searchMods(this,
|
m_parent->apiProvider()->searchMods(this,
|
||||||
{ nextSearchOffset, currentSearchTerm, getSorts()[currentSort], profile->getModLoader(), getMineVersions().at(0) });
|
{ nextSearchOffset, currentSearchTerm, getSorts()[currentSort], profile->getModLoader(), getMineVersions() });
|
||||||
}
|
}
|
||||||
|
|
||||||
void ListModel::searchWithTerm(const QString& term, const int sort)
|
void ListModel::refresh()
|
||||||
{
|
{
|
||||||
if (currentSearchTerm == term && currentSearchTerm.isNull() == term.isNull() && currentSort == sort) { return; }
|
|
||||||
currentSearchTerm = term;
|
|
||||||
currentSort = sort;
|
|
||||||
if (jobPtr) {
|
if (jobPtr) {
|
||||||
jobPtr->abort();
|
jobPtr->abort();
|
||||||
searchState = ResetRequested;
|
searchState = ResetRequested;
|
||||||
@ -94,6 +91,20 @@ void ListModel::searchWithTerm(const QString& term, const int sort)
|
|||||||
performPaginatedSearch();
|
performPaginatedSearch();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ListModel::searchWithTerm(const QString& term, const int sort, const bool filter_changed)
|
||||||
|
{
|
||||||
|
if (currentSearchTerm == term
|
||||||
|
&& currentSearchTerm.isNull() == term.isNull()
|
||||||
|
&& currentSort == sort
|
||||||
|
&& !filter_changed)
|
||||||
|
{ return; }
|
||||||
|
|
||||||
|
currentSearchTerm = term;
|
||||||
|
currentSort = sort;
|
||||||
|
|
||||||
|
refresh();
|
||||||
|
}
|
||||||
|
|
||||||
void ListModel::getLogo(const QString& logo, const QString& logoUrl, LogoCallback callback)
|
void ListModel::getLogo(const QString& logo, const QString& logoUrl, LogoCallback callback)
|
||||||
{
|
{
|
||||||
if (m_logoMap.contains(logo)) {
|
if (m_logoMap.contains(logo)) {
|
||||||
@ -223,9 +234,7 @@ void ListModel::versionRequestSucceeded(QJsonDocument doc, QString addonId)
|
|||||||
|
|
||||||
/******** Helpers ********/
|
/******** Helpers ********/
|
||||||
|
|
||||||
auto ModPlatform::ListModel::getMineVersions() const -> QList<QString>
|
auto ModPlatform::ListModel::getMineVersions() const -> std::list<Version>
|
||||||
{
|
{
|
||||||
return { (dynamic_cast<MinecraftInstance*>((dynamic_cast<ModPage*>(parent()))->m_instance))
|
return m_parent->getFilter()->versions;
|
||||||
->getPackProfile()
|
|
||||||
->getComponentVersion("net.minecraft") };
|
|
||||||
}
|
}
|
||||||
|
@ -7,6 +7,7 @@
|
|||||||
#include "net/NetJob.h"
|
#include "net/NetJob.h"
|
||||||
|
|
||||||
class ModPage;
|
class ModPage;
|
||||||
|
class Version;
|
||||||
|
|
||||||
namespace ModPlatform {
|
namespace ModPlatform {
|
||||||
|
|
||||||
@ -33,7 +34,8 @@ class ListModel : public QAbstractListModel {
|
|||||||
|
|
||||||
/* Ask the API for more information */
|
/* Ask the API for more information */
|
||||||
void fetchMore(const QModelIndex& parent) override;
|
void fetchMore(const QModelIndex& parent) override;
|
||||||
void searchWithTerm(const QString& term, const int sort);
|
void refresh();
|
||||||
|
void searchWithTerm(const QString& term, const int sort, const bool filter_changed);
|
||||||
void requestModVersions(const ModPlatform::IndexedPack& current);
|
void requestModVersions(const ModPlatform::IndexedPack& current);
|
||||||
|
|
||||||
virtual void loadIndexedPack(ModPlatform::IndexedPack& m, QJsonObject& obj) = 0;
|
virtual void loadIndexedPack(ModPlatform::IndexedPack& m, QJsonObject& obj) = 0;
|
||||||
@ -62,7 +64,7 @@ class ListModel : public QAbstractListModel {
|
|||||||
|
|
||||||
void requestLogo(QString file, QString url);
|
void requestLogo(QString file, QString url);
|
||||||
|
|
||||||
inline auto getMineVersions() const -> QList<QString>;
|
inline auto getMineVersions() const -> std::list<Version>;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
ModPage* m_parent;
|
ModPage* m_parent;
|
||||||
|
@ -2,21 +2,39 @@
|
|||||||
#include "ui_ModPage.h"
|
#include "ui_ModPage.h"
|
||||||
|
|
||||||
#include <QKeyEvent>
|
#include <QKeyEvent>
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
#include "minecraft/MinecraftInstance.h"
|
#include "minecraft/MinecraftInstance.h"
|
||||||
#include "minecraft/PackProfile.h"
|
#include "minecraft/PackProfile.h"
|
||||||
#include "ui/dialogs/ModDownloadDialog.h"
|
#include "ui/dialogs/ModDownloadDialog.h"
|
||||||
|
|
||||||
ModPage::ModPage(ModDownloadDialog* dialog, BaseInstance* instance, ModAPI* api)
|
ModPage::ModPage(ModDownloadDialog* dialog, BaseInstance* instance, ModAPI* api)
|
||||||
: QWidget(dialog), m_instance(instance), ui(new Ui::ModPage), dialog(dialog), api(api)
|
: QWidget(dialog)
|
||||||
|
, m_instance(instance)
|
||||||
|
, ui(new Ui::ModPage)
|
||||||
|
, dialog(dialog)
|
||||||
|
, filter_widget(static_cast<MinecraftInstance*>(instance)->getPackProfile()->getComponentVersion("net.minecraft"), this)
|
||||||
|
, api(api)
|
||||||
{
|
{
|
||||||
ui->setupUi(this);
|
ui->setupUi(this);
|
||||||
connect(ui->searchButton, &QPushButton::clicked, this, &ModPage::triggerSearch);
|
connect(ui->searchButton, &QPushButton::clicked, this, &ModPage::triggerSearch);
|
||||||
|
connect(ui->modFilterButton, &QPushButton::clicked, this, &ModPage::filterMods);
|
||||||
ui->searchEdit->installEventFilter(this);
|
ui->searchEdit->installEventFilter(this);
|
||||||
|
|
||||||
ui->versionSelectionBox->view()->setVerticalScrollBarPolicy(Qt::ScrollBarAsNeeded);
|
ui->versionSelectionBox->view()->setVerticalScrollBarPolicy(Qt::ScrollBarAsNeeded);
|
||||||
ui->versionSelectionBox->view()->parentWidget()->setMaximumHeight(300);
|
ui->versionSelectionBox->view()->parentWidget()->setMaximumHeight(300);
|
||||||
|
|
||||||
|
ui->gridLayout_3->addWidget(&filter_widget, 0, 0, 1, ui->gridLayout_3->columnCount());
|
||||||
|
|
||||||
|
filter_widget.setInstance(static_cast<MinecraftInstance*>(m_instance));
|
||||||
|
m_filter = filter_widget.getFilter();
|
||||||
|
|
||||||
|
connect(&filter_widget, &ModFilterWidget::filterChanged, this, [&]{
|
||||||
|
ui->searchButton->setStyleSheet("text-decoration: underline");
|
||||||
|
});
|
||||||
|
connect(&filter_widget, &ModFilterWidget::filterUnchanged, this, [&]{
|
||||||
|
ui->searchButton->setStyleSheet("text-decoration: none");
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
ModPage::~ModPage()
|
ModPage::~ModPage()
|
||||||
@ -49,9 +67,24 @@ auto ModPage::eventFilter(QObject* watched, QEvent* event) -> bool
|
|||||||
|
|
||||||
/******** Callbacks to events in the UI (set up in the derived classes) ********/
|
/******** Callbacks to events in the UI (set up in the derived classes) ********/
|
||||||
|
|
||||||
|
void ModPage::filterMods()
|
||||||
|
{
|
||||||
|
filter_widget.setHidden(!filter_widget.isHidden());
|
||||||
|
}
|
||||||
|
|
||||||
void ModPage::triggerSearch()
|
void ModPage::triggerSearch()
|
||||||
{
|
{
|
||||||
listModel->searchWithTerm(ui->searchEdit->text(), ui->sortByBox->currentIndex());
|
auto changed = filter_widget.changed();
|
||||||
|
m_filter = filter_widget.getFilter();
|
||||||
|
|
||||||
|
if(changed){
|
||||||
|
ui->packView->clearSelection();
|
||||||
|
ui->packDescription->clear();
|
||||||
|
ui->versionSelectionBox->clear();
|
||||||
|
updateSelectionButton();
|
||||||
|
}
|
||||||
|
|
||||||
|
listModel->searchWithTerm(ui->searchEdit->text(), ui->sortByBox->currentIndex(), changed);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ModPage::onSelectionChanged(QModelIndex first, QModelIndex second)
|
void ModPage::onSelectionChanged(QModelIndex first, QModelIndex second)
|
||||||
@ -131,7 +164,7 @@ void ModPage::retranslate()
|
|||||||
ui->retranslateUi(this);
|
ui->retranslateUi(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ModPage::updateModVersions()
|
void ModPage::updateModVersions(int prev_count)
|
||||||
{
|
{
|
||||||
auto packProfile = (dynamic_cast<MinecraftInstance*>(m_instance))->getPackProfile();
|
auto packProfile = (dynamic_cast<MinecraftInstance*>(m_instance))->getPackProfile();
|
||||||
|
|
||||||
@ -141,15 +174,22 @@ void ModPage::updateModVersions()
|
|||||||
|
|
||||||
for (int i = 0; i < current.versions.size(); i++) {
|
for (int i = 0; i < current.versions.size(); i++) {
|
||||||
auto version = current.versions[i];
|
auto version = current.versions[i];
|
||||||
|
bool valid = false;
|
||||||
|
for(auto& mcVer : m_filter->versions){
|
||||||
//NOTE: Flame doesn't care about loaderString, so passing it changes nothing.
|
//NOTE: Flame doesn't care about loaderString, so passing it changes nothing.
|
||||||
if (!validateVersion(version, mcVersion, loaderString)) {
|
if (validateVersion(version, mcVer.toString(), loaderString)) {
|
||||||
continue;
|
valid = true;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
if(valid || m_filter->versions.size() == 0)
|
||||||
ui->versionSelectionBox->addItem(version.version, QVariant(i));
|
ui->versionSelectionBox->addItem(version.version, QVariant(i));
|
||||||
}
|
}
|
||||||
if (ui->versionSelectionBox->count() == 0) { ui->versionSelectionBox->addItem(tr("No valid version found!"), QVariant(-1)); }
|
if (ui->versionSelectionBox->count() == 0 && prev_count != 0) {
|
||||||
|
ui->versionSelectionBox->addItem(tr("No valid version found!"), QVariant(-1));
|
||||||
ui->modSelectionButton->setText(tr("Cannot select invalid version :("));
|
ui->modSelectionButton->setText(tr("Cannot select invalid version :("));
|
||||||
|
}
|
||||||
|
|
||||||
updateSelectionButton();
|
updateSelectionButton();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -7,6 +7,7 @@
|
|||||||
#include "modplatform/ModIndex.h"
|
#include "modplatform/ModIndex.h"
|
||||||
#include "ui/pages/BasePage.h"
|
#include "ui/pages/BasePage.h"
|
||||||
#include "ui/pages/modplatform/ModModel.h"
|
#include "ui/pages/modplatform/ModModel.h"
|
||||||
|
#include "ui/widgets/ModFilterWidget.h"
|
||||||
|
|
||||||
class ModDownloadDialog;
|
class ModDownloadDialog;
|
||||||
|
|
||||||
@ -39,9 +40,10 @@ class ModPage : public QWidget, public BasePage {
|
|||||||
virtual auto validateVersion(ModPlatform::IndexedVersion& ver, QString mineVer, QString loaderVer = "") const -> bool = 0;
|
virtual auto validateVersion(ModPlatform::IndexedVersion& ver, QString mineVer, QString loaderVer = "") const -> bool = 0;
|
||||||
|
|
||||||
auto apiProvider() const -> const ModAPI* { return api.get(); };
|
auto apiProvider() const -> const ModAPI* { return api.get(); };
|
||||||
|
auto getFilter() const -> const std::shared_ptr<ModFilterWidget::Filter> { return m_filter; }
|
||||||
|
|
||||||
auto getCurrent() -> ModPlatform::IndexedPack& { return current; }
|
auto getCurrent() -> ModPlatform::IndexedPack& { return current; }
|
||||||
void updateModVersions();
|
void updateModVersions(int prev_count = -1);
|
||||||
|
|
||||||
void openedImpl() override;
|
void openedImpl() override;
|
||||||
auto eventFilter(QObject* watched, QEvent* event) -> bool override;
|
auto eventFilter(QObject* watched, QEvent* event) -> bool override;
|
||||||
@ -52,6 +54,7 @@ class ModPage : public QWidget, public BasePage {
|
|||||||
void updateSelectionButton();
|
void updateSelectionButton();
|
||||||
|
|
||||||
protected slots:
|
protected slots:
|
||||||
|
virtual void filterMods();
|
||||||
void triggerSearch();
|
void triggerSearch();
|
||||||
void onSelectionChanged(QModelIndex first, QModelIndex second);
|
void onSelectionChanged(QModelIndex first, QModelIndex second);
|
||||||
void onVersionSelectionChanged(QString data);
|
void onVersionSelectionChanged(QString data);
|
||||||
@ -60,6 +63,10 @@ class ModPage : public QWidget, public BasePage {
|
|||||||
protected:
|
protected:
|
||||||
Ui::ModPage* ui = nullptr;
|
Ui::ModPage* ui = nullptr;
|
||||||
ModDownloadDialog* dialog = nullptr;
|
ModDownloadDialog* dialog = nullptr;
|
||||||
|
|
||||||
|
ModFilterWidget filter_widget;
|
||||||
|
std::shared_ptr<ModFilterWidget::Filter> m_filter;
|
||||||
|
|
||||||
ModPlatform::ListModel* listModel = nullptr;
|
ModPlatform::ListModel* listModel = nullptr;
|
||||||
ModPlatform::IndexedPack current;
|
ModPlatform::IndexedPack current;
|
||||||
|
|
||||||
|
@ -11,7 +11,7 @@
|
|||||||
</rect>
|
</rect>
|
||||||
</property>
|
</property>
|
||||||
<layout class="QGridLayout" name="gridLayout">
|
<layout class="QGridLayout" name="gridLayout">
|
||||||
<item row="1" column="0" colspan="2">
|
<item row="1" column="0" colspan="4">
|
||||||
<layout class="QGridLayout" name="gridLayout_3">
|
<layout class="QGridLayout" name="gridLayout_3">
|
||||||
<item row="1" column="2">
|
<item row="1" column="2">
|
||||||
<widget class="QTextBrowser" name="packDescription">
|
<widget class="QTextBrowser" name="packDescription">
|
||||||
@ -41,7 +41,7 @@
|
|||||||
</item>
|
</item>
|
||||||
</layout>
|
</layout>
|
||||||
</item>
|
</item>
|
||||||
<item row="0" column="1">
|
<item row="0" column="3">
|
||||||
<widget class="QPushButton" name="searchButton">
|
<widget class="QPushButton" name="searchButton">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Search</string>
|
<string>Search</string>
|
||||||
@ -51,12 +51,12 @@
|
|||||||
<item row="0" column="0">
|
<item row="0" column="0">
|
||||||
<widget class="QLineEdit" name="searchEdit">
|
<widget class="QLineEdit" name="searchEdit">
|
||||||
<property name="placeholderText">
|
<property name="placeholderText">
|
||||||
<string>Search and filter...</string>
|
<string>Search for mods...</string>
|
||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item row="2" column="0" colspan="2">
|
<item row="2" column="0" colspan="4">
|
||||||
<layout class="QGridLayout" name="gridLayout_4" columnstretch="0,0,0,0">
|
<layout class="QGridLayout" name="gridLayout_4" columnstretch="0,0,0">
|
||||||
<item row="0" column="2">
|
<item row="0" column="2">
|
||||||
<widget class="QComboBox" name="versionSelectionBox"/>
|
<widget class="QComboBox" name="versionSelectionBox"/>
|
||||||
</item>
|
</item>
|
||||||
@ -82,6 +82,20 @@
|
|||||||
</item>
|
</item>
|
||||||
</layout>
|
</layout>
|
||||||
</item>
|
</item>
|
||||||
|
<item row="0" column="1">
|
||||||
|
<widget class="QPushButton" name="modFilterButton">
|
||||||
|
<property name="text">
|
||||||
|
<string>Filter options</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item row="0" column="2">
|
||||||
|
<widget class="Line" name="line">
|
||||||
|
<property name="orientation">
|
||||||
|
<enum>Qt::Vertical</enum>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
</layout>
|
</layout>
|
||||||
</widget>
|
</widget>
|
||||||
<tabstops>
|
<tabstops>
|
||||||
|
109
launcher/ui/widgets/ModFilterWidget.cpp
Normal file
109
launcher/ui/widgets/ModFilterWidget.cpp
Normal file
@ -0,0 +1,109 @@
|
|||||||
|
#include "ModFilterWidget.h"
|
||||||
|
#include "ui_ModFilterWidget.h"
|
||||||
|
|
||||||
|
ModFilterWidget::ModFilterWidget(Version def, QWidget* parent)
|
||||||
|
: QTabWidget(parent), m_filter(new Filter()), ui(new Ui::ModFilterWidget)
|
||||||
|
{
|
||||||
|
ui->setupUi(this);
|
||||||
|
|
||||||
|
m_mcVersion_buttons.addButton(ui->strictVersionButton, VersionButtonID::Strict);
|
||||||
|
ui->strictVersionButton->click();
|
||||||
|
m_mcVersion_buttons.addButton(ui->majorVersionButton, VersionButtonID::Major);
|
||||||
|
m_mcVersion_buttons.addButton(ui->allVersionsButton, VersionButtonID::All);
|
||||||
|
//m_mcVersion_buttons.addButton(ui->betweenVersionsButton, VersionButtonID::Between);
|
||||||
|
|
||||||
|
connect(&m_mcVersion_buttons, SIGNAL(idClicked(int)), this, SLOT(onVersionFilterChanged(int)));
|
||||||
|
|
||||||
|
m_filter->versions.push_front(def);
|
||||||
|
|
||||||
|
setHidden(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ModFilterWidget::setInstance(MinecraftInstance* instance)
|
||||||
|
{
|
||||||
|
m_instance = instance;
|
||||||
|
|
||||||
|
auto mcVersionSplit = mcVersionStr().split(".");
|
||||||
|
|
||||||
|
ui->strictVersionButton->setText(
|
||||||
|
tr("Strict match (= %1)").arg(mcVersionStr()));
|
||||||
|
ui->majorVersionButton->setText(
|
||||||
|
tr("Major version match (= %1.%2.x)").arg(mcVersionSplit[0], mcVersionSplit[1]));
|
||||||
|
ui->allVersionsButton->setText(
|
||||||
|
tr("Any version"));
|
||||||
|
//ui->betweenVersionsButton->setText(
|
||||||
|
// tr("Between two versions"));
|
||||||
|
}
|
||||||
|
|
||||||
|
auto ModFilterWidget::getFilter() -> std::shared_ptr<Filter>
|
||||||
|
{
|
||||||
|
m_last_version_id = m_version_id;
|
||||||
|
emit filterUnchanged();
|
||||||
|
return m_filter;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ModFilterWidget::disableVersionButton(VersionButtonID id)
|
||||||
|
{
|
||||||
|
switch(id){
|
||||||
|
case(VersionButtonID::Strict):
|
||||||
|
ui->strictVersionButton->setEnabled(false);
|
||||||
|
break;
|
||||||
|
case(VersionButtonID::Major):
|
||||||
|
ui->majorVersionButton->setEnabled(false);
|
||||||
|
break;
|
||||||
|
case(VersionButtonID::All):
|
||||||
|
ui->allVersionsButton->setEnabled(false);
|
||||||
|
break;
|
||||||
|
case(VersionButtonID::Between):
|
||||||
|
// ui->betweenVersionsButton->setEnabled(false);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ModFilterWidget::onVersionFilterChanged(int id)
|
||||||
|
{
|
||||||
|
//ui->lowerVersionComboBox->setEnabled(id == VersionButtonID::Between);
|
||||||
|
//ui->upperVersionComboBox->setEnabled(id == VersionButtonID::Between);
|
||||||
|
|
||||||
|
auto versionSplit = mcVersionStr().split(".");
|
||||||
|
int index = 0;
|
||||||
|
|
||||||
|
auto cast_id = (VersionButtonID) id;
|
||||||
|
if (cast_id != m_version_id) {
|
||||||
|
m_version_id = cast_id;
|
||||||
|
} else {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_filter->versions.clear();
|
||||||
|
|
||||||
|
switch(cast_id){
|
||||||
|
case(VersionButtonID::Strict):
|
||||||
|
m_filter->versions.push_front(mcVersion());
|
||||||
|
break;
|
||||||
|
case(VersionButtonID::Major):
|
||||||
|
for(auto i = Version(QString("%1.%2").arg(versionSplit[0], versionSplit[1])); i <= mcVersion(); index++){
|
||||||
|
m_filter->versions.push_front(i);
|
||||||
|
i = Version(QString("%1.%2.%3").arg(versionSplit[0], versionSplit[1], QString("%1").arg(index)));
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case(VersionButtonID::All):
|
||||||
|
// Empty list to avoid enumerating all versions :P
|
||||||
|
break;
|
||||||
|
case(VersionButtonID::Between):
|
||||||
|
// TODO
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(changed())
|
||||||
|
emit filterChanged();
|
||||||
|
else
|
||||||
|
emit filterUnchanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
ModFilterWidget::~ModFilterWidget()
|
||||||
|
{
|
||||||
|
delete ui;
|
||||||
|
}
|
69
launcher/ui/widgets/ModFilterWidget.h
Normal file
69
launcher/ui/widgets/ModFilterWidget.h
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <QTabWidget>
|
||||||
|
#include <QButtonGroup>
|
||||||
|
|
||||||
|
#include "Version.h"
|
||||||
|
#include "minecraft/MinecraftInstance.h"
|
||||||
|
#include "minecraft/PackProfile.h"
|
||||||
|
|
||||||
|
class MinecraftInstance;
|
||||||
|
|
||||||
|
namespace Ui {
|
||||||
|
class ModFilterWidget;
|
||||||
|
}
|
||||||
|
|
||||||
|
class ModFilterWidget : public QTabWidget
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
enum VersionButtonID {
|
||||||
|
Strict = 0,
|
||||||
|
Major = 1,
|
||||||
|
All = 2,
|
||||||
|
Between = 3
|
||||||
|
};
|
||||||
|
|
||||||
|
struct Filter {
|
||||||
|
std::list<Version> versions;
|
||||||
|
|
||||||
|
bool operator==(const Filter& other) const { return versions == other.versions; }
|
||||||
|
bool operator!=(const Filter& other) const { return !(*this == other); }
|
||||||
|
};
|
||||||
|
|
||||||
|
std::shared_ptr<Filter> m_filter;
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit ModFilterWidget(Version def, QWidget* parent = nullptr);
|
||||||
|
~ModFilterWidget();
|
||||||
|
|
||||||
|
void setInstance(MinecraftInstance* instance);
|
||||||
|
|
||||||
|
/// By default all buttons are enabled
|
||||||
|
void disableVersionButton(VersionButtonID);
|
||||||
|
|
||||||
|
auto getFilter() -> std::shared_ptr<Filter>;
|
||||||
|
auto changed() const -> bool { return m_last_version_id != m_version_id; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
inline auto mcVersionStr() const -> QString { return m_instance ? m_instance->getPackProfile()->getComponentVersion("net.minecraft") : ""; }
|
||||||
|
inline auto mcVersion() const -> Version { return { mcVersionStr() }; }
|
||||||
|
|
||||||
|
private slots:
|
||||||
|
void onVersionFilterChanged(int id);
|
||||||
|
|
||||||
|
public: signals:
|
||||||
|
void filterChanged();
|
||||||
|
void filterUnchanged();
|
||||||
|
|
||||||
|
private:
|
||||||
|
Ui::ModFilterWidget* ui;
|
||||||
|
|
||||||
|
MinecraftInstance* m_instance = nullptr;
|
||||||
|
|
||||||
|
QButtonGroup m_mcVersion_buttons;
|
||||||
|
|
||||||
|
/* Used to tell if the filter was changed since the last getFilter() call */
|
||||||
|
VersionButtonID m_last_version_id = VersionButtonID::Strict;
|
||||||
|
VersionButtonID m_version_id = VersionButtonID::Strict;
|
||||||
|
};
|
54
launcher/ui/widgets/ModFilterWidget.ui
Normal file
54
launcher/ui/widgets/ModFilterWidget.ui
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<ui version="4.0">
|
||||||
|
<class>ModFilterWidget</class>
|
||||||
|
<widget class="QTabWidget" name="ModFilterWidget">
|
||||||
|
<property name="geometry">
|
||||||
|
<rect>
|
||||||
|
<x>0</x>
|
||||||
|
<y>0</y>
|
||||||
|
<width>400</width>
|
||||||
|
<height>300</height>
|
||||||
|
</rect>
|
||||||
|
</property>
|
||||||
|
<property name="sizePolicy">
|
||||||
|
<sizepolicy hsizetype="Expanding" vsizetype="Minimum">
|
||||||
|
<horstretch>0</horstretch>
|
||||||
|
<verstretch>0</verstretch>
|
||||||
|
</sizepolicy>
|
||||||
|
</property>
|
||||||
|
<widget class="QWidget" name="VersionPage">
|
||||||
|
<attribute name="title">
|
||||||
|
<string>Minecraft versions</string>
|
||||||
|
</attribute>
|
||||||
|
<layout class="QGridLayout" name="gridLayout">
|
||||||
|
<item row="0" column="0">
|
||||||
|
<layout class="QFormLayout" name="formLayout">
|
||||||
|
<item row="2" column="0">
|
||||||
|
<widget class="QRadioButton" name="allVersionsButton">
|
||||||
|
<property name="text">
|
||||||
|
<string>allVersions</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item row="0" column="0">
|
||||||
|
<widget class="QRadioButton" name="strictVersionButton">
|
||||||
|
<property name="text">
|
||||||
|
<string>strictVersion</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item row="1" column="0">
|
||||||
|
<widget class="QRadioButton" name="majorVersionButton">
|
||||||
|
<property name="text">
|
||||||
|
<string>majorVersion</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
</layout>
|
||||||
|
</item>
|
||||||
|
</layout>
|
||||||
|
</widget>
|
||||||
|
</widget>
|
||||||
|
<resources/>
|
||||||
|
<connections/>
|
||||||
|
</ui>
|
Loading…
x
Reference in New Issue
Block a user