2023-07-16 23:23:33 +01:00
|
|
|
// SPDX-License-Identifier: GPL-3.0-only
|
|
|
|
/*
|
|
|
|
* Prism Launcher - Minecraft Launcher
|
|
|
|
* Copyright (C) 2023 TheKodeToad <TheKodeToad@proton.me>
|
2013-01-14 17:42:38 -06:00
|
|
|
*
|
2023-07-16 23:23:33 +01:00
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, version 3.
|
2013-10-28 20:55:12 +01:00
|
|
|
*
|
2023-07-16 23:23:33 +01:00
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
2013-01-14 17:42:38 -06:00
|
|
|
*
|
2023-07-16 23:23:33 +01:00
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* This file incorporates work covered by the following copyright and
|
|
|
|
* permission notice:
|
|
|
|
*
|
|
|
|
* Copyright 2013-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.
|
2013-01-14 17:42:38 -06:00
|
|
|
*/
|
|
|
|
|
2013-07-29 00:59:35 +02:00
|
|
|
#pragma once
|
2013-01-14 17:42:38 -06:00
|
|
|
|
2013-08-26 06:30:11 +02:00
|
|
|
#include <QAbstractListModel>
|
2016-10-03 00:55:54 +02:00
|
|
|
#include <QList>
|
2023-08-02 18:35:35 +02:00
|
|
|
#include <QObject>
|
2022-07-31 01:42:33 +08:00
|
|
|
#include <QPair>
|
2023-08-02 18:35:35 +02:00
|
|
|
#include <QSet>
|
|
|
|
#include <QStack>
|
2013-02-14 22:40:00 -06:00
|
|
|
|
2015-02-09 01:51:14 +01:00
|
|
|
#include "BaseInstance.h"
|
2013-02-14 22:40:00 -06:00
|
|
|
|
2018-07-24 01:24:06 +02:00
|
|
|
class QFileSystemWatcher;
|
|
|
|
class InstanceTask;
|
2022-07-14 16:13:23 -03:00
|
|
|
struct InstanceName;
|
|
|
|
|
2018-07-24 01:24:06 +02:00
|
|
|
using InstanceId = QString;
|
|
|
|
using GroupId = QString;
|
|
|
|
using InstanceLocator = std::pair<InstancePtr, int>;
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
enum class InstCreateError { NoCreateError = 0, NoSuchVersion, UnknownCreateError, InstExists, CantCreateDir };
|
2018-07-24 01:24:06 +02:00
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
enum class GroupsState { NotLoaded, Steady, Dirty };
|
2018-07-31 01:54:08 +02:00
|
|
|
|
2022-07-31 01:42:33 +08:00
|
|
|
struct TrashHistoryItem {
|
|
|
|
QString id;
|
|
|
|
QString polyPath;
|
|
|
|
QString trashPath;
|
|
|
|
QString groupName;
|
|
|
|
};
|
2018-07-24 01:24:06 +02:00
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
class InstanceList : public QAbstractListModel {
|
2013-02-14 22:40:00 -06:00
|
|
|
Q_OBJECT
|
2013-10-28 20:55:12 +01:00
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
public:
|
|
|
|
explicit InstanceList(SettingsObjectPtr settings, const QString& instDir, QObject* parent = 0);
|
2013-08-25 22:48:41 +02:00
|
|
|
virtual ~InstanceList();
|
2013-10-28 20:55:12 +01:00
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
public:
|
|
|
|
QModelIndex index(int row, int column = 0, const QModelIndex& parent = QModelIndex()) const override;
|
|
|
|
int rowCount(const QModelIndex& parent = QModelIndex()) const override;
|
|
|
|
QVariant data(const QModelIndex& index, int role) const override;
|
|
|
|
Qt::ItemFlags flags(const QModelIndex& index) const override;
|
2018-11-22 01:50:32 +01:00
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
bool setData(const QModelIndex& index, const QVariant& value, int role) override;
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
enum AdditionalRoles {
|
2015-02-01 18:54:58 +01:00
|
|
|
GroupRole = Qt::UserRole,
|
2023-08-02 18:35:35 +02:00
|
|
|
InstancePointerRole = 0x34B1CB48, ///< Return pointer to real instance
|
|
|
|
InstanceIDRole = 0x34B1CB49 ///< Return id if the instance
|
2013-08-26 06:30:11 +02:00
|
|
|
};
|
2013-02-19 12:15:22 -06:00
|
|
|
/*!
|
|
|
|
* \brief Error codes returned by functions in the InstanceList class.
|
|
|
|
* NoError Indicates that no error occurred.
|
|
|
|
* UnknownError indicates that an unspecified error occurred.
|
|
|
|
*/
|
2023-08-02 18:35:35 +02:00
|
|
|
enum InstListError { NoError = 0, UnknownError };
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
InstancePtr at(int i) const { return m_instances.at(i); }
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
int count() const { return m_instances.count(); }
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2018-07-24 00:11:24 +02:00
|
|
|
InstListError loadList();
|
2017-12-03 18:36:28 +01:00
|
|
|
void saveNow();
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2022-07-08 13:00:44 -03:00
|
|
|
/* O(n) */
|
2013-11-27 15:00:44 +01:00
|
|
|
InstancePtr getInstanceById(QString id) const;
|
2022-07-08 13:00:44 -03:00
|
|
|
/* O(n) */
|
2022-09-26 11:50:31 +02:00
|
|
|
InstancePtr getInstanceByManagedName(const QString& managed_name) const;
|
2023-08-02 18:35:35 +02:00
|
|
|
QModelIndex getInstanceIndexById(const QString& id) const;
|
2013-12-15 18:10:51 +01:00
|
|
|
QStringList getGroups();
|
2023-08-02 18:35:35 +02:00
|
|
|
bool isGroupCollapsed(const QString& groupName);
|
2019-08-20 02:58:27 +02:00
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
GroupId getInstanceGroup(const InstanceId& id) const;
|
2023-08-29 13:27:06 +01:00
|
|
|
void setInstanceGroup(const InstanceId& id, GroupId name);
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
void deleteGroup(const GroupId& name);
|
2023-07-16 23:23:33 +01:00
|
|
|
void renameGroup(const GroupId& src, const GroupId& dst);
|
2023-08-02 18:35:35 +02:00
|
|
|
bool trashInstance(const InstanceId& id);
|
2022-07-31 01:42:33 +08:00
|
|
|
bool trashedSomething();
|
|
|
|
void undoTrashInstance();
|
2023-08-02 18:35:35 +02:00
|
|
|
void deleteInstance(const InstanceId& id);
|
2016-04-11 01:30:50 +02:00
|
|
|
|
2018-07-24 01:24:06 +02:00
|
|
|
// Wrap an instance creation task in some more task machinery and make it ready to be used
|
2023-08-02 18:35:35 +02:00
|
|
|
Task* wrapInstanceTask(InstanceTask* task);
|
2018-07-24 01:24:06 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a new empty staging area for instance creation and @return a path/key top commit it later.
|
|
|
|
* Used by instance manipulation tasks.
|
|
|
|
*/
|
|
|
|
QString getStagedInstancePath();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Commit the staging area given by @keyPath to the provider - used when creation succeeds.
|
|
|
|
* Used by instance manipulation tasks.
|
2022-07-08 13:00:44 -03:00
|
|
|
* should_override is used when another similar instance already exists, and we want to override it
|
|
|
|
* - for instance, when updating it.
|
2018-07-24 01:24:06 +02:00
|
|
|
*/
|
2022-12-03 10:15:38 -03:00
|
|
|
bool commitStagedInstance(const QString& keyPath, const InstanceName& instanceName, const QString& groupName, const InstanceTask&);
|
2018-07-24 01:24:06 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Destroy a previously created staging area given by @keyPath - used when creation fails.
|
|
|
|
* Used by instance manipulation tasks.
|
|
|
|
*/
|
2023-08-02 18:35:35 +02:00
|
|
|
bool destroyStagingPath(const QString& keyPath);
|
2018-07-24 01:24:06 +02:00
|
|
|
|
2019-07-09 22:23:24 -04:00
|
|
|
int getTotalPlayTime();
|
|
|
|
|
2021-10-25 23:51:42 +02:00
|
|
|
Qt::DropActions supportedDragActions() const override;
|
|
|
|
|
|
|
|
Qt::DropActions supportedDropActions() const override;
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
bool canDropMimeData(const QMimeData* data, Qt::DropAction action, int row, int column, const QModelIndex& parent) const override;
|
2021-10-25 23:51:42 +02:00
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
bool dropMimeData(const QMimeData* data, Qt::DropAction action, int row, int column, const QModelIndex& parent) override;
|
2021-10-25 23:51:42 +02:00
|
|
|
|
|
|
|
QStringList mimeTypes() const override;
|
2023-08-02 18:35:35 +02:00
|
|
|
QMimeData* mimeData(const QModelIndexList& indexes) const override;
|
2021-10-25 23:51:42 +02:00
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
QStringList getLinkedInstancesById(const QString& id) const;
|
2023-02-15 22:01:27 -07:00
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
signals:
|
2013-08-26 06:30:11 +02:00
|
|
|
void dataIsInvalid();
|
2018-07-24 01:24:06 +02:00
|
|
|
void instancesChanged();
|
2019-06-23 14:54:17 +02:00
|
|
|
void instanceSelectRequest(QString instanceId);
|
2018-07-24 01:24:06 +02:00
|
|
|
void groupsChanged(QSet<QString> groups);
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
public slots:
|
|
|
|
void on_InstFolderChanged(const Setting& setting, QVariant value);
|
|
|
|
void on_GroupStateChanged(const QString& group, bool collapsed);
|
2013-10-28 20:55:12 +01:00
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
private slots:
|
|
|
|
void propertiesChanged(BaseInstance* inst);
|
2016-10-03 00:55:54 +02:00
|
|
|
void providerUpdated();
|
2023-08-02 18:35:35 +02:00
|
|
|
void instanceDirContentsChanged(const QString& path);
|
2013-10-28 20:55:12 +01:00
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
private:
|
|
|
|
int getInstIndex(BaseInstance* inst) const;
|
2019-07-09 22:23:24 -04:00
|
|
|
void updateTotalPlayTime();
|
2016-10-03 00:55:54 +02:00
|
|
|
void suspendWatch();
|
|
|
|
void resumeWatch();
|
2023-08-02 18:35:35 +02:00
|
|
|
void add(const QList<InstancePtr>& list);
|
2018-07-24 01:24:06 +02:00
|
|
|
void loadGroupList();
|
|
|
|
void saveGroupList();
|
|
|
|
QList<InstanceId> discoverInstances();
|
|
|
|
InstancePtr loadInstance(const InstanceId& id);
|
2013-12-20 14:47:26 +01:00
|
|
|
|
2023-07-16 23:36:01 +01:00
|
|
|
void increaseGroupCount(const QString& group);
|
|
|
|
void decreaseGroupCount(const QString& group);
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
private:
|
2016-10-03 00:55:54 +02:00
|
|
|
int m_watchLevel = 0;
|
2019-07-09 22:23:24 -04:00
|
|
|
int totalPlayTime = 0;
|
2018-07-24 00:11:24 +02:00
|
|
|
bool m_dirty = false;
|
2013-10-28 20:55:12 +01:00
|
|
|
QList<InstancePtr> m_instances;
|
2023-07-16 23:23:33 +01:00
|
|
|
// id -> refs
|
|
|
|
QMap<QString, int> m_groupNameCache;
|
2018-07-24 01:24:06 +02:00
|
|
|
|
|
|
|
SettingsObjectPtr m_globalSettings;
|
|
|
|
QString m_instDir;
|
2023-08-02 18:35:35 +02:00
|
|
|
QFileSystemWatcher* m_watcher;
|
2019-08-20 02:58:27 +02:00
|
|
|
// FIXME: this is so inefficient that looking at it is almost painful.
|
|
|
|
QSet<QString> m_collapsedGroups;
|
|
|
|
QMap<InstanceId, GroupId> m_instanceGroupIndex;
|
2018-07-24 01:24:06 +02:00
|
|
|
QSet<InstanceId> instanceSet;
|
|
|
|
bool m_groupsLoaded = false;
|
|
|
|
bool m_instancesProbed = false;
|
2022-07-31 01:42:33 +08:00
|
|
|
|
|
|
|
QStack<TrashHistoryItem> m_trashHistory;
|
2013-01-14 17:42:38 -06:00
|
|
|
};
|