Merge pull request #1150 from flowln/fix_crash_on_game_quit
This commit is contained in:
commit
40c68595d7
@ -714,7 +714,7 @@ QStringList MinecraftInstance::verboseDescription(AuthSessionPtr session, Minecr
|
|||||||
{
|
{
|
||||||
out << QString("%1:").arg(label);
|
out << QString("%1:").arg(label);
|
||||||
auto modList = model.allMods();
|
auto modList = model.allMods();
|
||||||
std::sort(modList.begin(), modList.end(), [](Mod::Ptr a, Mod::Ptr b) {
|
std::sort(modList.begin(), modList.end(), [](auto a, auto b) {
|
||||||
auto aName = a->fileinfo().completeBaseName();
|
auto aName = a->fileinfo().completeBaseName();
|
||||||
auto bName = b->fileinfo().completeBaseName();
|
auto bName = b->fileinfo().completeBaseName();
|
||||||
return aName.localeAwareCompare(bName) < 0;
|
return aName.localeAwareCompare(bName) < 0;
|
||||||
|
@ -234,7 +234,7 @@ auto ModFolderModel::allMods() -> QList<Mod*>
|
|||||||
{
|
{
|
||||||
QList<Mod*> mods;
|
QList<Mod*> mods;
|
||||||
|
|
||||||
for (auto& res : m_resources) {
|
for (auto& res : qAsConst(m_resources)) {
|
||||||
mods.append(static_cast<Mod*>(res.get()));
|
mods.append(static_cast<Mod*>(res.get()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -251,7 +251,7 @@ bool ResourceFolderModel::update()
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ResourceFolderModel::resolveResource(Resource::Ptr res)
|
void ResourceFolderModel::resolveResource(Resource* res)
|
||||||
{
|
{
|
||||||
if (!res->shouldResolve()) {
|
if (!res->shouldResolve()) {
|
||||||
return;
|
return;
|
||||||
|
@ -68,7 +68,7 @@ class ResourceFolderModel : public QAbstractListModel {
|
|||||||
virtual bool update();
|
virtual bool update();
|
||||||
|
|
||||||
/** Creates a new parse task, if needed, for 'res' and start it.*/
|
/** Creates a new parse task, if needed, for 'res' and start it.*/
|
||||||
virtual void resolveResource(Resource::Ptr res);
|
virtual void resolveResource(Resource* res);
|
||||||
|
|
||||||
[[nodiscard]] size_t size() const { return m_resources.size(); };
|
[[nodiscard]] size_t size() const { return m_resources.size(); };
|
||||||
[[nodiscard]] bool empty() const { return size() == 0; }
|
[[nodiscard]] bool empty() const { return size() == 0; }
|
||||||
@ -247,7 +247,7 @@ void ResourceFolderModel::applyUpdates(QSet<QString>& current_set, QSet<QString>
|
|||||||
auto row = row_it.value();
|
auto row = row_it.value();
|
||||||
|
|
||||||
auto& new_resource = new_resources[kept];
|
auto& new_resource = new_resources[kept];
|
||||||
auto const& current_resource = m_resources[row];
|
auto const& current_resource = m_resources.at(row);
|
||||||
|
|
||||||
if (new_resource->dateTimeChanged() == current_resource->dateTimeChanged()) {
|
if (new_resource->dateTimeChanged() == current_resource->dateTimeChanged()) {
|
||||||
// no significant change, ignore...
|
// no significant change, ignore...
|
||||||
@ -265,7 +265,7 @@ void ResourceFolderModel::applyUpdates(QSet<QString>& current_set, QSet<QString>
|
|||||||
}
|
}
|
||||||
|
|
||||||
m_resources[row].reset(new_resource);
|
m_resources[row].reset(new_resource);
|
||||||
resolveResource(m_resources.at(row));
|
resolveResource(m_resources.at(row).get());
|
||||||
emit dataChanged(index(row, 0), index(row, columnCount(QModelIndex()) - 1));
|
emit dataChanged(index(row, 0), index(row, columnCount(QModelIndex()) - 1));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -313,7 +313,7 @@ void ResourceFolderModel::applyUpdates(QSet<QString>& current_set, QSet<QString>
|
|||||||
for (auto& added : added_set) {
|
for (auto& added : added_set) {
|
||||||
auto res = new_resources[added];
|
auto res = new_resources[added];
|
||||||
m_resources.append(res);
|
m_resources.append(res);
|
||||||
resolveResource(res);
|
resolveResource(m_resources.last().get());
|
||||||
}
|
}
|
||||||
|
|
||||||
endInsertRows();
|
endInsertRows();
|
||||||
@ -324,7 +324,7 @@ void ResourceFolderModel::applyUpdates(QSet<QString>& current_set, QSet<QString>
|
|||||||
{
|
{
|
||||||
m_resources_index.clear();
|
m_resources_index.clear();
|
||||||
int idx = 0;
|
int idx = 0;
|
||||||
for (auto const& mod : m_resources) {
|
for (auto const& mod : qAsConst(m_resources)) {
|
||||||
m_resources_index[mod->internal_id()] = idx;
|
m_resources_index[mod->internal_id()] = idx;
|
||||||
idx++;
|
idx++;
|
||||||
}
|
}
|
||||||
|
@ -3,6 +3,7 @@
|
|||||||
#include <QDir>
|
#include <QDir>
|
||||||
#include <QMap>
|
#include <QMap>
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
|
#include <QThread>
|
||||||
|
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
|
||||||
@ -23,14 +24,14 @@ class BasicFolderLoadTask : public Task {
|
|||||||
[[nodiscard]] ResultPtr result() const { return m_result; }
|
[[nodiscard]] ResultPtr result() const { return m_result; }
|
||||||
|
|
||||||
public:
|
public:
|
||||||
BasicFolderLoadTask(QDir dir) : Task(nullptr, false), m_dir(dir), m_result(new Result)
|
BasicFolderLoadTask(QDir dir) : Task(nullptr, false), m_dir(dir), m_result(new Result), m_thread_to_spawn_into(thread())
|
||||||
{
|
{
|
||||||
m_create_func = [](QFileInfo const& entry) -> Resource* {
|
m_create_func = [](QFileInfo const& entry) -> Resource* {
|
||||||
return new Resource(entry);
|
return new Resource(entry);
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
BasicFolderLoadTask(QDir dir, std::function<Resource*(QFileInfo const&)> create_function)
|
BasicFolderLoadTask(QDir dir, std::function<Resource*(QFileInfo const&)> create_function)
|
||||||
: Task(nullptr, false), m_dir(dir), m_result(new Result), m_create_func(std::move(create_function))
|
: Task(nullptr, false), m_dir(dir), m_result(new Result), m_create_func(std::move(create_function)), m_thread_to_spawn_into(thread())
|
||||||
{}
|
{}
|
||||||
|
|
||||||
[[nodiscard]] bool canAbort() const override { return true; }
|
[[nodiscard]] bool canAbort() const override { return true; }
|
||||||
@ -42,9 +43,13 @@ class BasicFolderLoadTask : public Task {
|
|||||||
|
|
||||||
void executeTask() override
|
void executeTask() override
|
||||||
{
|
{
|
||||||
|
if (thread() != m_thread_to_spawn_into)
|
||||||
|
connect(this, &Task::finished, this->thread(), &QThread::quit);
|
||||||
|
|
||||||
m_dir.refresh();
|
m_dir.refresh();
|
||||||
for (auto entry : m_dir.entryInfoList()) {
|
for (auto entry : m_dir.entryInfoList()) {
|
||||||
auto resource = m_create_func(entry);
|
auto resource = m_create_func(entry);
|
||||||
|
resource->moveToThread(m_thread_to_spawn_into);
|
||||||
m_result->resources.insert(resource->internal_id(), resource);
|
m_result->resources.insert(resource->internal_id(), resource);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -61,4 +66,7 @@ private:
|
|||||||
std::atomic<bool> m_aborted = false;
|
std::atomic<bool> m_aborted = false;
|
||||||
|
|
||||||
std::function<Resource*(QFileInfo const&)> m_create_func;
|
std::function<Resource*(QFileInfo const&)> m_create_func;
|
||||||
|
|
||||||
|
/** This is the thread in which we should put new mod objects */
|
||||||
|
QThread* m_thread_to_spawn_into;
|
||||||
};
|
};
|
||||||
|
@ -38,12 +38,23 @@
|
|||||||
|
|
||||||
#include "minecraft/mod/MetadataHandler.h"
|
#include "minecraft/mod/MetadataHandler.h"
|
||||||
|
|
||||||
|
#include <QThread>
|
||||||
|
|
||||||
ModFolderLoadTask::ModFolderLoadTask(QDir mods_dir, QDir index_dir, bool is_indexed, bool clean_orphan)
|
ModFolderLoadTask::ModFolderLoadTask(QDir mods_dir, QDir index_dir, bool is_indexed, bool clean_orphan)
|
||||||
: Task(nullptr, false), m_mods_dir(mods_dir), m_index_dir(index_dir), m_is_indexed(is_indexed), m_clean_orphan(clean_orphan), m_result(new Result())
|
: Task(nullptr, false)
|
||||||
|
, m_mods_dir(mods_dir)
|
||||||
|
, m_index_dir(index_dir)
|
||||||
|
, m_is_indexed(is_indexed)
|
||||||
|
, m_clean_orphan(clean_orphan)
|
||||||
|
, m_result(new Result())
|
||||||
|
, m_thread_to_spawn_into(thread())
|
||||||
{}
|
{}
|
||||||
|
|
||||||
void ModFolderLoadTask::executeTask()
|
void ModFolderLoadTask::executeTask()
|
||||||
{
|
{
|
||||||
|
if (thread() != m_thread_to_spawn_into)
|
||||||
|
connect(this, &Task::finished, this->thread(), &QThread::quit);
|
||||||
|
|
||||||
if (m_is_indexed) {
|
if (m_is_indexed) {
|
||||||
// Read metadata first
|
// Read metadata first
|
||||||
getFromMetadata();
|
getFromMetadata();
|
||||||
@ -57,6 +68,8 @@ void ModFolderLoadTask::executeTask()
|
|||||||
if (mod->enabled()) {
|
if (mod->enabled()) {
|
||||||
if (m_result->mods.contains(mod->internal_id())) {
|
if (m_result->mods.contains(mod->internal_id())) {
|
||||||
m_result->mods[mod->internal_id()]->setStatus(ModStatus::Installed);
|
m_result->mods[mod->internal_id()]->setStatus(ModStatus::Installed);
|
||||||
|
// Delete the object we just created, since a valid one is already in the mods list.
|
||||||
|
delete mod;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
m_result->mods[mod->internal_id()] = mod;
|
m_result->mods[mod->internal_id()] = mod;
|
||||||
@ -86,7 +99,7 @@ void ModFolderLoadTask::executeTask()
|
|||||||
// Remove orphan metadata to prevent issues
|
// Remove orphan metadata to prevent issues
|
||||||
// See https://github.com/PolyMC/PolyMC/issues/996
|
// See https://github.com/PolyMC/PolyMC/issues/996
|
||||||
if (m_clean_orphan) {
|
if (m_clean_orphan) {
|
||||||
QMutableMapIterator<QString, Mod::Ptr> iter(m_result->mods);
|
QMutableMapIterator iter(m_result->mods);
|
||||||
while (iter.hasNext()) {
|
while (iter.hasNext()) {
|
||||||
auto mod = iter.next().value();
|
auto mod = iter.next().value();
|
||||||
if (mod->status() == ModStatus::NotInstalled) {
|
if (mod->status() == ModStatus::NotInstalled) {
|
||||||
@ -96,6 +109,9 @@ void ModFolderLoadTask::executeTask()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for (auto mod : m_result->mods)
|
||||||
|
mod->moveToThread(m_thread_to_spawn_into);
|
||||||
|
|
||||||
if (m_aborted)
|
if (m_aborted)
|
||||||
emit finished();
|
emit finished();
|
||||||
else
|
else
|
||||||
|
@ -79,4 +79,7 @@ private:
|
|||||||
ResultPtr m_result;
|
ResultPtr m_result;
|
||||||
|
|
||||||
std::atomic<bool> m_aborted = false;
|
std::atomic<bool> m_aborted = false;
|
||||||
|
|
||||||
|
/** This is the thread in which we should put new mod objects */
|
||||||
|
QThread* m_thread_to_spawn_into;
|
||||||
};
|
};
|
||||||
|
Loading…
x
Reference in New Issue
Block a user