diff options
author | Petr Mrázek <peterix@gmail.com> | 2016-10-03 00:55:54 +0200 |
---|---|---|
committer | Petr Mrázek <peterix@gmail.com> | 2016-10-26 18:21:24 +0200 |
commit | d66fdcd4cc6913508d2987c14cd9fc4d6760b8a5 (patch) | |
tree | 4f385106ce732d4f7338feab5391f2a06c68a0e6 /api/logic/InstanceImportTask.cpp | |
parent | bbe139dce51a7965394c800cac974946820d3869 (diff) | |
download | PrismLauncher-d66fdcd4cc6913508d2987c14cd9fc4d6760b8a5.tar.gz PrismLauncher-d66fdcd4cc6913508d2987c14cd9fc4d6760b8a5.tar.bz2 PrismLauncher-d66fdcd4cc6913508d2987c14cd9fc4d6760b8a5.zip |
NOISSUE Granular instance reload
Diffstat (limited to 'api/logic/InstanceImportTask.cpp')
-rw-r--r-- | api/logic/InstanceImportTask.cpp | 146 |
1 files changed, 146 insertions, 0 deletions
diff --git a/api/logic/InstanceImportTask.cpp b/api/logic/InstanceImportTask.cpp new file mode 100644 index 00000000..99f2553f --- /dev/null +++ b/api/logic/InstanceImportTask.cpp @@ -0,0 +1,146 @@ + +#include "InstanceImportTask.h" +#include "BaseInstance.h" +#include "BaseInstanceProvider.h" +#include "FileSystem.h" +#include "Env.h" +#include "MMCZip.h" +#include "NullInstance.h" +#include "settings/INISettingsObject.h" +#include "icons/IIconList.h" + +InstanceImportTask::InstanceImportTask(SettingsObjectPtr settings, const QUrl sourceUrl, BaseInstanceProvider * target, + const QString &instName, const QString &instIcon, const QString &instGroup) +{ + m_globalSettings = settings; + m_sourceUrl = sourceUrl; + m_target = target; + m_instName = instName; + m_instIcon = instIcon; + m_instGroup = instGroup; +} + +void InstanceImportTask::executeTask() +{ + InstancePtr newInstance; + + if (m_sourceUrl.isLocalFile()) + { + m_archivePath = m_sourceUrl.toLocalFile(); + extractAndTweak(); + } + else + { + setStatus(tr("Downloading modpack:\n%1").arg(m_sourceUrl.toString())); + m_downloadRequired = true; + + const QString path = m_sourceUrl.host() + '/' + m_sourceUrl.path(); + auto entry = ENV.metacache()->resolveEntry("general", path); + entry->setStale(true); + m_filesNetJob.reset(new NetJob(tr("Modpack download"))); + m_filesNetJob->addNetAction(Net::Download::makeCached(m_sourceUrl, entry)); + m_archivePath = entry->getFullPath(); + auto job = m_filesNetJob.get(); + connect(job, &NetJob::succeeded, this, &InstanceImportTask::downloadSucceeded); + connect(job, &NetJob::progress, this, &InstanceImportTask::downloadProgressChanged); + connect(job, &NetJob::failed, this, &InstanceImportTask::downloadFailed); + m_filesNetJob->start(); + } +} + +void InstanceImportTask::downloadSucceeded() +{ + extractAndTweak(); +} + +void InstanceImportTask::downloadFailed(QString reason) +{ + emitFailed(reason); +} + +void InstanceImportTask::downloadProgressChanged(qint64 current, qint64 total) +{ + setProgress(current / 2, total); +} + +static QFileInfo findRecursive(const QString &dir, const QString &name) +{ + for (const auto info : QDir(dir).entryInfoList(QDir::NoDotAndDotDot | QDir::Dirs | QDir::Files, QDir::DirsLast)) + { + if (info.isFile() && info.fileName() == name) + { + return info; + } + else if (info.isDir()) + { + const QFileInfo res = findRecursive(info.absoluteFilePath(), name); + if (res.isFile() && res.exists()) + { + return res; + } + } + } + return QFileInfo(); +} + +void InstanceImportTask::extractAndTweak() +{ + setStatus(tr("Extracting modpack")); + QString stagingPath = m_target->getStagedInstancePath(); + QDir extractDir(stagingPath); + qDebug() << "Attempting to create instance from" << m_archivePath; + if (MMCZip::extractDir(m_archivePath, extractDir.absolutePath()).isEmpty()) + { + m_target->destroyStagingPath(stagingPath); + emitFailed(tr("Failed to extract modpack")); + return; + } + const QFileInfo instanceCfgFile = findRecursive(extractDir.absolutePath(), "instance.cfg"); + if (!instanceCfgFile.isFile() || !instanceCfgFile.exists()) + { + m_target->destroyStagingPath(stagingPath); + emitFailed(tr("Archive does not contain instance.cfg")); + return; + } + + // FIXME: copy from FolderInstanceProvider!!! FIX IT!!! + auto instanceSettings = std::make_shared<INISettingsObject>(instanceCfgFile.absoluteFilePath()); + instanceSettings->registerSetting("InstanceType", "Legacy"); + + QString actualDir = instanceCfgFile.absolutePath(); + NullInstance instance(m_globalSettings, instanceSettings, actualDir); + + // reset time played on import... because packs. + instance.resetTimePlayed(); + + // set a new nice name + instance.setName(m_instName); + + // if the icon was specified by user, use that. otherwise pull icon from the pack + if (m_instIcon != "default") + { + instance.setIconKey(m_instIcon); + } + else + { + m_instIcon = instance.iconKey(); + auto importIconPath = FS::PathCombine(instance.instanceRoot(), m_instIcon + ".png"); + if (QFile::exists(importIconPath)) + { + // import icon + auto iconList = ENV.icons(); + if (iconList->iconFileExists(m_instIcon)) + { + iconList->deleteIcon(m_instIcon); + } + iconList->installIcons({importIconPath}); + } + } + if (!m_target->commitStagedInstance(stagingPath, actualDir, m_instName, m_instGroup)) + { + m_target->destroyStagingPath(stagingPath); + emitFailed(tr("Unable to commit instance")); + return; + } + emitSucceeded(); +} |