PrismLauncher/launcher/InstanceList.h

205 lines
7.0 KiB
C
Raw Permalink Normal View History

// SPDX-License-Identifier: GPL-3.0-only
/*
* Prism Launcher - Minecraft Launcher
* Copyright (C) 2023 TheKodeToad <TheKodeToad@proton.me>
2013-01-14 23:42:38 +00: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.
*
* 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 23:42:38 +00: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 23:42:38 +00:00
*/
#pragma once
2013-01-14 23:42:38 +00:00
#include <QAbstractListModel>
2016-10-02 23:55:54 +01:00
#include <QList>
#include <QObject>
#include <QPair>
#include <QSet>
#include <QStack>
2013-02-15 04:40:00 +00:00
2015-02-09 00:51:14 +00:00
#include "BaseInstance.h"
2013-02-15 04:40:00 +00:00
2018-07-24 00:24:06 +01:00
class QFileSystemWatcher;
class InstanceTask;
struct InstanceName;
2018-07-24 00:24:06 +01:00
using InstanceId = QString;
using GroupId = QString;
using InstanceLocator = std::pair<InstancePtr, int>;
enum class InstCreateError { NoCreateError = 0, NoSuchVersion, UnknownCreateError, InstExists, CantCreateDir };
2018-07-24 00:24:06 +01:00
enum class GroupsState { NotLoaded, Steady, Dirty };
struct TrashHistoryItem {
QString id;
QString polyPath;
QString trashPath;
QString groupName;
};
2018-07-24 00:24:06 +01:00
class InstanceList : public QAbstractListModel {
2018-07-15 13:51:05 +01:00
Q_OBJECT
public:
explicit InstanceList(SettingsObjectPtr settings, const QString& instDir, QObject* parent = 0);
2018-07-15 13:51:05 +01:00
virtual ~InstanceList();
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 00:50:32 +00:00
bool setData(const QModelIndex& index, const QVariant& value, int role) override;
2018-07-15 13:51:05 +01:00
enum AdditionalRoles {
2018-07-15 13:51:05 +01:00
GroupRole = Qt::UserRole,
InstancePointerRole = 0x34B1CB48, ///< Return pointer to real instance
InstanceIDRole = 0x34B1CB49 ///< Return id if the instance
2018-07-15 13:51:05 +01:00
};
/*!
* \brief Error codes returned by functions in the InstanceList class.
* NoError Indicates that no error occurred.
* UnknownError indicates that an unspecified error occurred.
*/
enum InstListError { NoError = 0, UnknownError };
2018-07-15 13:51:05 +01:00
InstancePtr at(int i) const { return m_instances.at(i); }
2018-07-15 13:51:05 +01:00
int count() const { return m_instances.count(); }
2018-07-15 13:51:05 +01:00
2018-07-23 23:11:24 +01:00
InstListError loadList();
2018-07-15 13:51:05 +01:00
void saveNow();
/* O(n) */
2018-07-15 13:51:05 +01:00
InstancePtr getInstanceById(QString id) const;
/* O(n) */
InstancePtr getInstanceByManagedName(const QString& managed_name) const;
QModelIndex getInstanceIndexById(const QString& id) const;
2018-07-15 13:51:05 +01:00
QStringList getGroups();
bool isGroupCollapsed(const QString& groupName);
GroupId getInstanceGroup(const InstanceId& id) const;
void setInstanceGroup(const InstanceId& id, GroupId name);
2018-07-15 13:51:05 +01:00
void deleteGroup(const GroupId& name);
void renameGroup(const GroupId& src, const GroupId& dst);
bool trashInstance(const InstanceId& id);
bool trashedSomething();
void undoTrashInstance();
void deleteInstance(const InstanceId& id);
2018-07-24 00:24:06 +01:00
// Wrap an instance creation task in some more task machinery and make it ready to be used
Task* wrapInstanceTask(InstanceTask* task);
2018-07-24 00:24:06 +01: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.
* should_override is used when another similar instance already exists, and we want to override it
* - for instance, when updating it.
2018-07-24 00:24:06 +01:00
*/
bool commitStagedInstance(const QString& keyPath, const InstanceName& instanceName, QString groupName, const InstanceTask&);
2018-07-24 00:24:06 +01:00
/**
* Destroy a previously created staging area given by @keyPath - used when creation fails.
* Used by instance manipulation tasks.
*/
bool destroyStagingPath(const QString& keyPath);
2018-07-24 00:24:06 +01:00
2019-07-10 03:23:24 +01:00
int getTotalPlayTime();
Qt::DropActions supportedDragActions() const override;
Qt::DropActions supportedDropActions() const override;
bool canDropMimeData(const QMimeData* data, Qt::DropAction action, int row, int column, const QModelIndex& parent) const override;
bool dropMimeData(const QMimeData* data, Qt::DropAction action, int row, int column, const QModelIndex& parent) override;
QStringList mimeTypes() const override;
QMimeData* mimeData(const QModelIndexList& indexes) const override;
QStringList getLinkedInstancesById(const QString& id) const;
signals:
2018-07-15 13:51:05 +01:00
void dataIsInvalid();
2018-07-24 00:24:06 +01:00
void instancesChanged();
void instanceSelectRequest(QString instanceId);
2018-07-24 00:24:06 +01:00
void groupsChanged(QSet<QString> groups);
public slots:
void on_InstFolderChanged(const Setting& setting, QVariant value);
void on_GroupStateChanged(const QString& group, bool collapsed);
private slots:
void propertiesChanged(BaseInstance* inst);
2018-07-15 13:51:05 +01:00
void providerUpdated();
void instanceDirContentsChanged(const QString& path);
private:
int getInstIndex(BaseInstance* inst) const;
2019-07-10 03:23:24 +01:00
void updateTotalPlayTime();
2018-07-15 13:51:05 +01:00
void suspendWatch();
void resumeWatch();
void add(const QList<InstancePtr>& list);
2018-07-24 00:24:06 +01:00
void loadGroupList();
void saveGroupList();
QList<InstanceId> discoverInstances();
InstancePtr loadInstance(const InstanceId& id);
void increaseGroupCount(const QString& group);
void decreaseGroupCount(const QString& group);
private:
2018-07-15 13:51:05 +01:00
int m_watchLevel = 0;
2019-07-10 03:23:24 +01:00
int totalPlayTime = 0;
2018-07-23 23:11:24 +01:00
bool m_dirty = false;
2018-07-15 13:51:05 +01:00
QList<InstancePtr> m_instances;
// id -> refs
QMap<QString, int> m_groupNameCache;
2018-07-24 00:24:06 +01:00
SettingsObjectPtr m_globalSettings;
QString m_instDir;
QFileSystemWatcher* m_watcher;
// FIXME: this is so inefficient that looking at it is almost painful.
QSet<QString> m_collapsedGroups;
QMap<InstanceId, GroupId> m_instanceGroupIndex;
2018-07-24 00:24:06 +01:00
QSet<InstanceId> instanceSet;
bool m_groupsLoaded = false;
bool m_instancesProbed = false;
QStack<TrashHistoryItem> m_trashHistory;
2013-01-14 23:42:38 +00:00
};