2023-09-08 01:43:01 +02:00
|
|
|
/*
|
|
|
|
* modsettings.cpp, part of VCMI engine
|
|
|
|
*
|
|
|
|
* Authors: listed in file AUTHORS in main folder
|
|
|
|
*
|
|
|
|
* License: GNU General Public License v2.0 or later
|
|
|
|
* Full text of license available in license.txt file, in main folder
|
|
|
|
*
|
|
|
|
*/
|
2023-09-05 03:26:38 +04:00
|
|
|
#include "StdInc.h"
|
|
|
|
#include "modsettings.h"
|
|
|
|
#include "ui_modsettings.h"
|
|
|
|
#include "../mapcontroller.h"
|
2024-11-09 20:29:07 +00:00
|
|
|
#include "../../lib/modding/ModDescription.h"
|
2023-09-05 03:26:38 +04:00
|
|
|
#include "../../lib/modding/CModHandler.h"
|
|
|
|
#include "../../lib/mapping/CMapService.h"
|
|
|
|
|
|
|
|
void traverseNode(QTreeWidgetItem * item, std::function<void(QTreeWidgetItem*)> action)
|
|
|
|
{
|
|
|
|
// Do something with item
|
|
|
|
action(item);
|
|
|
|
for (int i = 0; i < item->childCount(); ++i)
|
|
|
|
traverseNode(item->child(i), action);
|
|
|
|
}
|
|
|
|
|
|
|
|
ModSettings::ModSettings(QWidget *parent) :
|
|
|
|
AbstractSettings(parent),
|
|
|
|
ui(new Ui::ModSettings)
|
|
|
|
{
|
|
|
|
ui->setupUi(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
ModSettings::~ModSettings()
|
|
|
|
{
|
|
|
|
delete ui;
|
|
|
|
}
|
|
|
|
|
2023-09-11 19:16:24 +02:00
|
|
|
void ModSettings::initialize(MapController & c)
|
2023-09-05 03:26:38 +04:00
|
|
|
{
|
2023-09-11 19:16:24 +02:00
|
|
|
AbstractSettings::initialize(c);
|
2023-09-05 03:26:38 +04:00
|
|
|
|
|
|
|
//mods management
|
|
|
|
//collect all active mods
|
|
|
|
QMap<QString, QTreeWidgetItem*> addedMods;
|
|
|
|
QSet<QString> modsToProcess;
|
|
|
|
ui->treeMods->blockSignals(true);
|
|
|
|
|
2024-11-09 20:29:07 +00:00
|
|
|
auto createModTreeWidgetItem = [&](QTreeWidgetItem * parent, const ModDescription & modInfo)
|
2023-09-05 03:26:38 +04:00
|
|
|
{
|
2024-11-09 20:29:07 +00:00
|
|
|
auto item = new QTreeWidgetItem(parent, {QString::fromStdString(modInfo.getName()), QString::fromStdString(modInfo.getVersion().toString())});
|
|
|
|
item->setData(0, Qt::UserRole, QVariant(QString::fromStdString(modInfo.getID())));
|
2023-09-05 03:26:38 +04:00
|
|
|
item->setFlags(item->flags() | Qt::ItemIsUserCheckable);
|
2024-11-09 20:29:07 +00:00
|
|
|
item->setCheckState(0, controller->map()->mods.count(modInfo.getID()) ? Qt::Checked : Qt::Unchecked);
|
2023-09-05 03:26:38 +04:00
|
|
|
//set parent check
|
|
|
|
if(parent && item->checkState(0) == Qt::Checked)
|
|
|
|
parent->setCheckState(0, Qt::Checked);
|
|
|
|
return item;
|
|
|
|
};
|
|
|
|
|
|
|
|
for(const auto & modName : VLC->modh->getActiveMods())
|
|
|
|
{
|
|
|
|
QString qmodName = QString::fromStdString(modName);
|
|
|
|
if(qmodName.split(".").size() == 1)
|
|
|
|
{
|
|
|
|
const auto & modInfo = VLC->modh->getModInfo(modName);
|
|
|
|
addedMods[qmodName] = createModTreeWidgetItem(nullptr, modInfo);
|
|
|
|
ui->treeMods->addTopLevelItem(addedMods[qmodName]);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
modsToProcess.insert(qmodName);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for(auto qmodIter = modsToProcess.begin(); qmodIter != modsToProcess.end();)
|
|
|
|
{
|
|
|
|
auto qmodName = *qmodIter;
|
|
|
|
auto pieces = qmodName.split(".");
|
|
|
|
assert(pieces.size() > 1);
|
|
|
|
|
2024-02-03 10:36:47 +01:00
|
|
|
pieces.pop_back();
|
|
|
|
auto qs = pieces.join(".");
|
2023-09-05 03:26:38 +04:00
|
|
|
|
|
|
|
if(addedMods.count(qs))
|
|
|
|
{
|
|
|
|
const auto & modInfo = VLC->modh->getModInfo(qmodName.toStdString());
|
|
|
|
addedMods[qmodName] = createModTreeWidgetItem(addedMods[qs], modInfo);
|
|
|
|
modsToProcess.erase(qmodIter);
|
|
|
|
qmodIter = modsToProcess.begin();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
++qmodIter;
|
|
|
|
}
|
|
|
|
|
|
|
|
ui->treeMods->blockSignals(false);
|
|
|
|
}
|
|
|
|
|
2023-09-11 19:16:24 +02:00
|
|
|
void ModSettings::update()
|
2023-09-05 03:26:38 +04:00
|
|
|
{
|
|
|
|
//Mod management
|
|
|
|
auto widgetAction = [&](QTreeWidgetItem * item)
|
|
|
|
{
|
|
|
|
if(item->checkState(0) == Qt::Checked)
|
|
|
|
{
|
|
|
|
auto modName = item->data(0, Qt::UserRole).toString().toStdString();
|
2023-09-21 04:31:08 +02:00
|
|
|
controller->map()->mods[modName] = VLC->modh->getModInfo(modName).getVerificationInfo();
|
2023-09-05 03:26:38 +04:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2023-09-11 19:16:24 +02:00
|
|
|
controller->map()->mods.clear();
|
2023-09-05 03:26:38 +04:00
|
|
|
for (int i = 0; i < ui->treeMods->topLevelItemCount(); ++i)
|
|
|
|
{
|
|
|
|
QTreeWidgetItem *item = ui->treeMods->topLevelItem(i);
|
|
|
|
traverseNode(item, widgetAction);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ModSettings::updateModWidgetBasedOnMods(const ModCompatibilityInfo & mods)
|
|
|
|
{
|
|
|
|
//Mod management
|
|
|
|
auto widgetAction = [&](QTreeWidgetItem * item)
|
|
|
|
{
|
|
|
|
auto modName = item->data(0, Qt::UserRole).toString().toStdString();
|
|
|
|
item->setCheckState(0, mods.count(modName) ? Qt::Checked : Qt::Unchecked);
|
|
|
|
};
|
|
|
|
|
|
|
|
for (int i = 0; i < ui->treeMods->topLevelItemCount(); ++i)
|
|
|
|
{
|
|
|
|
QTreeWidgetItem *item = ui->treeMods->topLevelItem(i);
|
|
|
|
traverseNode(item, widgetAction);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ModSettings::on_modResolution_map_clicked()
|
|
|
|
{
|
2023-09-11 19:16:24 +02:00
|
|
|
updateModWidgetBasedOnMods(MapController::modAssessmentMap(*controller->map()));
|
2023-09-05 03:26:38 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ModSettings::on_modResolution_full_clicked()
|
|
|
|
{
|
|
|
|
updateModWidgetBasedOnMods(MapController::modAssessmentAll());
|
|
|
|
}
|
|
|
|
|
|
|
|
void ModSettings::on_treeMods_itemChanged(QTreeWidgetItem *item, int column)
|
|
|
|
{
|
|
|
|
//set state for children
|
|
|
|
for (int i = 0; i < item->childCount(); ++i)
|
|
|
|
item->child(i)->setCheckState(0, item->checkState(0));
|
|
|
|
|
|
|
|
//set state for parent
|
|
|
|
ui->treeMods->blockSignals(true);
|
|
|
|
if(item->checkState(0) == Qt::Checked)
|
|
|
|
{
|
|
|
|
while(item->parent())
|
|
|
|
{
|
|
|
|
item->parent()->setCheckState(0, Qt::Checked);
|
|
|
|
item = item->parent();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ui->treeMods->blockSignals(false);
|
|
|
|
}
|