2014-04-24 15:09:25 +02:00
|
|
|
#include "loader.hpp"
|
|
|
|
|
2022-10-19 19:02:00 +02:00
|
|
|
#include <algorithm>
|
|
|
|
#include <exception>
|
|
|
|
#include <filesystem>
|
|
|
|
|
|
|
|
#include <apps/opencs/model/doc/messages.hpp>
|
|
|
|
#include <apps/opencs/model/world/data.hpp>
|
|
|
|
#include <apps/opencs/model/world/universalid.hpp>
|
|
|
|
|
2023-02-15 20:56:32 +01:00
|
|
|
#include <components/debug/debuglog.hpp>
|
2022-07-03 00:02:29 +02:00
|
|
|
#include <components/files/conversion.hpp>
|
|
|
|
|
2022-10-09 10:39:43 +00:00
|
|
|
#include <QTimer>
|
2014-04-24 15:09:25 +02:00
|
|
|
|
2014-05-10 12:04:36 +02:00
|
|
|
#include "../tools/reportmodel.hpp"
|
|
|
|
|
2014-04-24 15:09:25 +02:00
|
|
|
#include "document.hpp"
|
|
|
|
|
2014-06-26 11:41:21 +02:00
|
|
|
CSMDoc::Loader::Stage::Stage()
|
|
|
|
: mFile(0)
|
|
|
|
, mRecordsLoaded(0)
|
|
|
|
, mRecordsLeft(false)
|
|
|
|
{
|
|
|
|
}
|
2014-05-03 12:07:05 +02:00
|
|
|
|
2014-04-24 15:09:25 +02:00
|
|
|
CSMDoc::Loader::Loader()
|
2016-06-10 12:10:14 -04:00
|
|
|
: mShouldStop(false)
|
2014-04-24 15:09:25 +02:00
|
|
|
{
|
2016-06-10 12:10:14 -04:00
|
|
|
mTimer = new QTimer(this);
|
2014-04-24 15:09:25 +02:00
|
|
|
|
2022-08-22 23:28:58 -03:00
|
|
|
connect(mTimer, &QTimer::timeout, this, &Loader::load);
|
2016-06-10 12:10:14 -04:00
|
|
|
mTimer->start();
|
2014-04-24 15:09:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
QWaitCondition& CSMDoc::Loader::hasThingsToDo()
|
|
|
|
{
|
|
|
|
return mThingsToDo;
|
|
|
|
}
|
|
|
|
|
2016-06-10 12:10:14 -04:00
|
|
|
void CSMDoc::Loader::stop()
|
|
|
|
{
|
|
|
|
mShouldStop = true;
|
|
|
|
}
|
|
|
|
|
2014-04-24 15:09:25 +02:00
|
|
|
void CSMDoc::Loader::load()
|
|
|
|
{
|
|
|
|
if (mDocuments.empty())
|
|
|
|
{
|
|
|
|
mMutex.lock();
|
|
|
|
mThingsToDo.wait(&mMutex);
|
|
|
|
mMutex.unlock();
|
2016-06-10 12:10:14 -04:00
|
|
|
|
|
|
|
if (mShouldStop)
|
|
|
|
mTimer->stop();
|
|
|
|
|
2014-04-24 15:09:25 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-02-15 20:56:32 +01:00
|
|
|
if (!mStart.has_value())
|
|
|
|
mStart = std::chrono::steady_clock::now();
|
|
|
|
|
2014-05-03 12:07:05 +02:00
|
|
|
std::vector<std::pair<Document*, Stage>>::iterator iter = mDocuments.begin();
|
2014-04-24 15:09:25 +02:00
|
|
|
|
|
|
|
Document* document = iter->first;
|
|
|
|
|
2014-05-03 12:07:05 +02:00
|
|
|
int size = static_cast<int>(document->getContentFiles().size());
|
2014-06-17 11:01:17 +02:00
|
|
|
int editedIndex = size - 1; // index of the file to be edited/created
|
2014-05-03 12:07:05 +02:00
|
|
|
|
|
|
|
if (document->isNew())
|
|
|
|
--size;
|
|
|
|
|
|
|
|
bool done = false;
|
2014-04-24 15:09:25 +02:00
|
|
|
|
|
|
|
try
|
|
|
|
{
|
2014-05-03 15:05:02 +02:00
|
|
|
if (iter->second.mRecordsLeft)
|
|
|
|
{
|
2015-06-20 19:04:19 +02:00
|
|
|
Messages messages(Message::Severity_Error);
|
2016-01-03 18:20:34 +01:00
|
|
|
const int batchingSize = 50;
|
2014-05-03 15:33:35 +02:00
|
|
|
for (int i = 0; i < batchingSize; ++i) // do not flood the system with update signals
|
2014-05-10 12:04:36 +02:00
|
|
|
if (document->getData().continueLoading(messages))
|
2014-05-03 15:33:35 +02:00
|
|
|
{
|
|
|
|
iter->second.mRecordsLeft = false;
|
|
|
|
break;
|
|
|
|
}
|
2014-06-26 11:41:21 +02:00
|
|
|
else
|
|
|
|
++(iter->second.mRecordsLoaded);
|
2014-05-03 15:33:35 +02:00
|
|
|
|
2014-05-10 12:04:36 +02:00
|
|
|
CSMWorld::UniversalId log(CSMWorld::UniversalId::Type_LoadErrorLog, 0);
|
|
|
|
|
2014-06-26 11:41:21 +02:00
|
|
|
{ // silence a g++ warning
|
2016-10-16 01:34:54 +09:00
|
|
|
for (CSMDoc::Messages::Iterator messageIter(messages.begin()); messageIter != messages.end();
|
|
|
|
++messageIter)
|
2014-05-10 13:18:40 +02:00
|
|
|
{
|
2016-10-16 01:34:54 +09:00
|
|
|
document->getReport(log)->add(*messageIter);
|
|
|
|
emit loadMessage(document, messageIter->mMessage);
|
2014-05-10 13:18:40 +02:00
|
|
|
}
|
2014-06-26 11:41:21 +02:00
|
|
|
}
|
2014-05-10 12:04:36 +02:00
|
|
|
|
2014-06-26 11:41:21 +02:00
|
|
|
emit nextRecord(document, iter->second.mRecordsLoaded);
|
2014-05-03 15:05:02 +02:00
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-07-23 17:34:27 +10:00
|
|
|
if (iter->second.mFile < size) // start loading the files
|
2014-05-03 12:07:05 +02:00
|
|
|
{
|
2022-06-08 23:25:50 +02:00
|
|
|
std::filesystem::path path = document->getContentFiles()[iter->second.mFile];
|
2014-05-03 13:01:29 +02:00
|
|
|
|
2021-07-23 17:34:27 +10:00
|
|
|
int steps = document->getData().startLoading(path, iter->second.mFile != editedIndex, /*project*/ false);
|
2014-05-03 15:05:02 +02:00
|
|
|
iter->second.mRecordsLeft = true;
|
2014-06-26 11:41:21 +02:00
|
|
|
iter->second.mRecordsLoaded = 0;
|
2014-05-03 15:33:35 +02:00
|
|
|
|
2022-07-03 00:02:29 +02:00
|
|
|
emit nextStage(document, Files::pathToUnicodeString(path.filename()), steps);
|
2014-05-03 12:07:05 +02:00
|
|
|
}
|
2021-07-23 17:34:27 +10:00
|
|
|
else if (iter->second.mFile == size) // start loading the last (project) file
|
2014-05-03 12:07:05 +02:00
|
|
|
{
|
2021-07-23 17:34:27 +10:00
|
|
|
int steps = document->getData().startLoading(document->getProjectPath(), /*base*/ false, true);
|
2014-05-03 15:05:02 +02:00
|
|
|
iter->second.mRecordsLeft = true;
|
2014-06-26 11:41:21 +02:00
|
|
|
iter->second.mRecordsLoaded = 0;
|
2014-05-03 15:33:35 +02:00
|
|
|
|
2014-06-26 11:41:21 +02:00
|
|
|
emit nextStage(document, "Project File", steps);
|
2014-05-03 12:07:05 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-03-08 02:17:15 +01:00
|
|
|
document->getData().finishLoading();
|
2014-05-03 12:07:05 +02:00
|
|
|
done = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
++(iter->second.mFile);
|
2014-04-24 15:09:25 +02:00
|
|
|
}
|
|
|
|
catch (const std::exception& e)
|
|
|
|
{
|
2014-05-03 12:07:05 +02:00
|
|
|
mDocuments.erase(iter);
|
2014-04-24 15:09:25 +02:00
|
|
|
emit documentNotLoaded(document, e.what());
|
2014-05-03 12:07:05 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (done)
|
|
|
|
{
|
2023-02-15 20:56:32 +01:00
|
|
|
if (mStart.has_value())
|
|
|
|
{
|
|
|
|
const auto duration = std::chrono::steady_clock::now() - *mStart;
|
|
|
|
Log(Debug::Verbose) << "Loaded content files in "
|
|
|
|
<< std::chrono::duration_cast<std::chrono::duration<double>>(duration).count() << 's';
|
|
|
|
mStart.reset();
|
|
|
|
}
|
|
|
|
|
2014-05-03 12:07:05 +02:00
|
|
|
mDocuments.erase(iter);
|
|
|
|
emit documentLoaded(document);
|
2014-04-24 15:09:25 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-29 14:27:44 +02:00
|
|
|
void CSMDoc::Loader::loadDocument(CSMDoc::Document* document)
|
2014-04-24 15:09:25 +02:00
|
|
|
{
|
2020-10-17 12:26:35 +04:00
|
|
|
mDocuments.emplace_back(document, Stage());
|
2014-04-24 15:09:25 +02:00
|
|
|
}
|
|
|
|
|
2014-05-03 16:44:50 +02:00
|
|
|
void CSMDoc::Loader::abortLoading(CSMDoc::Document* document)
|
2014-04-24 15:09:25 +02:00
|
|
|
{
|
2014-05-03 12:07:05 +02:00
|
|
|
for (std::vector<std::pair<Document*, Stage>>::iterator iter = mDocuments.begin(); iter != mDocuments.end(); ++iter)
|
2014-04-24 15:09:25 +02:00
|
|
|
{
|
|
|
|
if (iter->first == document)
|
|
|
|
{
|
|
|
|
mDocuments.erase(iter);
|
|
|
|
emit documentNotLoaded(document, "");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2014-05-03 16:44:50 +02:00
|
|
|
}
|