2022-03-19 12:46:56 +01:00
|
|
|
// SPDX-License-Identifier: GPL-3.0-only
|
|
|
|
/*
|
|
|
|
* PolyMC - Minecraft Launcher
|
|
|
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
2022-05-28 23:26:47 +01:00
|
|
|
* Copyright (c) 2022 Jamie Mansfield <jmansfield@cadixdev.org>
|
2013-02-14 22:40:00 -06:00
|
|
|
*
|
2022-03-19 12:46:56 +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-10 01:47:48 +02:00
|
|
|
*
|
2022-03-19 12:46:56 +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-02-14 22:40:00 -06:00
|
|
|
*
|
2022-03-19 12:46:56 +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-02-14 22:40:00 -06:00
|
|
|
*/
|
|
|
|
|
2013-07-29 00:59:35 +02:00
|
|
|
#pragma once
|
2016-10-03 00:55:54 +02:00
|
|
|
#include <cassert>
|
2013-02-14 22:40:00 -06:00
|
|
|
|
|
|
|
#include <QObject>
|
2016-08-14 02:33:31 +02:00
|
|
|
#include "QObjectPtr.h"
|
2013-02-14 22:40:00 -06:00
|
|
|
#include <QDateTime>
|
2014-03-10 17:38:27 +01:00
|
|
|
#include <QSet>
|
2015-07-21 02:38:15 +02:00
|
|
|
#include <QProcess>
|
2013-02-14 22:40:00 -06:00
|
|
|
|
2015-02-09 01:51:14 +01:00
|
|
|
#include "settings/SettingsObject.h"
|
2013-02-25 14:44:36 -06:00
|
|
|
|
2015-02-09 01:51:14 +01:00
|
|
|
#include "settings/INIFile.h"
|
|
|
|
#include "BaseVersionList.h"
|
2021-07-26 21:44:11 +02:00
|
|
|
#include "minecraft/auth/MinecraftAccount.h"
|
2016-11-03 01:10:16 +01:00
|
|
|
#include "MessageLevel.h"
|
2015-08-18 02:25:24 +02:00
|
|
|
#include "pathmatcher/IPathMatcher.h"
|
2013-02-20 19:10:09 -06:00
|
|
|
|
2017-11-11 01:38:31 +01:00
|
|
|
#include "net/Mode.h"
|
|
|
|
|
2021-05-22 16:14:25 +02:00
|
|
|
#include "minecraft/launch/MinecraftServerTarget.h"
|
2022-07-11 09:01:07 +02:00
|
|
|
#include "RuntimeContext.h"
|
2021-05-22 16:14:25 +02:00
|
|
|
|
2014-02-21 19:15:59 +01:00
|
|
|
class QDir;
|
2013-11-24 06:36:16 +01:00
|
|
|
class Task;
|
2015-07-10 01:11:06 +02:00
|
|
|
class LaunchTask;
|
2015-02-04 21:10:10 +01:00
|
|
|
class BaseInstance;
|
2013-02-14 22:40:00 -06:00
|
|
|
|
2014-09-06 18:16:56 +02:00
|
|
|
// pointer for lazy people
|
|
|
|
typedef std::shared_ptr<BaseInstance> InstancePtr;
|
|
|
|
|
2013-02-14 22:40:00 -06:00
|
|
|
/*!
|
|
|
|
* \brief Base class for instances.
|
2013-10-10 01:47:48 +02:00
|
|
|
* This class implements many functions that are common between instances and
|
2013-02-14 22:40:00 -06:00
|
|
|
* provides a standard interface for all instances.
|
2013-10-10 01:47:48 +02:00
|
|
|
*
|
2013-02-14 22:40:00 -06:00
|
|
|
* To create a new instance type, create a new class inheriting from this class
|
|
|
|
* and implement the pure virtual functions.
|
|
|
|
*/
|
2021-07-25 19:11:59 +02:00
|
|
|
class BaseInstance : public QObject, public std::enable_shared_from_this<BaseInstance>
|
2013-02-14 22:40:00 -06:00
|
|
|
{
|
|
|
|
Q_OBJECT
|
2013-08-03 15:57:33 +02:00
|
|
|
protected:
|
|
|
|
/// no-touchy!
|
2015-02-01 03:08:25 +01:00
|
|
|
BaseInstance(SettingsObjectPtr globalSettings, SettingsObjectPtr settings, const QString &rootDir);
|
2013-10-10 01:47:48 +02:00
|
|
|
|
2016-10-03 00:55:54 +02:00
|
|
|
public: /* types */
|
|
|
|
enum class Status
|
|
|
|
{
|
|
|
|
Present,
|
|
|
|
Gone // either nuked or invalidated
|
|
|
|
};
|
|
|
|
|
2013-02-14 22:40:00 -06:00
|
|
|
public:
|
2013-08-03 15:57:33 +02:00
|
|
|
/// virtual destructor to make sure the destruction is COMPLETE
|
|
|
|
virtual ~BaseInstance() {};
|
2013-10-10 01:47:48 +02:00
|
|
|
|
2017-12-03 18:36:28 +01:00
|
|
|
virtual void saveNow() = 0;
|
2015-01-27 22:31:07 +01:00
|
|
|
|
2016-10-03 00:55:54 +02:00
|
|
|
/***
|
2021-10-18 00:47:02 +02:00
|
|
|
* the instance has been invalidated - it is no longer tracked by the launcher for some reason,
|
2016-10-03 00:55:54 +02:00
|
|
|
* but it has not necessarily been deleted.
|
|
|
|
*
|
|
|
|
* Happens when the instance folder changes to some other location, or the instance is removed by external means.
|
|
|
|
*/
|
|
|
|
void invalidate();
|
|
|
|
|
2021-10-13 01:59:25 +02:00
|
|
|
/// The instance's ID. The ID SHALL be determined by LAUNCHER internally. The ID IS guaranteed to
|
2013-10-10 01:47:48 +02:00
|
|
|
/// be unique.
|
2014-01-15 22:49:37 +01:00
|
|
|
virtual QString id() const;
|
2013-10-10 01:47:48 +02:00
|
|
|
|
2014-12-18 02:48:14 +01:00
|
|
|
void setRunning(bool running);
|
|
|
|
bool isRunning() const;
|
2015-09-22 01:25:34 +02:00
|
|
|
int64_t totalTimePlayed() const;
|
2021-07-13 16:59:33 +01:00
|
|
|
int64_t lastTimePlayed() const;
|
2015-09-22 01:25:34 +02:00
|
|
|
void resetTimePlayed();
|
2014-06-30 02:02:57 +02:00
|
|
|
|
2013-08-03 15:57:33 +02:00
|
|
|
/// get the type of this instance
|
|
|
|
QString instanceType() const;
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2013-08-03 15:57:33 +02:00
|
|
|
/// Path to the instance's root directory.
|
2013-08-12 00:39:19 +02:00
|
|
|
QString instanceRoot() const;
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2018-07-27 23:57:09 +02:00
|
|
|
/// Path to the instance's game root directory.
|
|
|
|
virtual QString gameRoot() const
|
|
|
|
{
|
|
|
|
return instanceRoot();
|
|
|
|
}
|
|
|
|
|
2021-12-12 22:39:25 +01:00
|
|
|
/// Path to the instance's mods directory.
|
|
|
|
virtual QString modsRoot() const = 0;
|
|
|
|
|
2013-08-03 15:57:33 +02:00
|
|
|
QString name() const;
|
|
|
|
void setName(QString val);
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2014-01-12 23:38:12 +01:00
|
|
|
/// Value used for instance window titles
|
|
|
|
QString windowTitle() const;
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2013-08-03 15:57:33 +02:00
|
|
|
QString iconKey() const;
|
|
|
|
void setIconKey(QString val);
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2013-08-03 15:57:33 +02:00
|
|
|
QString notes() const;
|
|
|
|
void setNotes(QString val);
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2015-07-21 02:38:15 +02:00
|
|
|
QString getPreLaunchCommand();
|
|
|
|
QString getPostExitCommand();
|
|
|
|
QString getWrapperCommand();
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2022-08-20 11:33:34 -03:00
|
|
|
bool isManagedPack() const;
|
|
|
|
QString getManagedPackType() const;
|
|
|
|
QString getManagedPackID() const;
|
|
|
|
QString getManagedPackName() const;
|
|
|
|
QString getManagedPackVersionID() const;
|
|
|
|
QString getManagedPackVersionName() const;
|
2022-05-28 23:26:47 +01:00
|
|
|
void setManagedPack(const QString& type, const QString& id, const QString& name, const QString& versionId, const QString& version);
|
2022-07-31 19:48:38 -03:00
|
|
|
void copyManagedPack(BaseInstance& other);
|
2022-05-28 23:26:47 +01:00
|
|
|
|
2015-07-22 09:01:04 +02:00
|
|
|
/// guess log level from a line of game log
|
2022-11-09 21:15:35 +00:00
|
|
|
virtual MessageLevel::Enum guessLevel([[maybe_unused]] const QString &line, MessageLevel::Enum level)
|
2015-07-22 09:01:04 +02:00
|
|
|
{
|
|
|
|
return level;
|
|
|
|
};
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2022-07-06 17:17:54 -03:00
|
|
|
virtual QStringList extraArguments();
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2014-06-08 20:11:09 +02:00
|
|
|
/// Traits. Normally inside the version, depends on instance implementation.
|
2017-09-22 00:27:30 +02:00
|
|
|
virtual QSet <QString> traits() const = 0;
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2013-08-03 15:57:33 +02:00
|
|
|
/**
|
|
|
|
* Gets the time that the instance was last launched.
|
|
|
|
* Stored in milliseconds since epoch.
|
|
|
|
*/
|
|
|
|
qint64 lastLaunch() const;
|
|
|
|
/// Sets the last launched time to 'val' milliseconds since epoch
|
|
|
|
void setLastLaunch(qint64 val = QDateTime::currentMSecsSinceEpoch());
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2013-02-25 14:44:36 -06:00
|
|
|
/*!
|
|
|
|
* \brief Gets this instance's settings object.
|
|
|
|
* This settings object stores instance-specific settings.
|
2022-07-06 17:17:54 -03:00
|
|
|
*
|
|
|
|
* Note that this method is not const.
|
|
|
|
* It may call loadSpecificSettings() to ensure those are loaded.
|
|
|
|
*
|
2013-02-25 14:44:36 -06:00
|
|
|
* \return A pointer to this instance's settings object.
|
|
|
|
*/
|
2022-07-06 17:17:54 -03:00
|
|
|
virtual SettingsObjectPtr settings();
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2022-07-06 15:56:03 -03:00
|
|
|
/*!
|
2022-07-06 17:17:54 -03:00
|
|
|
* \brief Loads settings specific to an instance type if they're not already loaded.
|
2022-07-06 15:56:03 -03:00
|
|
|
*/
|
2022-07-06 17:17:54 -03:00
|
|
|
virtual void loadSpecificSettings() = 0;
|
2022-07-06 15:56:03 -03:00
|
|
|
|
2013-12-15 18:10:51 +01:00
|
|
|
/// returns a valid update task
|
2021-11-21 23:36:55 +01:00
|
|
|
virtual Task::Ptr createUpdateTask(Net::Mode mode) = 0;
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2015-07-10 00:06:05 +02:00
|
|
|
/// returns a valid launcher (task container)
|
2021-05-22 18:07:08 +02:00
|
|
|
virtual shared_qobject_ptr<LaunchTask> createLaunchTask(
|
|
|
|
AuthSessionPtr account, MinecraftServerTargetPtr serverToJoin) = 0;
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2016-08-06 15:39:29 +02:00
|
|
|
/// returns the current launch task (if any)
|
2019-04-07 23:59:04 +02:00
|
|
|
shared_qobject_ptr<LaunchTask> getLaunchTask();
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2015-07-21 02:38:15 +02:00
|
|
|
/*!
|
|
|
|
* Create envrironment variables for running the instance
|
|
|
|
*/
|
|
|
|
virtual QProcessEnvironment createEnvironment() = 0;
|
2022-06-29 22:37:25 +02:00
|
|
|
virtual QProcessEnvironment createLaunchEnvironment() = 0;
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2015-08-18 02:25:24 +02:00
|
|
|
/*!
|
|
|
|
* Returns a matcher that can maps relative paths within the instance to whether they are 'log files'
|
|
|
|
*/
|
|
|
|
virtual IPathMatcher::Ptr getLogFileMatcher() = 0;
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2015-08-19 02:04:56 +02:00
|
|
|
/*!
|
|
|
|
* Returns the root folder to use for looking up log files
|
|
|
|
*/
|
|
|
|
virtual QString getLogFileRoot() = 0;
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2013-08-25 01:32:42 +02:00
|
|
|
virtual QString getStatusbarDescription() = 0;
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2013-08-25 01:32:42 +02:00
|
|
|
/// FIXME: this really should be elsewhere...
|
|
|
|
virtual QString instanceConfigFolder() const = 0;
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2015-07-21 02:38:15 +02:00
|
|
|
/// get variables this instance exports
|
2022-07-06 17:17:54 -03:00
|
|
|
virtual QMap<QString, QString> getVariables() = 0;
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2015-09-22 01:06:45 +02:00
|
|
|
virtual QString typeName() const = 0;
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2022-07-11 09:01:07 +02:00
|
|
|
void updateRuntimeContext();
|
|
|
|
RuntimeContext runtimeContext() const
|
|
|
|
{
|
|
|
|
return m_runtimeContext;
|
|
|
|
}
|
|
|
|
|
2016-11-04 01:17:28 +01:00
|
|
|
bool hasVersionBroken() const
|
2014-02-17 20:31:50 +01:00
|
|
|
{
|
2016-11-04 01:17:28 +01:00
|
|
|
return m_hasBrokenVersion;
|
|
|
|
}
|
|
|
|
void setVersionBroken(bool value)
|
|
|
|
{
|
|
|
|
if(m_hasBrokenVersion != value)
|
|
|
|
{
|
|
|
|
m_hasBrokenVersion = value;
|
|
|
|
emit propertiesChanged(this);
|
|
|
|
}
|
|
|
|
}
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2016-11-04 01:17:28 +01:00
|
|
|
bool hasUpdateAvailable() const
|
|
|
|
{
|
|
|
|
return m_hasUpdate;
|
|
|
|
}
|
|
|
|
void setUpdateAvailable(bool value)
|
|
|
|
{
|
|
|
|
if(m_hasUpdate != value)
|
|
|
|
{
|
|
|
|
m_hasUpdate = value;
|
|
|
|
emit propertiesChanged(this);
|
|
|
|
}
|
|
|
|
}
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2016-11-04 01:17:28 +01:00
|
|
|
bool hasCrashed() const
|
|
|
|
{
|
|
|
|
return m_crashed;
|
|
|
|
}
|
|
|
|
void setCrashed(bool value)
|
|
|
|
{
|
|
|
|
if(m_crashed != value)
|
|
|
|
{
|
|
|
|
m_crashed = value;
|
|
|
|
emit propertiesChanged(this);
|
|
|
|
}
|
|
|
|
}
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2017-09-20 23:38:31 +02:00
|
|
|
virtual bool canLaunch() const;
|
|
|
|
virtual bool canEdit() const = 0;
|
2016-04-28 00:04:37 +02:00
|
|
|
virtual bool canExport() const = 0;
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2017-11-11 01:38:31 +01:00
|
|
|
bool reloadSettings();
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2016-06-16 02:20:23 +02:00
|
|
|
/**
|
2021-05-22 16:14:25 +02:00
|
|
|
* 'print' a verbose description of the instance into a QStringList
|
2016-06-16 02:20:23 +02:00
|
|
|
*/
|
2021-05-22 16:14:25 +02:00
|
|
|
virtual QStringList verboseDescription(AuthSessionPtr session, MinecraftServerTargetPtr serverToJoin) = 0;
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2016-10-03 00:55:54 +02:00
|
|
|
Status currentStatus() const;
|
2018-07-15 14:51:05 +02:00
|
|
|
|
2017-02-08 20:01:42 +01:00
|
|
|
int getConsoleMaxLines() const;
|
|
|
|
bool shouldStopOnConsoleOverflow() const;
|
|
|
|
|
2023-02-15 22:01:27 -07:00
|
|
|
QStringList getLinkedInstances() const;
|
|
|
|
void setLinkedInstances(const QStringList& list);
|
|
|
|
void addLinkedInstanceId(const QString& id);
|
|
|
|
bool removeLinkedInstanceId(const QString& id);
|
|
|
|
bool isLinkedToInstanceId(const QString& id) const;
|
|
|
|
|
2016-10-03 00:55:54 +02:00
|
|
|
protected:
|
|
|
|
void changeStatus(Status newStatus);
|
|
|
|
|
2022-07-06 17:17:54 -03:00
|
|
|
SettingsObjectPtr globalSettings() const { return m_global_settings.lock(); };
|
|
|
|
|
|
|
|
bool isSpecificSettingsLoaded() const { return m_specific_settings_loaded; }
|
|
|
|
void setSpecificSettingsLoaded(bool loaded) { m_specific_settings_loaded = loaded; }
|
|
|
|
|
2013-03-19 06:24:34 +01:00
|
|
|
signals:
|
|
|
|
/*!
|
|
|
|
* \brief Signal emitted when properties relevant to the instance view change
|
|
|
|
*/
|
2013-10-10 01:47:48 +02:00
|
|
|
void propertiesChanged(BaseInstance *inst);
|
|
|
|
|
2019-04-07 23:59:04 +02:00
|
|
|
void launchTaskChanged(shared_qobject_ptr<LaunchTask>);
|
2016-08-06 15:39:29 +02:00
|
|
|
|
|
|
|
void runningStatusChanged(bool running);
|
|
|
|
|
2016-10-03 00:55:54 +02:00
|
|
|
void statusChanged(Status from, Status to);
|
|
|
|
|
2013-12-31 01:24:28 +01:00
|
|
|
protected slots:
|
|
|
|
void iconUpdated(QString key);
|
|
|
|
|
2016-10-03 00:55:54 +02:00
|
|
|
protected: /* data */
|
2014-12-18 02:48:14 +01:00
|
|
|
QString m_rootDir;
|
2015-02-09 01:51:14 +01:00
|
|
|
SettingsObjectPtr m_settings;
|
2016-11-04 01:17:28 +01:00
|
|
|
// InstanceFlags m_flags;
|
2014-12-18 02:48:14 +01:00
|
|
|
bool m_isRunning = false;
|
2019-04-07 23:59:04 +02:00
|
|
|
shared_qobject_ptr<LaunchTask> m_launchProcess;
|
2015-09-22 01:06:45 +02:00
|
|
|
QDateTime m_timeStarted;
|
2022-07-11 09:01:07 +02:00
|
|
|
RuntimeContext m_runtimeContext;
|
2016-10-03 00:55:54 +02:00
|
|
|
|
|
|
|
private: /* data */
|
|
|
|
Status m_status = Status::Present;
|
2016-11-04 01:17:28 +01:00
|
|
|
bool m_crashed = false;
|
|
|
|
bool m_hasUpdate = false;
|
|
|
|
bool m_hasBrokenVersion = false;
|
2022-07-06 17:17:54 -03:00
|
|
|
|
|
|
|
SettingsObjectWeakPtr m_global_settings;
|
|
|
|
bool m_specific_settings_loaded = false;
|
|
|
|
|
2013-02-14 22:40:00 -06:00
|
|
|
};
|
|
|
|
|
2019-04-07 23:59:04 +02:00
|
|
|
Q_DECLARE_METATYPE(shared_qobject_ptr<BaseInstance>)
|
2016-11-04 01:17:28 +01:00
|
|
|
//Q_DECLARE_METATYPE(BaseInstance::InstanceFlag)
|
|
|
|
//Q_DECLARE_OPERATORS_FOR_FLAGS(BaseInstance::InstanceFlags)
|