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"
|
|
|
|
|
2013-09-21 21:29:26 +03:00
|
|
|
#include "../../lib/JsonNode.h"
|
|
|
|
#include "../../lib/filesystem/CFileInputStream.h"
|
|
|
|
|
2013-08-22 17:22:49 +03:00
|
|
|
bool CModEntry::compareVersions(QString lesser, QString greater)
|
|
|
|
{
|
|
|
|
static const int maxSections = 3; // versions consist from up to 3 sections, major.minor.patch
|
|
|
|
|
|
|
|
QStringList lesserList = lesser.split(".");
|
|
|
|
QStringList greaterList = greater.split(".");
|
|
|
|
|
|
|
|
assert(lesserList.size() <= maxSections);
|
|
|
|
assert(greaterList.size() <= maxSections);
|
|
|
|
|
|
|
|
for (int i=0; i< maxSections; i++)
|
|
|
|
{
|
|
|
|
if (greaterList.size() <= i) // 1.1.1 > 1.1
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (lesserList.size() <= i) // 1.1 < 1.1.1
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (lesserList[i].toInt() != greaterList[i].toInt())
|
|
|
|
return lesserList[i].toInt() < greaterList[i].toInt(); // 1.1 < 1.2
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-01-30 15:35:17 +03:00
|
|
|
QString CModEntry::sizeToString(double size)
|
|
|
|
{
|
|
|
|
static const QString sizes[] =
|
|
|
|
{
|
|
|
|
/*"%1 B", */"%1 KiB", "%1 MiB", "%1 GiB", "%1 TiB"
|
|
|
|
};
|
|
|
|
size_t index = 0;
|
|
|
|
while (size > 1024 && index < 4)
|
|
|
|
{
|
|
|
|
size /= 1024;
|
|
|
|
index++;
|
|
|
|
}
|
|
|
|
return sizes[index].arg(QString::number(size, 'f', 1));
|
|
|
|
}
|
|
|
|
|
2013-11-03 15:07:23 +03:00
|
|
|
CModEntry::CModEntry(QVariantMap repository, QVariantMap localData, QVariantMap modSettings, QString modname):
|
2013-08-22 17:22:49 +03:00
|
|
|
repository(repository),
|
|
|
|
localData(localData),
|
|
|
|
modSettings(modSettings),
|
|
|
|
modname(modname)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CModEntry::isEnabled() const
|
|
|
|
{
|
|
|
|
if (!isInstalled())
|
|
|
|
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
|
|
|
|
{
|
|
|
|
if (!isInstalled())
|
|
|
|
return false;
|
|
|
|
return !isEnabled();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CModEntry::isAvailable() const
|
|
|
|
{
|
|
|
|
if (isInstalled())
|
|
|
|
return false;
|
|
|
|
return !repository.isEmpty();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CModEntry::isUpdateable() const
|
|
|
|
{
|
|
|
|
if (!isInstalled())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
QString installedVer = localData["installedVersion"].toString();
|
|
|
|
QString availableVer = repository["latestVersion"].toString();
|
|
|
|
|
|
|
|
if (compareVersions(installedVer, availableVer))
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CModEntry::isInstalled() const
|
|
|
|
{
|
|
|
|
return !localData.isEmpty();
|
|
|
|
}
|
|
|
|
|
|
|
|
int CModEntry::getModStatus() const
|
|
|
|
{
|
|
|
|
return
|
|
|
|
(isEnabled() ? ModStatus::ENABLED : 0) |
|
|
|
|
(isInstalled() ? ModStatus::INSTALLED : 0) |
|
|
|
|
(isUpdateable()? ModStatus::UPDATEABLE : 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
QString CModEntry::getName() const
|
|
|
|
{
|
|
|
|
return modname;
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant CModEntry::getValue(QString value) const
|
|
|
|
{
|
2014-03-20 20:06:25 +03:00
|
|
|
if (repository.contains(value) && localData.contains(value))
|
|
|
|
{
|
|
|
|
// value is present in both repo and locally installed. Select one from latest version
|
|
|
|
QString installedVer = localData["installedVersion"].toString();
|
|
|
|
QString availableVer = repository["latestVersion"].toString();
|
|
|
|
|
|
|
|
if (compareVersions(installedVer, availableVer))
|
|
|
|
return repository[value];
|
|
|
|
else
|
|
|
|
return localData[value];
|
|
|
|
}
|
|
|
|
|
2013-08-22 17:22:49 +03:00
|
|
|
if (repository.contains(value))
|
2013-09-21 21:29:26 +03:00
|
|
|
return repository[value];
|
2013-08-22 17:22:49 +03:00
|
|
|
|
|
|
|
if (localData.contains(value))
|
2013-09-21 21:29:26 +03:00
|
|
|
return localData[value];
|
2013-08-22 17:22:49 +03:00
|
|
|
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
|
2013-09-21 21:29:26 +03:00
|
|
|
QVariantMap CModList::copyField(QVariantMap data, QString from, QString to)
|
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
|
|
|
|
|
|
|
for (auto it = data.begin(); it != data.end(); it++)
|
|
|
|
{
|
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;
|
|
|
|
}
|
|
|
|
|
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
|
|
|
{
|
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
|
|
|
}
|
|
|
|
|
|
|
|
static QVariant getValue(QVariantMap input, QString path)
|
|
|
|
{
|
|
|
|
if (path.size() > 1)
|
|
|
|
{
|
|
|
|
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);
|
|
|
|
return getValue(input.value(entryName).toMap(), remainder);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return input;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CModEntry CModList::getMod(QString modname) const
|
|
|
|
{
|
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);
|
|
|
|
|
2014-01-30 17:51:15 +03:00
|
|
|
if (conf.isNull())
|
|
|
|
{
|
|
|
|
settings["active"] = true; // default
|
|
|
|
}
|
2013-11-03 15:07:23 +03:00
|
|
|
else
|
2014-01-30 17:51:15 +03:00
|
|
|
{
|
|
|
|
if (conf.canConvert<QVariantMap>())
|
2014-03-20 20:06:25 +03:00
|
|
|
settings = conf.toMap();
|
2014-01-30 17:51:15 +03:00
|
|
|
else
|
|
|
|
settings.insert("active", conf);
|
|
|
|
}
|
2013-08-22 17:22:49 +03:00
|
|
|
|
2013-08-24 23:11:51 +03: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);
|
|
|
|
if (repoVal.isValid())
|
2013-08-22 17:22:49 +03:00
|
|
|
{
|
|
|
|
if (repo.empty())
|
2014-03-20 20:06:25 +03:00
|
|
|
repo = repoVal.toMap();
|
2013-08-22 17:22:49 +03:00
|
|
|
else
|
|
|
|
{
|
2014-03-20 20:06:25 +03:00
|
|
|
if (CModEntry::compareVersions(repo["version"].toString(), repoVal.toMap()["version"].toString()))
|
|
|
|
repo = repoVal.toMap();
|
2013-08-22 17:22:49 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return CModEntry(repo, local, settings, modname);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CModList::hasMod(QString modname) const
|
|
|
|
{
|
|
|
|
if (localModList.contains(modname))
|
|
|
|
return true;
|
|
|
|
|
2013-08-24 23:11:51 +03:00
|
|
|
for (auto entry : repositories)
|
2013-08-22 17:22:49 +03:00
|
|
|
if (entry.contains(modname))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList CModList::getRequirements(QString modname)
|
|
|
|
{
|
|
|
|
QStringList ret;
|
|
|
|
|
|
|
|
if (hasMod(modname))
|
|
|
|
{
|
|
|
|
auto mod = getMod(modname);
|
|
|
|
|
|
|
|
for (auto entry : mod.getValue("depends").toStringList())
|
|
|
|
ret += getRequirements(entry);
|
|
|
|
}
|
|
|
|
ret += modname;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
QVector<QString> CModList::getModList() const
|
|
|
|
{
|
|
|
|
QSet<QString> knownMods;
|
|
|
|
QVector<QString> modList;
|
2013-08-24 23:11:51 +03:00
|
|
|
for (auto repo : repositories)
|
2013-08-22 17:22:49 +03:00
|
|
|
{
|
|
|
|
for (auto it = repo.begin(); it != repo.end(); it++)
|
|
|
|
{
|
|
|
|
knownMods.insert(it.key());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (auto it = localModList.begin(); it != localModList.end(); it++)
|
|
|
|
{
|
|
|
|
knownMods.insert(it.key());
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto entry : knownMods)
|
|
|
|
{
|
|
|
|
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;
|
|
|
|
for (const QString & mod : getModList())
|
|
|
|
{
|
|
|
|
if (mod.count('.') == depth && mod.startsWith(parent))
|
|
|
|
children.push_back(mod);
|
|
|
|
}
|
|
|
|
return children;
|
|
|
|
}
|