2017-07-13 10:26:03 +02:00
|
|
|
/*
|
|
|
|
* cmodlist.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
|
|
|
|
*
|
|
|
|
*/
|
2013-08-22 17:22:49 +03:00
|
|
|
#include "StdInc.h"
|
|
|
|
#include "cmodlist.h"
|
|
|
|
|
2022-12-25 13:19:16 +02:00
|
|
|
#include "../lib/CConfigHandler.h"
|
2013-09-21 21:29:26 +03:00
|
|
|
#include "../../lib/JsonNode.h"
|
|
|
|
#include "../../lib/filesystem/CFileInputStream.h"
|
2022-09-17 15:43:59 +02:00
|
|
|
#include "../../lib/GameConstants.h"
|
2023-09-01 02:12:41 +02:00
|
|
|
#include "../../lib/modding/CModVersion.h"
|
2013-08-22 17:22:49 +03:00
|
|
|
|
2014-01-30 15:35:17 +03:00
|
|
|
QString CModEntry::sizeToString(double size)
|
|
|
|
{
|
2023-09-01 22:18:23 +02:00
|
|
|
static const std::array<QString, 5> sizes { "%1 B", "%1 KiB", "%1 MiB", "%1 GiB", "%1 TiB" };
|
2014-01-30 15:35:17 +03:00
|
|
|
size_t index = 0;
|
2023-09-01 22:18:23 +02:00
|
|
|
while(size > 1024 && index < sizes.size())
|
2014-01-30 15:35:17 +03:00
|
|
|
{
|
|
|
|
size /= 1024;
|
|
|
|
index++;
|
|
|
|
}
|
|
|
|
return sizes[index].arg(QString::number(size, 'f', 1));
|
|
|
|
}
|
|
|
|
|
2018-04-13 07:34:58 +02:00
|
|
|
CModEntry::CModEntry(QVariantMap repository, QVariantMap localData, QVariantMap modSettings, QString modname)
|
|
|
|
: repository(repository), localData(localData), modSettings(modSettings), modname(modname)
|
2013-08-22 17:22:49 +03:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CModEntry::isEnabled() const
|
|
|
|
{
|
2018-04-13 07:34:58 +02:00
|
|
|
if(!isInstalled())
|
2013-08-22 17:22:49 +03:00
|
|
|
return false;
|
|
|
|
|
2013-11-03 15:07:23 +03:00
|
|
|
return modSettings["active"].toBool();
|
2013-08-22 17:22:49 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CModEntry::isDisabled() const
|
|
|
|
{
|
2018-04-13 07:34:58 +02:00
|
|
|
if(!isInstalled())
|
2013-08-22 17:22:49 +03:00
|
|
|
return false;
|
|
|
|
return !isEnabled();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CModEntry::isAvailable() const
|
|
|
|
{
|
2018-04-13 07:34:58 +02:00
|
|
|
if(isInstalled())
|
2013-08-22 17:22:49 +03:00
|
|
|
return false;
|
|
|
|
return !repository.isEmpty();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CModEntry::isUpdateable() const
|
|
|
|
{
|
2018-04-13 07:34:58 +02:00
|
|
|
if(!isInstalled())
|
2013-08-22 17:22:49 +03:00
|
|
|
return false;
|
|
|
|
|
2023-09-01 02:12:41 +02:00
|
|
|
auto installedVer = localData["installedVersion"].toString().toStdString();
|
|
|
|
auto availableVer = repository["latestVersion"].toString().toStdString();
|
2013-08-22 17:22:49 +03:00
|
|
|
|
2023-09-01 02:12:41 +02:00
|
|
|
return (CModVersion::fromString(installedVer) < CModVersion::fromString(availableVer));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isCompatible(const QVariantMap & compatibility)
|
|
|
|
{
|
|
|
|
auto compatibleMin = CModVersion::fromString(compatibility["min"].toString().toStdString());
|
|
|
|
auto compatibleMax = CModVersion::fromString(compatibility["max"].toString().toStdString());
|
|
|
|
|
|
|
|
return (compatibleMin.isNull() || CModVersion::GameVersion().compatible(compatibleMin, true, true))
|
|
|
|
&& (compatibleMax.isNull() || compatibleMax.compatible(CModVersion::GameVersion(), true, true));
|
2013-08-22 17:22:49 +03:00
|
|
|
}
|
|
|
|
|
2022-09-17 15:43:59 +02:00
|
|
|
bool CModEntry::isCompatible() const
|
|
|
|
{
|
2023-09-01 02:12:41 +02:00
|
|
|
return ::isCompatible(localData["compatibility"].toMap());
|
2022-09-17 15:43:59 +02:00
|
|
|
}
|
|
|
|
|
2022-09-10 18:30:41 +02:00
|
|
|
bool CModEntry::isEssential() const
|
|
|
|
{
|
2023-01-09 23:38:04 +02:00
|
|
|
return getName() == "vcmi";
|
2022-09-10 18:30:41 +02:00
|
|
|
}
|
|
|
|
|
2013-08-22 17:22:49 +03:00
|
|
|
bool CModEntry::isInstalled() const
|
|
|
|
{
|
|
|
|
return !localData.isEmpty();
|
|
|
|
}
|
|
|
|
|
2023-10-21 22:55:20 +02:00
|
|
|
bool CModEntry::isVisible() const
|
2022-09-17 15:43:59 +02:00
|
|
|
{
|
2023-10-21 23:07:21 +02:00
|
|
|
if (getBaseValue("modType").toString() == "Compatibility")
|
|
|
|
{
|
|
|
|
if (isSubmod())
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (getBaseValue("modType").toString() == "Translation")
|
|
|
|
{
|
|
|
|
// Do not show not installed translation mods to languages other than player language
|
|
|
|
if (localData.empty() && getBaseValue("language") != QString::fromStdString(settings["general"]["language"].String()) )
|
|
|
|
return false;
|
|
|
|
}
|
2023-10-21 22:55:20 +02:00
|
|
|
|
2023-09-19 17:19:00 +02:00
|
|
|
return !localData.isEmpty() || (!repository.isEmpty() && !repository.contains("mod"));
|
2022-09-17 15:43:59 +02:00
|
|
|
}
|
|
|
|
|
2023-03-14 15:59:33 +02:00
|
|
|
bool CModEntry::isTranslation() const
|
|
|
|
{
|
2023-10-21 22:55:20 +02:00
|
|
|
return getBaseValue("modType").toString() == "Translation";
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CModEntry::isSubmod() const
|
|
|
|
{
|
|
|
|
return getName().contains('.');
|
2023-03-14 15:59:33 +02:00
|
|
|
}
|
|
|
|
|
2013-08-22 17:22:49 +03:00
|
|
|
int CModEntry::getModStatus() const
|
|
|
|
{
|
2018-04-13 07:34:58 +02:00
|
|
|
int status = 0;
|
|
|
|
if(isEnabled())
|
|
|
|
status |= ModStatus::ENABLED;
|
|
|
|
if(isInstalled())
|
|
|
|
status |= ModStatus::INSTALLED;
|
|
|
|
if(isUpdateable())
|
|
|
|
status |= ModStatus::UPDATEABLE;
|
|
|
|
|
|
|
|
return status;
|
2013-08-22 17:22:49 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
QString CModEntry::getName() const
|
|
|
|
{
|
|
|
|
return modname;
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant CModEntry::getValue(QString value) const
|
2023-03-14 13:37:22 +02:00
|
|
|
{
|
|
|
|
return getValueImpl(value, true);
|
|
|
|
}
|
|
|
|
|
2023-11-19 20:44:28 +02:00
|
|
|
QStringList CModEntry::getDependencies() const
|
|
|
|
{
|
|
|
|
QStringList result;
|
|
|
|
for (auto const & entry : getValue("depends").toStringList())
|
|
|
|
result.push_back(entry.toLower());
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList CModEntry::getConflicts() const
|
|
|
|
{
|
|
|
|
QStringList result;
|
|
|
|
for (auto const & entry : getValue("conflicts").toStringList())
|
|
|
|
result.push_back(entry.toLower());
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-03-14 13:37:22 +02:00
|
|
|
QVariant CModEntry::getBaseValue(QString value) const
|
|
|
|
{
|
|
|
|
return getValueImpl(value, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant CModEntry::getValueImpl(QString value, bool localized) const
|
|
|
|
|
2013-08-22 17:22:49 +03:00
|
|
|
{
|
2022-12-25 22:18:14 +02:00
|
|
|
QString langValue = QString::fromStdString(settings["general"]["language"].String());
|
2022-12-25 13:19:16 +02:00
|
|
|
|
|
|
|
// Priorities
|
|
|
|
// 1) data from newest version
|
|
|
|
// 2) data from preferred language
|
|
|
|
|
|
|
|
bool useRepositoryData = repository.contains(value);
|
|
|
|
|
2018-04-13 07:34:58 +02:00
|
|
|
if(repository.contains(value) && localData.contains(value))
|
2014-03-20 20:06:25 +03:00
|
|
|
{
|
|
|
|
// value is present in both repo and locally installed. Select one from latest version
|
2023-09-01 02:12:41 +02:00
|
|
|
auto installedVer = localData["installedVersion"].toString().toStdString();
|
|
|
|
auto availableVer = repository["latestVersion"].toString().toStdString();
|
2014-03-20 20:06:25 +03:00
|
|
|
|
2023-09-01 02:12:41 +02:00
|
|
|
useRepositoryData = CModVersion::fromString(installedVer) < CModVersion::fromString(availableVer);
|
2014-03-20 20:06:25 +03:00
|
|
|
}
|
|
|
|
|
2022-12-25 13:19:16 +02:00
|
|
|
auto & storage = useRepositoryData ? repository : localData;
|
|
|
|
|
2023-03-14 13:37:22 +02:00
|
|
|
if(localized && storage.contains(langValue))
|
2022-12-25 13:19:16 +02:00
|
|
|
{
|
|
|
|
auto langStorage = storage[langValue].toMap();
|
|
|
|
if (langStorage.contains(value))
|
|
|
|
return langStorage[value];
|
|
|
|
}
|
2013-08-22 17:22:49 +03:00
|
|
|
|
2022-12-28 22:22:05 +02:00
|
|
|
if(storage.contains(value))
|
2022-12-25 13:19:16 +02:00
|
|
|
return storage[value];
|
2013-08-22 17:22:49 +03:00
|
|
|
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
|
2023-10-28 18:44:58 +02:00
|
|
|
QVariantMap CModList::copyField(QVariantMap data, QString from, QString to) const
|
2013-08-22 17:22:49 +03:00
|
|
|
{
|
2013-09-21 21:29:26 +03:00
|
|
|
QVariantMap renamed;
|
2013-08-22 17:22:49 +03:00
|
|
|
|
2018-04-13 07:34:58 +02:00
|
|
|
for(auto it = data.begin(); it != data.end(); it++)
|
2013-08-22 17:22:49 +03:00
|
|
|
{
|
2013-10-01 18:14:05 +03:00
|
|
|
QVariantMap modConf = it.value().toMap();
|
2013-08-22 17:22:49 +03:00
|
|
|
|
2013-10-01 18:14:05 +03:00
|
|
|
modConf.insert(to, modConf.value(from));
|
|
|
|
renamed.insert(it.key(), modConf);
|
2013-08-22 17:22:49 +03:00
|
|
|
}
|
|
|
|
return renamed;
|
|
|
|
}
|
|
|
|
|
2022-09-10 18:30:41 +02:00
|
|
|
void CModList::reloadRepositories()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2013-08-24 23:11:51 +03:00
|
|
|
void CModList::resetRepositories()
|
|
|
|
{
|
|
|
|
repositories.clear();
|
|
|
|
}
|
|
|
|
|
2013-09-21 21:29:26 +03:00
|
|
|
void CModList::addRepository(QVariantMap data)
|
2013-08-22 17:22:49 +03:00
|
|
|
{
|
2022-09-25 13:36:46 +02:00
|
|
|
for(auto & key : data.keys())
|
|
|
|
data[key.toLower()] = data.take(key);
|
2013-08-24 23:11:51 +03:00
|
|
|
repositories.push_back(copyField(data, "version", "latestVersion"));
|
2013-08-22 17:22:49 +03:00
|
|
|
}
|
|
|
|
|
2013-09-21 21:29:26 +03:00
|
|
|
void CModList::setLocalModList(QVariantMap data)
|
2013-08-22 17:22:49 +03:00
|
|
|
{
|
|
|
|
localModList = copyField(data, "version", "installedVersion");
|
|
|
|
}
|
|
|
|
|
2013-09-21 21:29:26 +03:00
|
|
|
void CModList::setModSettings(QVariant data)
|
2013-08-22 17:22:49 +03:00
|
|
|
{
|
2013-09-21 21:29:26 +03:00
|
|
|
modSettings = data.toMap();
|
2013-08-22 17:22:49 +03:00
|
|
|
}
|
|
|
|
|
2014-03-20 20:06:25 +03:00
|
|
|
void CModList::modChanged(QString modID)
|
2013-08-22 17:22:49 +03:00
|
|
|
{
|
2014-03-20 20:06:25 +03:00
|
|
|
}
|
|
|
|
|
2022-09-10 18:30:41 +02:00
|
|
|
static QVariant getValue(QVariant input, QString path)
|
2014-03-20 20:06:25 +03:00
|
|
|
{
|
2018-04-13 07:34:58 +02:00
|
|
|
if(path.size() > 1)
|
2014-03-20 20:06:25 +03:00
|
|
|
{
|
|
|
|
QString entryName = path.section('/', 0, 1);
|
|
|
|
QString remainder = "/" + path.section('/', 2, -1);
|
2013-08-22 17:22:49 +03:00
|
|
|
|
2014-03-20 20:06:25 +03:00
|
|
|
entryName.remove(0, 1);
|
2022-09-25 13:36:46 +02:00
|
|
|
return getValue(input.toMap().value(entryName), remainder);
|
2014-03-20 20:06:25 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return input;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CModEntry CModList::getMod(QString modname) const
|
|
|
|
{
|
2022-09-17 15:43:59 +02:00
|
|
|
modname = modname.toLower();
|
2013-09-21 21:29:26 +03:00
|
|
|
QVariantMap repo;
|
|
|
|
QVariantMap local = localModList[modname].toMap();
|
2013-11-03 15:07:23 +03:00
|
|
|
QVariantMap settings;
|
|
|
|
|
2014-03-20 20:06:25 +03:00
|
|
|
QString path = modname;
|
|
|
|
path = "/" + path.replace(".", "/mods/");
|
|
|
|
QVariant conf = getValue(modSettings, path);
|
|
|
|
|
2018-04-13 07:34:58 +02:00
|
|
|
if(conf.isNull())
|
2014-01-30 17:51:15 +03:00
|
|
|
{
|
2023-07-22 21:45:39 +02:00
|
|
|
settings["active"] = !local.value("keepDisabled").toBool();
|
2014-01-30 17:51:15 +03:00
|
|
|
}
|
2013-11-03 15:07:23 +03:00
|
|
|
else
|
2014-01-30 17:51:15 +03:00
|
|
|
{
|
2022-09-10 18:30:41 +02:00
|
|
|
if(!conf.toMap().isEmpty())
|
|
|
|
{
|
2014-03-20 20:06:25 +03:00
|
|
|
settings = conf.toMap();
|
2022-09-10 18:30:41 +02:00
|
|
|
if(settings.value("active").isNull())
|
2023-07-22 21:45:39 +02:00
|
|
|
settings["active"] = !local.value("keepDisabled").toBool();
|
2022-09-10 18:30:41 +02:00
|
|
|
}
|
2014-01-30 17:51:15 +03:00
|
|
|
else
|
|
|
|
settings.insert("active", conf);
|
|
|
|
}
|
2022-09-10 18:30:41 +02:00
|
|
|
|
|
|
|
if(settings["active"].toBool())
|
|
|
|
{
|
|
|
|
QString rootPath = path.section('/', 0, 1);
|
|
|
|
if(path != rootPath)
|
|
|
|
{
|
|
|
|
conf = getValue(modSettings, rootPath);
|
|
|
|
const auto confMap = conf.toMap();
|
|
|
|
if(!conf.isNull() && !confMap["active"].isNull() && !confMap["active"].toBool())
|
|
|
|
{
|
|
|
|
settings = confMap;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-08-22 17:22:49 +03:00
|
|
|
|
2022-09-22 14:41:13 +02:00
|
|
|
if(settings.value("active").toBool())
|
|
|
|
{
|
2023-09-01 02:12:41 +02:00
|
|
|
if(!::isCompatible(local.value("compatibility").toMap()))
|
|
|
|
settings["active"] = false;
|
2022-09-22 14:41:13 +02:00
|
|
|
}
|
|
|
|
|
2018-04-13 07:34:58 +02:00
|
|
|
for(auto entry : repositories)
|
2013-08-22 17:22:49 +03:00
|
|
|
{
|
2014-03-20 20:06:25 +03:00
|
|
|
QVariant repoVal = getValue(entry, path);
|
2018-04-13 07:34:58 +02:00
|
|
|
if(repoVal.isValid())
|
2013-08-22 17:22:49 +03:00
|
|
|
{
|
2022-09-17 15:43:59 +02:00
|
|
|
auto repoValMap = repoVal.toMap();
|
2023-09-01 02:12:41 +02:00
|
|
|
if(::isCompatible(repoValMap["compatibility"].toMap()))
|
2013-08-22 17:22:49 +03:00
|
|
|
{
|
2023-09-01 02:12:41 +02:00
|
|
|
if(repo.empty()
|
|
|
|
|| CModVersion::fromString(repo["version"].toString().toStdString())
|
|
|
|
< CModVersion::fromString(repoValMap["version"].toString().toStdString()))
|
2022-09-17 15:43:59 +02:00
|
|
|
{
|
2023-09-02 12:10:50 +02:00
|
|
|
//take valid download link, screenshots and mod size before assignment
|
2022-09-29 15:38:03 +02:00
|
|
|
auto download = repo.value("download");
|
2022-10-21 02:03:54 +02:00
|
|
|
auto screenshots = repo.value("screenshots");
|
2023-09-07 19:57:57 +02:00
|
|
|
auto size = repo.value("downloadSize");
|
2022-09-17 15:43:59 +02:00
|
|
|
repo = repoValMap;
|
2022-09-29 15:38:03 +02:00
|
|
|
if(repo.value("download").isNull())
|
2022-10-21 02:03:54 +02:00
|
|
|
{
|
2022-09-29 15:38:03 +02:00
|
|
|
repo["download"] = download;
|
2022-10-21 02:03:54 +02:00
|
|
|
if(repo.value("screenshots").isNull()) //taking screenshot from the downloadable version
|
|
|
|
repo["screenshots"] = screenshots;
|
|
|
|
}
|
2023-09-07 19:57:57 +02:00
|
|
|
if(repo.value("downloadSize").isNull())
|
|
|
|
repo["downloadSize"] = size;
|
2022-09-17 15:43:59 +02:00
|
|
|
}
|
2013-08-22 17:22:49 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return CModEntry(repo, local, settings, modname);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CModList::hasMod(QString modname) const
|
|
|
|
{
|
2018-04-13 07:34:58 +02:00
|
|
|
if(localModList.contains(modname))
|
2013-08-22 17:22:49 +03:00
|
|
|
return true;
|
|
|
|
|
2018-04-13 07:34:58 +02:00
|
|
|
for(auto entry : repositories)
|
|
|
|
if(entry.contains(modname))
|
2013-08-22 17:22:49 +03:00
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList CModList::getRequirements(QString modname)
|
|
|
|
{
|
|
|
|
QStringList ret;
|
|
|
|
|
2018-04-13 07:34:58 +02:00
|
|
|
if(hasMod(modname))
|
2013-08-22 17:22:49 +03:00
|
|
|
{
|
|
|
|
auto mod = getMod(modname);
|
|
|
|
|
2023-11-19 20:44:28 +02:00
|
|
|
for(auto entry : mod.getDependencies())
|
|
|
|
ret += getRequirements(entry.toLower());
|
2013-08-22 17:22:49 +03:00
|
|
|
}
|
|
|
|
ret += modname;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
QVector<QString> CModList::getModList() const
|
|
|
|
{
|
|
|
|
QSet<QString> knownMods;
|
|
|
|
QVector<QString> modList;
|
2018-04-13 07:34:58 +02:00
|
|
|
for(auto repo : repositories)
|
2013-08-22 17:22:49 +03:00
|
|
|
{
|
2018-04-13 07:34:58 +02:00
|
|
|
for(auto it = repo.begin(); it != repo.end(); it++)
|
2013-08-22 17:22:49 +03:00
|
|
|
{
|
2022-09-17 15:43:59 +02:00
|
|
|
knownMods.insert(it.key().toLower());
|
2013-08-22 17:22:49 +03:00
|
|
|
}
|
|
|
|
}
|
2018-04-13 07:34:58 +02:00
|
|
|
for(auto it = localModList.begin(); it != localModList.end(); it++)
|
2013-08-22 17:22:49 +03:00
|
|
|
{
|
2022-09-17 15:43:59 +02:00
|
|
|
knownMods.insert(it.key().toLower());
|
2013-08-22 17:22:49 +03:00
|
|
|
}
|
|
|
|
|
2018-04-13 07:34:58 +02:00
|
|
|
for(auto entry : knownMods)
|
2013-08-22 17:22:49 +03:00
|
|
|
{
|
|
|
|
modList.push_back(entry);
|
|
|
|
}
|
|
|
|
return modList;
|
|
|
|
}
|
2014-03-23 15:08:01 +03:00
|
|
|
|
|
|
|
QVector<QString> CModList::getChildren(QString parent) const
|
|
|
|
{
|
|
|
|
QVector<QString> children;
|
|
|
|
|
|
|
|
int depth = parent.count('.') + 1;
|
2018-04-13 07:34:58 +02:00
|
|
|
for(const QString & mod : getModList())
|
2014-03-23 15:08:01 +03:00
|
|
|
{
|
2018-04-13 07:34:58 +02:00
|
|
|
if(mod.count('.') == depth && mod.startsWith(parent))
|
2014-03-23 15:08:01 +03:00
|
|
|
children.push_back(mod);
|
|
|
|
}
|
|
|
|
return children;
|
|
|
|
}
|