/* Copyright 2015-2021 MultiMC Contributors * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "VersionList.h" #include <QDateTime> #include "Version.h" #include "JsonFormat.h" #include "Version.h" namespace Meta { VersionList::VersionList(const QString &uid, QObject *parent) : BaseVersionList(parent), m_uid(uid) { setObjectName("Version list: " + uid); } Task::Ptr VersionList::getLoadTask() { load(Net::Mode::Online); return getCurrentTask(); } bool VersionList::isLoaded() { return BaseEntity::isLoaded(); } const BaseVersion::Ptr VersionList::at(int i) const { return m_versions.at(i); } int VersionList::count() const { return m_versions.size(); } void VersionList::sortVersions() { beginResetModel(); std::sort(m_versions.begin(), m_versions.end(), [](const Version::Ptr &a, const Version::Ptr &b) { return *a.get() < *b.get(); }); endResetModel(); } QVariant VersionList::data(const QModelIndex &index, int role) const { if (!index.isValid() || index.row() < 0 || index.row() >= m_versions.size() || index.parent().isValid()) { return QVariant(); } Version::Ptr version = m_versions.at(index.row()); switch (role) { case VersionPointerRole: return QVariant::fromValue(std::dynamic_pointer_cast<BaseVersion>(version)); case VersionRole: case VersionIdRole: return version->version(); case ParentVersionRole: { // FIXME: HACK: this should be generic and be replaced by something else. Anything that is a hard 'equals' dep is a 'parent uid'. auto & reqs = version->requiredSet(); auto iter = std::find_if(reqs.begin(), reqs.end(), [](const Require & req) { return req.uid == "net.minecraft"; }); if (iter != reqs.end()) { return (*iter).equalsVersion; } return QVariant(); } case TypeRole: return version->type(); case UidRole: return version->uid(); case TimeRole: return version->time(); case RequiresRole: return QVariant::fromValue(version->requiredSet()); case SortRole: return version->rawTime(); case VersionPtrRole: return QVariant::fromValue(version); case RecommendedRole: return version->isRecommended(); // FIXME: this should be determined in whatever view/proxy is used... // case LatestRole: return version == getLatestStable(); default: return QVariant(); } } BaseVersionList::RoleList VersionList::providesRoles() const { return {VersionPointerRole, VersionRole, VersionIdRole, ParentVersionRole, TypeRole, UidRole, TimeRole, RequiresRole, SortRole, RecommendedRole, LatestRole, VersionPtrRole}; } QHash<int, QByteArray> VersionList::roleNames() const { QHash<int, QByteArray> roles = BaseVersionList::roleNames(); roles.insert(UidRole, "uid"); roles.insert(TimeRole, "time"); roles.insert(SortRole, "sort"); roles.insert(RequiresRole, "requires"); return roles; } QString VersionList::localFilename() const { return m_uid + "/index.json"; } QString VersionList::humanReadable() const { return m_name.isEmpty() ? m_uid : m_name; } Version::Ptr VersionList::getVersion(const QString &version) { Version::Ptr out = m_lookup.value(version, nullptr); if(!out) { out = std::make_shared<Version>(m_uid, version); m_lookup[version] = out; } return out; } bool VersionList::hasVersion(QString version) const { auto ver = std::find_if(m_versions.constBegin(), m_versions.constEnd(), [&](Meta::Version::Ptr const& a){ return a->version() == version; }); return (ver != m_versions.constEnd()); } void VersionList::setName(const QString &name) { m_name = name; emit nameChanged(name); } void VersionList::setVersions(const QVector<Version::Ptr> &versions) { beginResetModel(); m_versions = versions; std::sort(m_versions.begin(), m_versions.end(), [](const Version::Ptr &a, const Version::Ptr &b) { return a->rawTime() > b->rawTime(); }); for (int i = 0; i < m_versions.size(); ++i) { m_lookup.insert(m_versions.at(i)->version(), m_versions.at(i)); setupAddedVersion(i, m_versions.at(i)); } // FIXME: this is dumb, we have 'recommended' as part of the metadata already... auto recommendedIt = std::find_if(m_versions.constBegin(), m_versions.constEnd(), [](const Version::Ptr &ptr) { return ptr->type() == "release"; }); m_recommended = recommendedIt == m_versions.constEnd() ? nullptr : *recommendedIt; endResetModel(); } void VersionList::parse(const QJsonObject& obj) { parseVersionList(obj, this); } // FIXME: this is dumb, we have 'recommended' as part of the metadata already... static const Meta::Version::Ptr &getBetterVersion(const Meta::Version::Ptr &a, const Meta::Version::Ptr &b) { if(!a) return b; if(!b) return a; if(a->type() == b->type()) { // newer of same type wins return (a->rawTime() > b->rawTime() ? a : b); } // 'release' type wins return (a->type() == "release" ? a : b); } void VersionList::mergeFromIndex(const VersionList::Ptr &other) { if (m_name != other->m_name) { setName(other->m_name); } } void VersionList::merge(const VersionList::Ptr &other) { if (m_name != other->m_name) { setName(other->m_name); } // TODO: do not reset the whole model. maybe? beginResetModel(); m_versions.clear(); if(other->m_versions.isEmpty()) { qWarning() << "Empty list loaded ..."; } for (const Version::Ptr &version : other->m_versions) { // we already have the version. merge the contents if (m_lookup.contains(version->version())) { m_lookup.value(version->version())->mergeFromList(version); } else { m_lookup.insert(version->uid(), version); } // connect it. setupAddedVersion(m_versions.size(), version); m_versions.append(version); m_recommended = getBetterVersion(m_recommended, version); } endResetModel(); } void VersionList::setupAddedVersion(const int row, const Version::Ptr &version) { // FIXME: do not disconnect from everythin, disconnect only the lambdas here version->disconnect(); connect(version.get(), &Version::requiresChanged, this, [this, row]() { emit dataChanged(index(row), index(row), QVector<int>() << RequiresRole); }); connect(version.get(), &Version::timeChanged, this, [this, row]() { emit dataChanged(index(row), index(row), QVector<int>() << TimeRole << SortRole); }); connect(version.get(), &Version::typeChanged, this, [this, row]() { emit dataChanged(index(row), index(row), QVector<int>() << TypeRole); }); } BaseVersion::Ptr VersionList::getRecommended() const { return m_recommended; } }