pollymc/launcher/BaseInstance.cpp

412 lines
12 KiB
C++
Raw Normal View History

// SPDX-License-Identifier: GPL-3.0-only
/*
* PolyMC - Minecraft Launcher
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
* Copyright (c) 2022 Jamie Mansfield <jmansfield@cadixdev.org>
2013-01-15 05:12:38 +05:30
*
* 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-15 05:12:38 +05:30
*
* 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-15 05:12:38 +05:30
*/
#include "BaseInstance.h"
2013-01-15 05:12:38 +05:30
2013-02-19 04:28:53 +05:30
#include <QFileInfo>
#include <QDir>
2016-10-03 04:25:54 +05:30
#include <QDebug>
#include <QRegularExpression>
#include <QJsonDocument>
#include <QJsonObject>
2013-02-19 04:28:53 +05:30
2015-02-09 06:21:14 +05:30
#include "settings/INISettingsObject.h"
#include "settings/Setting.h"
#include "settings/OverrideSetting.h"
2013-02-26 02:14:36 +05:30
2015-10-05 05:17:27 +05:30
#include "FileSystem.h"
#include "Commandline.h"
2021-10-18 04:17:02 +05:30
#include "BuildConfig.h"
2013-02-19 04:28:53 +05:30
2015-02-01 07:38:25 +05:30
BaseInstance::BaseInstance(SettingsObjectPtr globalSettings, SettingsObjectPtr settings, const QString &rootDir)
2018-07-15 18:21:05 +05:30
: QObject()
2013-01-15 05:12:38 +05:30
{
2018-07-15 18:21:05 +05:30
m_settings = settings;
m_global_settings = globalSettings;
2018-07-15 18:21:05 +05:30
m_rootDir = rootDir;
2014-12-18 07:18:14 +05:30
2018-07-15 18:21:05 +05:30
m_settings->registerSetting("name", "Unnamed Instance");
m_settings->registerSetting("iconKey", "default");
m_settings->registerSetting("notes", "");
2018-07-15 18:21:05 +05:30
m_settings->registerSetting("lastLaunchTime", 0);
m_settings->registerSetting("totalTimePlayed", 0);
m_settings->registerSetting("lastTimePlayed", 0);
2022-06-12 02:36:42 +05:30
m_settings->registerSetting("linkedInstancesList", "[]");
// Game time override
auto gameTimeOverride = m_settings->registerSetting("OverrideGameTime", false);
m_settings->registerOverride(globalSettings->getSetting("ShowGameTime"), gameTimeOverride);
m_settings->registerOverride(globalSettings->getSetting("RecordGameTime"), gameTimeOverride);
2022-06-12 02:36:42 +05:30
// NOTE: Sometimees InstanceType is already registered, as it was used to identify the type of
// a locally stored instance
if (!m_settings->getSetting("InstanceType"))
m_settings->registerSetting("InstanceType", "");
2018-07-15 18:21:05 +05:30
// Custom Commands
auto commandSetting = m_settings->registerSetting({"OverrideCommands","OverrideLaunchCmd"}, false);
m_settings->registerOverride(globalSettings->getSetting("PreLaunchCommand"), commandSetting);
m_settings->registerOverride(globalSettings->getSetting("WrapperCommand"), commandSetting);
m_settings->registerOverride(globalSettings->getSetting("PostExitCommand"), commandSetting);
2018-07-15 18:21:05 +05:30
// Console
auto consoleSetting = m_settings->registerSetting("OverrideConsole", false);
m_settings->registerOverride(globalSettings->getSetting("ShowConsole"), consoleSetting);
m_settings->registerOverride(globalSettings->getSetting("AutoCloseConsole"), consoleSetting);
m_settings->registerOverride(globalSettings->getSetting("ShowConsoleOnError"), consoleSetting);
m_settings->registerOverride(globalSettings->getSetting("LogPrePostOutput"), consoleSetting);
2018-07-15 18:21:05 +05:30
m_settings->registerPassthrough(globalSettings->getSetting("ConsoleMaxLines"), nullptr);
m_settings->registerPassthrough(globalSettings->getSetting("ConsoleOverflowStop"), nullptr);
// Managed Packs
m_settings->registerSetting("ManagedPack", false);
m_settings->registerSetting("ManagedPackType", "");
m_settings->registerSetting("ManagedPackID", "");
m_settings->registerSetting("ManagedPackName", "");
m_settings->registerSetting("ManagedPackVersionID", "");
m_settings->registerSetting("ManagedPackVersionName", "");
2013-02-19 04:28:53 +05:30
}
QString BaseInstance::getPreLaunchCommand()
{
2018-07-15 18:21:05 +05:30
return settings()->get("PreLaunchCommand").toString();
}
QString BaseInstance::getWrapperCommand()
{
2018-07-15 18:21:05 +05:30
return settings()->get("WrapperCommand").toString();
}
QString BaseInstance::getPostExitCommand()
{
2018-07-15 18:21:05 +05:30
return settings()->get("PostExitCommand").toString();
}
bool BaseInstance::isManagedPack() const
{
return m_settings->get("ManagedPack").toBool();
}
QString BaseInstance::getManagedPackType() const
{
return m_settings->get("ManagedPackType").toString();
}
QString BaseInstance::getManagedPackID() const
{
return m_settings->get("ManagedPackID").toString();
}
QString BaseInstance::getManagedPackName() const
{
return m_settings->get("ManagedPackName").toString();
}
QString BaseInstance::getManagedPackVersionID() const
{
return m_settings->get("ManagedPackVersionID").toString();
}
QString BaseInstance::getManagedPackVersionName() const
{
return m_settings->get("ManagedPackVersionName").toString();
}
void BaseInstance::setManagedPack(const QString& type, const QString& id, const QString& name, const QString& versionId, const QString& version)
{
m_settings->set("ManagedPack", true);
m_settings->set("ManagedPackType", type);
m_settings->set("ManagedPackID", id);
m_settings->set("ManagedPackName", name);
m_settings->set("ManagedPackVersionID", versionId);
m_settings->set("ManagedPackVersionName", version);
}
void BaseInstance::copyManagedPack(BaseInstance& other)
{
m_settings->set("ManagedPack", other.isManagedPack());
m_settings->set("ManagedPackType", other.getManagedPackType());
m_settings->set("ManagedPackID", other.getManagedPackID());
m_settings->set("ManagedPackName", other.getManagedPackName());
m_settings->set("ManagedPackVersionID", other.getManagedPackVersionID());
m_settings->set("ManagedPackVersionName", other.getManagedPackVersionName());
}
int BaseInstance::getConsoleMaxLines() const
{
auto lineSetting = m_settings->getSetting("ConsoleMaxLines");
2018-07-15 18:21:05 +05:30
bool conversionOk = false;
int maxLines = lineSetting->get().toInt(&conversionOk);
if(!conversionOk)
{
maxLines = lineSetting->defValue().toInt();
qWarning() << "ConsoleMaxLines has nonsensical value, defaulting to" << maxLines;
}
return maxLines;
}
bool BaseInstance::shouldStopOnConsoleOverflow() const
{
return m_settings->get("ConsoleOverflowStop").toBool();
}
QStringList BaseInstance::getLinkedInstances() const
{
return m_settings->getList<QString>("linkedInstancesList");
}
void BaseInstance::setLinkedInstances(const QStringList& list)
{
auto linkedInstancesList = m_settings->getList<QString>("linkedInstancesList");
m_settings->setList("linkedInstancesList", list);
}
void BaseInstance::addLinkedInstanceId(const QString& id)
{
auto linkedInstancesList = m_settings->getList<QString>("linkedInstancesList");
linkedInstancesList.append(id);
setLinkedInstances(linkedInstancesList);
}
bool BaseInstance::removeLinkedInstanceId(const QString& id)
{
auto linkedInstancesList = m_settings->getList<QString>("linkedInstancesList");
int numRemoved = linkedInstancesList.removeAll(id);
setLinkedInstances(linkedInstancesList);
return numRemoved > 0;
}
bool BaseInstance::isLinkedToInstanceId(const QString& id) const
{
auto linkedInstancesList = m_settings->getList<QString>("linkedInstancesList");
return linkedInstancesList.contains(id);
}
void BaseInstance::iconUpdated(QString key)
{
2018-07-15 18:21:05 +05:30
if(iconKey() == key)
{
emit propertiesChanged(this);
}
}
2016-10-03 04:25:54 +05:30
void BaseInstance::invalidate()
{
2018-07-15 18:21:05 +05:30
changeStatus(Status::Gone);
qDebug() << "Instance" << id() << "has been invalidated.";
2016-10-03 04:25:54 +05:30
}
void BaseInstance::changeStatus(BaseInstance::Status newStatus)
{
2018-07-15 18:21:05 +05:30
Status status = currentStatus();
if(status != newStatus)
{
m_status = newStatus;
emit statusChanged(status, newStatus);
}
2016-10-03 04:25:54 +05:30
}
BaseInstance::Status BaseInstance::currentStatus() const
{
2018-07-15 18:21:05 +05:30
return m_status;
}
QString BaseInstance::id() const
2013-02-19 04:28:53 +05:30
{
2018-07-15 18:21:05 +05:30
return QFileInfo(instanceRoot()).fileName();
2013-02-19 04:28:53 +05:30
}
bool BaseInstance::isRunning() const
{
2018-07-15 18:21:05 +05:30
return m_isRunning;
}
2014-12-18 07:18:14 +05:30
void BaseInstance::setRunning(bool running)
{
2018-07-15 18:21:05 +05:30
if(running == m_isRunning)
return;
2018-07-15 18:21:05 +05:30
m_isRunning = running;
if(!m_settings->get("RecordGameTime").toBool())
{
emit runningStatusChanged(running);
return;
}
2018-07-15 18:21:05 +05:30
if(running)
{
m_timeStarted = QDateTime::currentDateTime();
}
else
{
QDateTime timeEnded = QDateTime::currentDateTime();
qint64 current = settings()->get("totalTimePlayed").toLongLong();
2018-07-15 18:21:05 +05:30
settings()->set("totalTimePlayed", current + m_timeStarted.secsTo(timeEnded));
settings()->set("lastTimePlayed", m_timeStarted.secsTo(timeEnded));
2018-07-15 18:21:05 +05:30
emit propertiesChanged(this);
}
2018-07-15 18:21:05 +05:30
emit runningStatusChanged(running);
}
int64_t BaseInstance::totalTimePlayed() const
{
qint64 current = m_settings->get("totalTimePlayed").toLongLong();
2018-07-15 18:21:05 +05:30
if(m_isRunning)
{
QDateTime timeNow = QDateTime::currentDateTime();
return current + m_timeStarted.secsTo(timeNow);
}
return current;
}
int64_t BaseInstance::lastTimePlayed() const
{
if(m_isRunning)
{
QDateTime timeNow = QDateTime::currentDateTime();
return m_timeStarted.secsTo(timeNow);
}
return m_settings->get("lastTimePlayed").toLongLong();
}
void BaseInstance::resetTimePlayed()
{
2018-07-15 18:21:05 +05:30
settings()->reset("totalTimePlayed");
settings()->reset("lastTimePlayed");
}
2013-08-03 19:27:33 +05:30
QString BaseInstance::instanceType() const
{
2018-07-15 18:21:05 +05:30
return m_settings->get("InstanceType").toString();
2013-08-03 19:27:33 +05:30
}
QString BaseInstance::instanceRoot() const
2013-02-19 04:28:53 +05:30
{
2018-07-15 18:21:05 +05:30
return m_rootDir;
2013-02-19 04:28:53 +05:30
}
SettingsObjectPtr BaseInstance::settings()
{
loadSpecificSettings();
2018-07-15 18:21:05 +05:30
return m_settings;
}
bool BaseInstance::canLaunch() const
{
2018-07-15 18:21:05 +05:30
return (!hasVersionBroken() && !isRunning());
}
bool BaseInstance::reloadSettings()
{
2018-07-15 18:21:05 +05:30
return m_settings->reload();
}
2013-08-03 19:27:33 +05:30
qint64 BaseInstance::lastLaunch() const
{
2018-07-15 18:21:05 +05:30
return m_settings->get("lastLaunchTime").value<qint64>();
}
2014-12-18 07:18:14 +05:30
void BaseInstance::setLastLaunch(qint64 val)
{
2018-07-15 18:21:05 +05:30
//FIXME: if no change, do not set. setting involves saving a file.
m_settings->set("lastLaunchTime", val);
emit propertiesChanged(this);
}
void BaseInstance::setNotes(QString val)
{
2018-07-15 18:21:05 +05:30
//FIXME: if no change, do not set. setting involves saving a file.
m_settings->set("notes", val);
}
2014-12-18 07:18:14 +05:30
2013-08-03 19:27:33 +05:30
QString BaseInstance::notes() const
{
2018-07-15 18:21:05 +05:30
return m_settings->get("notes").toString();
}
void BaseInstance::setIconKey(QString val)
2013-03-13 23:43:28 +05:30
{
2018-07-15 18:21:05 +05:30
//FIXME: if no change, do not set. setting involves saving a file.
m_settings->set("iconKey", val);
emit propertiesChanged(this);
2013-03-13 23:43:28 +05:30
}
2014-12-18 07:18:14 +05:30
2013-08-03 19:27:33 +05:30
QString BaseInstance::iconKey() const
2013-03-13 23:43:28 +05:30
{
2018-07-15 18:21:05 +05:30
return m_settings->get("iconKey").toString();
2013-03-13 23:43:28 +05:30
}
void BaseInstance::setName(QString val)
{
2018-07-15 18:21:05 +05:30
//FIXME: if no change, do not set. setting involves saving a file.
m_settings->set("name", val);
emit propertiesChanged(this);
}
2013-08-03 19:27:33 +05:30
QString BaseInstance::name() const
2013-02-26 02:14:36 +05:30
{
2018-07-15 18:21:05 +05:30
return m_settings->get("name").toString();
2013-02-26 02:14:36 +05:30
}
2014-01-13 04:08:12 +05:30
QString BaseInstance::windowTitle() const
{
return BuildConfig.LAUNCHER_DISPLAYNAME + ": " + name().replace(QRegularExpression("\\s+"), " ");
2014-01-13 04:08:12 +05:30
}
// FIXME: why is this here? move it to MinecraftInstance!!!
QStringList BaseInstance::extraArguments()
{
2018-07-15 18:21:05 +05:30
return Commandline::splitArgs(settings()->get("JvmArgs").toString());
}
shared_qobject_ptr<LaunchTask> BaseInstance::getLaunchTask()
{
2018-07-15 18:21:05 +05:30
return m_launchProcess;
}
void BaseInstance::updateRuntimeContext()
{
// NOOP
}