/* Copyright 2015 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 "WonkoVersionList.h" #include #include "WonkoVersion.h" #include "tasks/BaseWonkoEntityRemoteLoadTask.h" #include "tasks/BaseWonkoEntityLocalLoadTask.h" #include "format/WonkoFormat.h" #include "WonkoReference.h" class WVLLoadTask : public Task { Q_OBJECT public: explicit WVLLoadTask(WonkoVersionList *list, QObject *parent = nullptr) : Task(parent), m_list(list) { } bool canAbort() const override { return !m_currentTask || m_currentTask->canAbort(); } bool abort() override { return m_currentTask->abort(); } private: void executeTask() override { if (!m_list->isLocalLoaded()) { m_currentTask = m_list->localUpdateTask(); connect(m_currentTask.get(), &Task::succeeded, this, &WVLLoadTask::next); } else { m_currentTask = m_list->remoteUpdateTask(); connect(m_currentTask.get(), &Task::succeeded, this, &WVLLoadTask::emitSucceeded); } connect(m_currentTask.get(), &Task::status, this, &WVLLoadTask::setStatus); connect(m_currentTask.get(), &Task::progress, this, &WVLLoadTask::setProgress); connect(m_currentTask.get(), &Task::failed, this, &WVLLoadTask::emitFailed); m_currentTask->start(); } void next() { m_currentTask = m_list->remoteUpdateTask(); connect(m_currentTask.get(), &Task::status, this, &WVLLoadTask::setStatus); connect(m_currentTask.get(), &Task::progress, this, &WVLLoadTask::setProgress); connect(m_currentTask.get(), &Task::succeeded, this, &WVLLoadTask::emitSucceeded); m_currentTask->start(); } WonkoVersionList *m_list; std::unique_ptr m_currentTask; }; WonkoVersionList::WonkoVersionList(const QString &uid, QObject *parent) : BaseVersionList(parent), m_uid(uid) { setObjectName("Wonko version list: " + uid); } Task *WonkoVersionList::getLoadTask() { return new WVLLoadTask(this); } bool WonkoVersionList::isLoaded() { return isLocalLoaded() && isRemoteLoaded(); } const BaseVersionPtr WonkoVersionList::at(int i) const { return m_versions.at(i); } int WonkoVersionList::count() const { return m_versions.size(); } void WonkoVersionList::sortVersions() { beginResetModel(); std::sort(m_versions.begin(), m_versions.end(), [](const WonkoVersionPtr &a, const WonkoVersionPtr &b) { return *a.get() < *b.get(); }); endResetModel(); } QVariant WonkoVersionList::data(const QModelIndex &index, int role) const { if (!index.isValid() || index.row() < 0 || index.row() >= m_versions.size() || index.parent().isValid()) { return QVariant(); } WonkoVersionPtr version = m_versions.at(index.row()); switch (role) { case VersionPointerRole: return QVariant::fromValue(std::dynamic_pointer_cast(version)); case VersionRole: case VersionIdRole: return version->version(); case ParentGameVersionRole: { const auto end = version->requires().end(); const auto it = std::find_if(version->requires().begin(), end, [](const WonkoReference &ref) { return ref.uid() == "net.minecraft"; }); if (it != end) { return (*it).version(); } return QVariant(); } case TypeRole: return version->type(); case UidRole: return version->uid(); case TimeRole: return version->time(); case RequiresRole: return QVariant::fromValue(version->requires()); case SortRole: return version->rawTime(); case WonkoVersionPtrRole: return QVariant::fromValue(version); case RecommendedRole: return version == getRecommended(); case LatestRole: return version == getLatestStable(); default: return QVariant(); } } BaseVersionList::RoleList WonkoVersionList::providesRoles() const { return {VersionPointerRole, VersionRole, VersionIdRole, ParentGameVersionRole, TypeRole, UidRole, TimeRole, RequiresRole, SortRole, RecommendedRole, LatestRole, WonkoVersionPtrRole}; } QHash WonkoVersionList::roleNames() const { QHash roles = BaseVersionList::roleNames(); roles.insert(UidRole, "uid"); roles.insert(TimeRole, "time"); roles.insert(SortRole, "sort"); roles.insert(RequiresRole, "requires"); return roles; } std::unique_ptr WonkoVersionList::remoteUpdateTask() { return std::unique_ptr(new WonkoVersionListRemoteLoadTask(this, this)); } std::unique_ptr WonkoVersionList::localUpdateTask() { return std::unique_ptr(new WonkoVersionListLocalLoadTask(this, this)); } QString WonkoVersionList::localFilename() const { return m_uid + ".json"; } QJsonObject WonkoVersionList::serialized() const { return WonkoFormat::serializeVersionList(this); } QString WonkoVersionList::humanReadable() const { return m_name.isEmpty() ? m_uid : m_name; } bool WonkoVersionList::hasVersion(const QString &version) const { return m_lookup.contains(version); } WonkoVersionPtr WonkoVersionList::getVersion(const QString &version) const { return m_lookup.value(version); } void WonkoVersionList::setName(const QString &name) { m_name = name; emit nameChanged(name); } void WonkoVersionList::setVersions(const QVector &versions) { beginResetModel(); m_versions = versions; std::sort(m_versions.begin(), m_versions.end(), [](const WonkoVersionPtr &a, const WonkoVersionPtr &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)); } m_latest = m_versions.isEmpty() ? nullptr : m_versions.first(); auto recommendedIt = std::find_if(m_versions.constBegin(), m_versions.constEnd(), [](const WonkoVersionPtr &ptr) { return ptr->type() == "release"; }); m_recommended = recommendedIt == m_versions.constEnd() ? nullptr : *recommendedIt; endResetModel(); } void WonkoVersionList::merge(const BaseWonkoEntity::Ptr &other) { const WonkoVersionListPtr list = std::dynamic_pointer_cast(other); if (m_name != list->m_name) { setName(list->m_name); } if (m_versions.isEmpty()) { setVersions(list->m_versions); } else { for (const WonkoVersionPtr &version : list->m_versions) { if (m_lookup.contains(version->version())) { m_lookup.value(version->version())->merge(version); } else { beginInsertRows(QModelIndex(), m_versions.size(), m_versions.size()); setupAddedVersion(m_versions.size(), version); m_versions.append(version); m_lookup.insert(version->uid(), version); endInsertRows(); if (!m_latest || version->rawTime() > m_latest->rawTime()) { m_latest = version; emit dataChanged(index(0), index(m_versions.size() - 1), QVector() << LatestRole); } if (!m_recommended || (version->type() == "release" && version->rawTime() > m_recommended->rawTime())) { m_recommended = version; emit dataChanged(index(0), index(m_versions.size() - 1), QVector() << RecommendedRole); } } } } } void WonkoVersionList::setupAddedVersion(const int row, const WonkoVersionPtr &version) { connect(version.get(), &WonkoVersion::requiresChanged, this, [this, row]() { emit dataChanged(index(row), index(row), QVector() << RequiresRole); }); connect(version.get(), &WonkoVersion::timeChanged, this, [this, row]() { emit dataChanged(index(row), index(row), QVector() << TimeRole << SortRole); }); connect(version.get(), &WonkoVersion::typeChanged, this, [this, row]() { emit dataChanged(index(row), index(row), QVector() << TypeRole); }); } BaseVersionPtr WonkoVersionList::getLatestStable() const { return m_latest; } BaseVersionPtr WonkoVersionList::getRecommended() const { return m_recommended; } #include "WonkoVersionList.moc"