mirror of
https://github.com/PrismLauncher/PrismLauncher.git
synced 2025-04-29 22:24:26 +02:00
150 lines
5.5 KiB
C++
150 lines
5.5 KiB
C++
// SPDX-License-Identifier: GPL-3.0-only
|
|
/*
|
|
* Prism Launcher - Minecraft Launcher
|
|
* Copyright (c) 2022 flowln <flowlnlnln@gmail.com>
|
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
|
*
|
|
* 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.
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
#include "ResourceFolderLoadTask.h"
|
|
|
|
#include "Application.h"
|
|
#include "FileSystem.h"
|
|
#include "minecraft/mod/MetadataHandler.h"
|
|
|
|
#include <QThread>
|
|
|
|
ResourceFolderLoadTask::ResourceFolderLoadTask(const QDir& resource_dir,
|
|
const QDir& index_dir,
|
|
bool is_indexed,
|
|
bool clean_orphan,
|
|
std::function<Resource*(const QFileInfo&)> create_function)
|
|
: Task(false)
|
|
, m_resource_dir(resource_dir)
|
|
, m_index_dir(index_dir)
|
|
, m_is_indexed(is_indexed)
|
|
, m_clean_orphan(clean_orphan)
|
|
, m_create_func(create_function)
|
|
, m_result(new Result())
|
|
, m_thread_to_spawn_into(thread())
|
|
{}
|
|
|
|
void ResourceFolderLoadTask::executeTask()
|
|
{
|
|
if (thread() != m_thread_to_spawn_into)
|
|
connect(this, &Task::finished, this->thread(), &QThread::quit);
|
|
|
|
if (m_is_indexed) {
|
|
// Read metadata first
|
|
getFromMetadata();
|
|
}
|
|
|
|
// Read JAR files that don't have metadata
|
|
m_resource_dir.refresh();
|
|
for (auto entry : m_resource_dir.entryInfoList()) {
|
|
auto filePath = entry.absoluteFilePath();
|
|
if (auto app = APPLICATION_DYN; app && app->checkQSavePath(filePath)) {
|
|
continue;
|
|
}
|
|
auto newFilePath = FS::getUniqueResourceName(filePath);
|
|
if (newFilePath != filePath) {
|
|
FS::move(filePath, newFilePath);
|
|
entry = QFileInfo(newFilePath);
|
|
}
|
|
|
|
Resource* resource = m_create_func(entry);
|
|
|
|
if (resource->enabled()) {
|
|
if (m_result->resources.contains(resource->internal_id())) {
|
|
m_result->resources[resource->internal_id()]->setStatus(ResourceStatus::INSTALLED);
|
|
// Delete the object we just created, since a valid one is already in the mods list.
|
|
delete resource;
|
|
} else {
|
|
m_result->resources[resource->internal_id()].reset(resource);
|
|
m_result->resources[resource->internal_id()]->setStatus(ResourceStatus::NO_METADATA);
|
|
}
|
|
} else {
|
|
QString chopped_id = resource->internal_id().chopped(9);
|
|
if (m_result->resources.contains(chopped_id)) {
|
|
m_result->resources[resource->internal_id()].reset(resource);
|
|
|
|
auto metadata = m_result->resources[chopped_id]->metadata();
|
|
if (metadata) {
|
|
resource->setMetadata(*metadata);
|
|
|
|
m_result->resources[resource->internal_id()]->setStatus(ResourceStatus::INSTALLED);
|
|
m_result->resources.remove(chopped_id);
|
|
}
|
|
} else {
|
|
m_result->resources[resource->internal_id()].reset(resource);
|
|
m_result->resources[resource->internal_id()]->setStatus(ResourceStatus::NO_METADATA);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Remove orphan metadata to prevent issues
|
|
// See https://github.com/PolyMC/PolyMC/issues/996
|
|
if (m_clean_orphan) {
|
|
QMutableMapIterator iter(m_result->resources);
|
|
while (iter.hasNext()) {
|
|
auto resource = iter.next().value();
|
|
if (resource->status() == ResourceStatus::NOT_INSTALLED) {
|
|
resource->destroy(m_index_dir, false, false);
|
|
iter.remove();
|
|
}
|
|
}
|
|
}
|
|
|
|
for (auto mod : m_result->resources)
|
|
mod->moveToThread(m_thread_to_spawn_into);
|
|
|
|
if (m_aborted)
|
|
emit finished();
|
|
else
|
|
emitSucceeded();
|
|
}
|
|
|
|
void ResourceFolderLoadTask::getFromMetadata()
|
|
{
|
|
m_index_dir.refresh();
|
|
for (auto entry : m_index_dir.entryList(QDir::Files)) {
|
|
auto metadata = Metadata::get(m_index_dir, entry);
|
|
|
|
if (!metadata.isValid())
|
|
continue;
|
|
|
|
auto* resource = m_create_func(QFileInfo(m_resource_dir.filePath(metadata.filename)));
|
|
resource->setMetadata(metadata);
|
|
resource->setStatus(ResourceStatus::NOT_INSTALLED);
|
|
m_result->resources[resource->internal_id()].reset(resource);
|
|
}
|
|
}
|