2012-11-21 16:31:18 +00:00
|
|
|
#include "document.hpp"
|
2013-09-27 13:04:30 +00:00
|
|
|
|
2022-08-19 17:19:42 +00:00
|
|
|
#include "state.hpp"
|
|
|
|
|
2022-10-19 17:02:00 +00:00
|
|
|
#include <apps/opencs/model/doc/blacklist.hpp>
|
|
|
|
#include <apps/opencs/model/doc/messages.hpp>
|
|
|
|
#include <apps/opencs/model/doc/operationholder.hpp>
|
|
|
|
#include <apps/opencs/model/doc/runner.hpp>
|
|
|
|
#include <apps/opencs/model/doc/saving.hpp>
|
|
|
|
#include <apps/opencs/model/tools/tools.hpp>
|
|
|
|
#include <apps/opencs/model/world/data.hpp>
|
|
|
|
#include <apps/opencs/model/world/idcollection.hpp>
|
|
|
|
#include <apps/opencs/model/world/record.hpp>
|
|
|
|
|
|
|
|
#include <components/debug/debuglog.hpp>
|
|
|
|
#include <components/esm3/loaddial.hpp>
|
|
|
|
#include <components/esm3/loadglob.hpp>
|
|
|
|
#include <components/esm3/loadgmst.hpp>
|
|
|
|
#include <components/esm3/loadmgef.hpp>
|
|
|
|
#include <components/esm3/loadskil.hpp>
|
|
|
|
#include <components/esm3/variant.hpp>
|
|
|
|
#include <components/files/conversion.hpp>
|
|
|
|
|
|
|
|
#include <algorithm>
|
2022-06-08 21:25:50 +00:00
|
|
|
#include <filesystem>
|
2022-10-19 17:02:00 +00:00
|
|
|
#include <fstream>
|
2021-07-23 04:21:21 +00:00
|
|
|
#include <memory>
|
2022-10-19 17:02:00 +00:00
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdexcept>
|
2022-07-02 22:02:29 +00:00
|
|
|
#include <utility>
|
2022-10-19 17:02:00 +00:00
|
|
|
#include <variant>
|
2013-09-27 13:04:30 +00:00
|
|
|
|
2016-01-18 02:55:03 +00:00
|
|
|
#include "../world/defaultgmsts.hpp"
|
|
|
|
|
2013-09-27 13:24:58 +00:00
|
|
|
#ifndef Q_MOC_RUN
|
|
|
|
#include <components/files/configurationmanager.hpp>
|
|
|
|
#endif
|
|
|
|
|
2022-10-19 17:02:00 +00:00
|
|
|
namespace CSMWorld
|
|
|
|
{
|
|
|
|
class IdCompletionManager;
|
|
|
|
}
|
2018-08-14 16:01:09 +00:00
|
|
|
|
2013-06-23 19:19:08 +00:00
|
|
|
void CSMDoc::Document::addGmsts()
|
2013-06-23 01:01:27 +00:00
|
|
|
{
|
2016-01-18 17:34:33 +00:00
|
|
|
for (size_t i = 0; i < CSMWorld::DefaultGmsts::FloatCount; ++i)
|
2013-06-23 01:01:27 +00:00
|
|
|
{
|
|
|
|
ESM::GameSetting gmst;
|
2022-10-06 17:39:46 +00:00
|
|
|
gmst.mId = ESM::RefId::stringRefId(CSMWorld::DefaultGmsts::Floats[i]);
|
2013-06-23 01:01:27 +00:00
|
|
|
gmst.mValue.setType(ESM::VT_Float);
|
2022-07-02 19:25:51 +00:00
|
|
|
gmst.mRecordFlags = 0;
|
2016-01-18 17:34:33 +00:00
|
|
|
gmst.mValue.setFloat(CSMWorld::DefaultGmsts::FloatsDefaultValues[i]);
|
2013-06-23 19:19:08 +00:00
|
|
|
getData().getGmsts().add(gmst);
|
2013-06-23 01:01:27 +00:00
|
|
|
}
|
|
|
|
|
2016-01-18 17:34:33 +00:00
|
|
|
for (size_t i = 0; i < CSMWorld::DefaultGmsts::IntCount; ++i)
|
2013-06-23 01:01:27 +00:00
|
|
|
{
|
|
|
|
ESM::GameSetting gmst;
|
2022-10-06 17:39:46 +00:00
|
|
|
gmst.mId = ESM::RefId::stringRefId(CSMWorld::DefaultGmsts::Ints[i]);
|
2013-06-23 01:01:27 +00:00
|
|
|
gmst.mValue.setType(ESM::VT_Int);
|
2022-07-02 19:25:51 +00:00
|
|
|
gmst.mRecordFlags = 0;
|
2016-01-18 17:34:33 +00:00
|
|
|
gmst.mValue.setInteger(CSMWorld::DefaultGmsts::IntsDefaultValues[i]);
|
2013-06-23 19:19:08 +00:00
|
|
|
getData().getGmsts().add(gmst);
|
2013-06-23 01:01:27 +00:00
|
|
|
}
|
|
|
|
|
2016-01-18 17:34:33 +00:00
|
|
|
for (size_t i = 0; i < CSMWorld::DefaultGmsts::StringCount; ++i)
|
2013-06-23 01:01:27 +00:00
|
|
|
{
|
|
|
|
ESM::GameSetting gmst;
|
2022-10-06 17:39:46 +00:00
|
|
|
gmst.mId = ESM::RefId::stringRefId(CSMWorld::DefaultGmsts::Strings[i]);
|
2013-06-23 01:01:27 +00:00
|
|
|
gmst.mValue.setType(ESM::VT_String);
|
2022-07-02 19:25:51 +00:00
|
|
|
gmst.mRecordFlags = 0;
|
2013-06-23 01:01:27 +00:00
|
|
|
gmst.mValue.setString("");
|
2013-06-23 19:19:08 +00:00
|
|
|
getData().getGmsts().add(gmst);
|
2013-06-23 01:01:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-17 19:03:39 +00:00
|
|
|
void CSMDoc::Document::addOptionalGmsts()
|
|
|
|
{
|
2016-01-18 17:34:33 +00:00
|
|
|
for (size_t i = 0; i < CSMWorld::DefaultGmsts::OptionalFloatCount; ++i)
|
2013-02-17 19:03:39 +00:00
|
|
|
{
|
|
|
|
ESM::GameSetting gmst;
|
2022-10-06 17:39:46 +00:00
|
|
|
gmst.mId = ESM::RefId::stringRefId(CSMWorld::DefaultGmsts::OptionalFloats[i]);
|
2014-09-27 10:38:42 +00:00
|
|
|
gmst.blank();
|
2013-03-05 07:02:27 +00:00
|
|
|
gmst.mValue.setType(ESM::VT_Float);
|
2013-02-17 19:03:39 +00:00
|
|
|
addOptionalGmst(gmst);
|
|
|
|
}
|
|
|
|
|
2016-01-18 17:34:33 +00:00
|
|
|
for (size_t i = 0; i < CSMWorld::DefaultGmsts::OptionalIntCount; ++i)
|
2013-02-17 19:03:39 +00:00
|
|
|
{
|
|
|
|
ESM::GameSetting gmst;
|
2022-10-06 17:39:46 +00:00
|
|
|
gmst.mId = ESM::RefId::stringRefId(CSMWorld::DefaultGmsts::OptionalInts[i]);
|
2014-09-27 10:38:42 +00:00
|
|
|
gmst.blank();
|
2013-03-05 07:02:27 +00:00
|
|
|
gmst.mValue.setType(ESM::VT_Int);
|
2013-02-17 19:03:39 +00:00
|
|
|
addOptionalGmst(gmst);
|
|
|
|
}
|
|
|
|
|
2016-01-18 17:34:33 +00:00
|
|
|
for (size_t i = 0; i < CSMWorld::DefaultGmsts::OptionalStringCount; ++i)
|
2013-02-17 19:03:39 +00:00
|
|
|
{
|
|
|
|
ESM::GameSetting gmst;
|
2022-10-06 17:39:46 +00:00
|
|
|
gmst.mId = ESM::RefId::stringRefId(CSMWorld::DefaultGmsts::OptionalStrings[i]);
|
2014-09-27 10:38:42 +00:00
|
|
|
gmst.blank();
|
2013-03-05 07:02:27 +00:00
|
|
|
gmst.mValue.setType(ESM::VT_String);
|
|
|
|
gmst.mValue.setString("<no text>");
|
2013-02-17 19:03:39 +00:00
|
|
|
addOptionalGmst(gmst);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-23 11:51:45 +00:00
|
|
|
void CSMDoc::Document::addOptionalGlobals()
|
|
|
|
{
|
|
|
|
static const char* sGlobals[] = {
|
2013-09-16 10:51:57 +00:00
|
|
|
"DaysPassed",
|
|
|
|
"PCWerewolf",
|
|
|
|
"PCYear",
|
2022-09-13 22:08:19 +00:00
|
|
|
nullptr,
|
2013-02-23 11:51:45 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
for (int i = 0; sGlobals[i]; ++i)
|
|
|
|
{
|
|
|
|
ESM::Global global;
|
2022-10-06 17:39:46 +00:00
|
|
|
global.mId = ESM::RefId::stringRefId(sGlobals[i]);
|
2014-09-27 10:38:42 +00:00
|
|
|
global.blank();
|
2013-03-05 10:37:13 +00:00
|
|
|
global.mValue.setType(ESM::VT_Long);
|
2013-03-25 10:59:25 +00:00
|
|
|
|
|
|
|
if (i == 0)
|
|
|
|
global.mValue.setInteger(1); // dayspassed starts counting at 1
|
|
|
|
|
2013-02-23 11:51:45 +00:00
|
|
|
addOptionalGlobal(global);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-27 10:38:42 +00:00
|
|
|
void CSMDoc::Document::addOptionalMagicEffects()
|
|
|
|
{
|
|
|
|
for (int i = ESM::MagicEffect::SummonFabricant; i <= ESM::MagicEffect::SummonCreature05; ++i)
|
|
|
|
{
|
|
|
|
ESM::MagicEffect effect;
|
|
|
|
effect.mIndex = i;
|
2022-10-06 17:39:46 +00:00
|
|
|
effect.mId = ESM::RefId::stringRefId(ESM::MagicEffect::indexToId(i));
|
2014-09-27 10:38:42 +00:00
|
|
|
effect.blank();
|
|
|
|
|
|
|
|
addOptionalMagicEffect(effect);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-17 19:03:39 +00:00
|
|
|
void CSMDoc::Document::addOptionalGmst(const ESM::GameSetting& gmst)
|
|
|
|
{
|
|
|
|
if (getData().getGmsts().searchId(gmst.mId) == -1)
|
|
|
|
{
|
2022-05-29 11:25:17 +00:00
|
|
|
auto record = std::make_unique<CSMWorld::Record<ESM::GameSetting>>();
|
2021-07-23 04:21:21 +00:00
|
|
|
record->mBase = gmst;
|
|
|
|
record->mState = CSMWorld::RecordBase::State_BaseOnly;
|
|
|
|
getData().getGmsts().appendRecord(std::move(record));
|
2013-02-17 19:03:39 +00:00
|
|
|
}
|
2013-02-04 12:46:54 +00:00
|
|
|
}
|
|
|
|
|
2013-02-23 11:51:45 +00:00
|
|
|
void CSMDoc::Document::addOptionalGlobal(const ESM::Global& global)
|
|
|
|
{
|
|
|
|
if (getData().getGlobals().searchId(global.mId) == -1)
|
|
|
|
{
|
2022-05-29 11:25:17 +00:00
|
|
|
auto record = std::make_unique<CSMWorld::Record<ESM::Global>>();
|
2021-07-23 04:21:21 +00:00
|
|
|
record->mBase = global;
|
|
|
|
record->mState = CSMWorld::RecordBase::State_BaseOnly;
|
|
|
|
getData().getGlobals().appendRecord(std::move(record));
|
2013-02-23 11:51:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-27 10:38:42 +00:00
|
|
|
void CSMDoc::Document::addOptionalMagicEffect(const ESM::MagicEffect& magicEffect)
|
|
|
|
{
|
|
|
|
if (getData().getMagicEffects().searchId(magicEffect.mId) == -1)
|
|
|
|
{
|
2022-05-29 11:25:17 +00:00
|
|
|
auto record = std::make_unique<CSMWorld::Record<ESM::MagicEffect>>();
|
2021-07-23 04:21:21 +00:00
|
|
|
record->mBase = magicEffect;
|
|
|
|
record->mState = CSMWorld::RecordBase::State_BaseOnly;
|
|
|
|
getData().getMagicEffects().appendRecord(std::move(record));
|
2014-09-27 10:38:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-04 12:46:54 +00:00
|
|
|
void CSMDoc::Document::createBase()
|
|
|
|
{
|
2013-02-04 12:50:38 +00:00
|
|
|
static const char* sGlobals[] = {
|
2013-06-23 01:01:27 +00:00
|
|
|
"Day",
|
|
|
|
"DaysPassed",
|
|
|
|
"GameHour",
|
|
|
|
"Month",
|
|
|
|
"PCRace",
|
|
|
|
"PCVampire",
|
|
|
|
"PCWerewolf",
|
|
|
|
"PCYear",
|
2022-09-13 22:08:19 +00:00
|
|
|
nullptr,
|
2013-02-04 12:50:38 +00:00
|
|
|
};
|
2013-02-04 12:46:54 +00:00
|
|
|
|
2013-02-04 12:50:38 +00:00
|
|
|
for (int i = 0; sGlobals[i]; ++i)
|
|
|
|
{
|
|
|
|
ESM::Global record;
|
2022-10-06 17:39:46 +00:00
|
|
|
record.mId = ESM::RefId::stringRefId(sGlobals[i]);
|
2022-07-02 19:25:51 +00:00
|
|
|
record.mRecordFlags = 0;
|
2013-03-25 10:59:25 +00:00
|
|
|
record.mValue.setType(i == 2 ? ESM::VT_Float : ESM::VT_Long);
|
2013-03-04 13:32:59 +00:00
|
|
|
|
2013-03-25 10:59:25 +00:00
|
|
|
if (i == 0 || i == 1)
|
2013-03-04 13:32:59 +00:00
|
|
|
record.mValue.setInteger(1);
|
|
|
|
|
2013-02-04 12:50:38 +00:00
|
|
|
getData().getGlobals().add(record);
|
|
|
|
}
|
2013-03-07 06:56:29 +00:00
|
|
|
|
2013-06-23 19:19:08 +00:00
|
|
|
addGmsts();
|
2013-03-24 14:10:03 +00:00
|
|
|
|
|
|
|
for (int i = 0; i < 27; ++i)
|
|
|
|
{
|
|
|
|
ESM::Skill record;
|
|
|
|
record.mIndex = i;
|
2022-10-06 17:39:46 +00:00
|
|
|
record.mId = ESM::RefId::stringRefId(ESM::Skill::indexToId(record.mIndex));
|
2013-03-24 14:10:03 +00:00
|
|
|
record.blank();
|
|
|
|
|
|
|
|
getData().getSkills().add(record);
|
|
|
|
}
|
2013-10-21 12:26:54 +00:00
|
|
|
|
|
|
|
static const char* sVoice[] = {
|
|
|
|
"Intruder",
|
|
|
|
"Attack",
|
|
|
|
"Hello",
|
|
|
|
"Thief",
|
|
|
|
"Alarm",
|
|
|
|
"Idle",
|
|
|
|
"Flee",
|
|
|
|
"Hit",
|
2022-09-13 22:08:19 +00:00
|
|
|
nullptr,
|
2013-10-21 12:26:54 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
for (int i = 0; sVoice[i]; ++i)
|
|
|
|
{
|
|
|
|
ESM::Dialogue record;
|
2022-10-06 17:39:46 +00:00
|
|
|
record.mId = ESM::RefId::stringRefId(sVoice[i]);
|
2013-10-21 12:26:54 +00:00
|
|
|
record.mType = ESM::Dialogue::Voice;
|
|
|
|
record.blank();
|
|
|
|
|
|
|
|
getData().getTopics().add(record);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char* sGreetings[] = {
|
|
|
|
"Greeting 0",
|
|
|
|
"Greeting 1",
|
|
|
|
"Greeting 2",
|
|
|
|
"Greeting 3",
|
|
|
|
"Greeting 4",
|
|
|
|
"Greeting 5",
|
|
|
|
"Greeting 6",
|
|
|
|
"Greeting 7",
|
|
|
|
"Greeting 8",
|
|
|
|
"Greeting 9",
|
2022-09-13 22:08:19 +00:00
|
|
|
nullptr,
|
2013-10-21 12:26:54 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
for (int i = 0; sGreetings[i]; ++i)
|
|
|
|
{
|
|
|
|
ESM::Dialogue record;
|
2022-10-06 17:39:46 +00:00
|
|
|
record.mId = ESM::RefId::stringRefId(sGreetings[i]);
|
2013-10-21 12:26:54 +00:00
|
|
|
record.mType = ESM::Dialogue::Greeting;
|
|
|
|
record.blank();
|
|
|
|
|
|
|
|
getData().getTopics().add(record);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char* sPersuasion[] = {
|
|
|
|
"Intimidate Success",
|
|
|
|
"Intimidate Fail",
|
|
|
|
"Service Refusal",
|
|
|
|
"Admire Success",
|
|
|
|
"Taunt Success",
|
|
|
|
"Bribe Success",
|
|
|
|
"Info Refusal",
|
|
|
|
"Admire Fail",
|
|
|
|
"Taunt Fail",
|
|
|
|
"Bribe Fail",
|
2022-09-13 22:08:19 +00:00
|
|
|
nullptr,
|
2013-10-21 12:26:54 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
for (int i = 0; sPersuasion[i]; ++i)
|
|
|
|
{
|
|
|
|
ESM::Dialogue record;
|
2022-10-06 17:39:46 +00:00
|
|
|
record.mId = ESM::RefId::stringRefId(sPersuasion[i]);
|
2013-10-21 12:26:54 +00:00
|
|
|
record.mType = ESM::Dialogue::Persuasion;
|
|
|
|
record.blank();
|
|
|
|
|
|
|
|
getData().getTopics().add(record);
|
|
|
|
}
|
2014-09-27 10:26:25 +00:00
|
|
|
|
|
|
|
for (int i = 0; i < ESM::MagicEffect::Length; ++i)
|
|
|
|
{
|
|
|
|
ESM::MagicEffect record;
|
|
|
|
|
|
|
|
record.mIndex = i;
|
2022-10-06 17:39:46 +00:00
|
|
|
record.mId = ESM::RefId::stringRefId(ESM::MagicEffect::indexToId(i));
|
2014-09-27 10:26:25 +00:00
|
|
|
|
|
|
|
record.blank();
|
|
|
|
|
|
|
|
getData().getMagicEffects().add(record);
|
|
|
|
}
|
2013-02-04 12:46:54 +00:00
|
|
|
}
|
|
|
|
|
2017-08-22 02:31:19 +00:00
|
|
|
CSMDoc::Document::Document(const Files::ConfigurationManager& configuration, std::vector<std::filesystem::path> files,
|
2022-06-08 21:25:50 +00:00
|
|
|
bool new_, const std::filesystem::path& savePath, const std::filesystem::path& resDir, ToUTF8::FromType encoding,
|
2017-08-22 02:31:19 +00:00
|
|
|
const std::vector<std::string>& blacklistedScripts, bool fsStrict, const Files::PathContainer& dataPaths,
|
|
|
|
const std::vector<std::string>& archives)
|
2022-06-08 21:25:50 +00:00
|
|
|
: mSavePath(savePath)
|
|
|
|
, mContentFiles(std::move(files))
|
|
|
|
, mNew(new_)
|
|
|
|
, mData(encoding, fsStrict, dataPaths, archives, resDir)
|
2015-08-16 16:27:17 +00:00
|
|
|
, mTools(*this, encoding)
|
2014-04-24 13:09:25 +00:00
|
|
|
, mProjectPath((configuration.getUserDataPath() / "projects") / (savePath.filename().u8string() + u8".project"))
|
2015-03-14 11:00:24 +00:00
|
|
|
, mSavingOperation(*this, mProjectPath, encoding)
|
|
|
|
, mSaving(&mSavingOperation)
|
2019-01-22 06:08:48 +00:00
|
|
|
, mResDir(resDir)
|
|
|
|
, mRunner(mProjectPath)
|
|
|
|
, mDirty(false)
|
|
|
|
, mIdCompletionManager(mData)
|
2012-11-22 22:42:17 +00:00
|
|
|
{
|
2014-04-24 13:09:25 +00:00
|
|
|
if (mContentFiles.empty())
|
2013-02-04 12:46:54 +00:00
|
|
|
throw std::runtime_error("Empty content file sequence");
|
|
|
|
|
2022-06-08 21:25:50 +00:00
|
|
|
if (mNew || !std::filesystem::exists(mProjectPath))
|
2013-09-27 13:04:30 +00:00
|
|
|
{
|
2022-07-02 22:02:29 +00:00
|
|
|
auto filtersPath = configuration.getUserDataPath() / "defaultfilters";
|
2020-01-02 07:48:01 +00:00
|
|
|
|
2022-06-08 21:25:50 +00:00
|
|
|
std::ofstream destination(mProjectPath, std::ios::out | std::ios::binary);
|
2020-01-02 07:48:01 +00:00
|
|
|
if (!destination.is_open())
|
2022-07-02 22:02:29 +00:00
|
|
|
throw std::runtime_error("Can not create project file: " + Files::pathToUnicodeString(mProjectPath));
|
2020-01-02 07:48:01 +00:00
|
|
|
destination.exceptions(std::ios::failbit | std::ios::badbit);
|
|
|
|
|
2022-06-08 21:25:50 +00:00
|
|
|
if (!std::filesystem::exists(filtersPath))
|
2020-01-02 07:48:01 +00:00
|
|
|
filtersPath = mResDir / "defaultfilters";
|
|
|
|
|
2022-06-08 21:25:50 +00:00
|
|
|
std::ifstream source(filtersPath, std::ios::in | std::ios::binary);
|
2020-01-02 07:48:01 +00:00
|
|
|
if (!source.is_open())
|
2022-07-02 22:02:29 +00:00
|
|
|
throw std::runtime_error("Can not read filters file: " + Files::pathToUnicodeString(filtersPath));
|
2020-01-02 07:48:01 +00:00
|
|
|
source.exceptions(std::ios::failbit | std::ios::badbit);
|
|
|
|
|
|
|
|
destination << source.rdbuf();
|
2013-09-27 13:04:30 +00:00
|
|
|
}
|
2013-11-13 13:56:04 +00:00
|
|
|
|
2014-05-01 11:04:48 +00:00
|
|
|
if (mNew)
|
|
|
|
{
|
|
|
|
if (mContentFiles.size() == 1)
|
|
|
|
createBase();
|
|
|
|
}
|
|
|
|
|
2014-07-21 10:15:21 +00:00
|
|
|
mBlacklist.add(CSMWorld::UniversalId::Type_Script, blacklistedScripts);
|
|
|
|
|
2014-05-01 11:04:48 +00:00
|
|
|
addOptionalGmsts();
|
|
|
|
addOptionalGlobals();
|
2014-09-27 10:38:42 +00:00
|
|
|
addOptionalMagicEffects();
|
2014-05-01 11:04:48 +00:00
|
|
|
|
2022-08-23 02:28:58 +00:00
|
|
|
connect(&mUndoStack, &QUndoStack::cleanChanged, this, &Document::modificationStateChanged);
|
2012-11-22 23:36:01 +00:00
|
|
|
|
2022-08-23 02:28:58 +00:00
|
|
|
connect(&mTools, &CSMTools::Tools::progress, this, qOverload<int, int, int>(&Document::progress));
|
|
|
|
connect(&mTools, &CSMTools::Tools::done, this, &Document::operationDone);
|
|
|
|
connect(&mTools, &CSMTools::Tools::done, this, &Document::operationDone2);
|
|
|
|
connect(&mTools, &CSMTools::Tools::mergeDone, this, &Document::mergeDone);
|
2012-12-08 10:59:13 +00:00
|
|
|
|
2022-08-23 02:28:58 +00:00
|
|
|
connect(&mSaving, &OperationHolder::progress, this, qOverload<int, int, int>(&Document::progress));
|
|
|
|
connect(&mSaving, &OperationHolder::done, this, &Document::operationDone2);
|
2014-05-08 11:17:24 +00:00
|
|
|
|
2022-08-23 02:28:58 +00:00
|
|
|
connect(&mSaving, &OperationHolder::reportMessage, this, &Document::reportMessage);
|
2014-09-02 09:56:35 +00:00
|
|
|
|
2022-08-23 02:28:58 +00:00
|
|
|
connect(&mRunner, &Runner::runStateChanged, this, &Document::runStateChanged);
|
2012-11-22 22:42:17 +00:00
|
|
|
}
|
2012-11-22 14:54:31 +00:00
|
|
|
|
2013-03-12 11:28:13 +00:00
|
|
|
CSMDoc::Document::~Document() {}
|
|
|
|
|
2012-11-22 14:54:31 +00:00
|
|
|
QUndoStack& CSMDoc::Document::getUndoStack()
|
|
|
|
{
|
|
|
|
return mUndoStack;
|
2012-11-22 22:42:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int CSMDoc::Document::getState() const
|
|
|
|
{
|
|
|
|
int state = 0;
|
|
|
|
|
2015-08-23 10:37:45 +00:00
|
|
|
if (!mUndoStack.isClean() || mDirty)
|
2012-11-22 22:42:17 +00:00
|
|
|
state |= State_Modified;
|
|
|
|
|
2013-09-15 10:48:57 +00:00
|
|
|
if (mSaving.isRunning())
|
2012-12-08 13:44:03 +00:00
|
|
|
state |= State_Locked | State_Saving | State_Operation;
|
2012-11-22 23:36:01 +00:00
|
|
|
|
2014-09-02 09:56:35 +00:00
|
|
|
if (mRunner.isRunning())
|
|
|
|
state |= State_Locked | State_Running;
|
|
|
|
|
2012-12-08 16:00:58 +00:00
|
|
|
if (int operations = mTools.getRunningOperations())
|
|
|
|
state |= State_Locked | State_Operation | operations;
|
2012-11-23 12:15:45 +00:00
|
|
|
|
2012-11-22 22:42:17 +00:00
|
|
|
return state;
|
|
|
|
}
|
|
|
|
|
2022-06-08 21:25:50 +00:00
|
|
|
const std::filesystem::path& CSMDoc::Document::getResourceDir() const
|
2018-12-21 12:17:09 +00:00
|
|
|
{
|
|
|
|
return mResDir;
|
|
|
|
}
|
|
|
|
|
2022-06-08 21:25:50 +00:00
|
|
|
const std::filesystem::path& CSMDoc::Document::getSavePath() const
|
2012-11-23 13:05:49 +00:00
|
|
|
{
|
2013-09-10 14:45:01 +00:00
|
|
|
return mSavePath;
|
2012-11-23 13:05:49 +00:00
|
|
|
}
|
|
|
|
|
2022-06-08 21:25:50 +00:00
|
|
|
const std::filesystem::path& CSMDoc::Document::getProjectPath() const
|
2014-05-03 10:07:05 +00:00
|
|
|
{
|
|
|
|
return mProjectPath;
|
|
|
|
}
|
|
|
|
|
2022-06-08 21:25:50 +00:00
|
|
|
const std::vector<std::filesystem::path>& CSMDoc::Document::getContentFiles() const
|
2013-09-23 10:16:56 +00:00
|
|
|
{
|
|
|
|
return mContentFiles;
|
|
|
|
}
|
|
|
|
|
2014-04-29 12:27:44 +00:00
|
|
|
bool CSMDoc::Document::isNew() const
|
|
|
|
{
|
|
|
|
return mNew;
|
|
|
|
}
|
|
|
|
|
2012-11-22 23:36:01 +00:00
|
|
|
void CSMDoc::Document::save()
|
|
|
|
{
|
2013-09-15 10:48:57 +00:00
|
|
|
if (mSaving.isRunning())
|
|
|
|
throw std::logic_error("Failed to initiate save, because a save operation is already running.");
|
|
|
|
|
|
|
|
mSaving.start();
|
|
|
|
|
2012-11-22 23:51:04 +00:00
|
|
|
emit stateChanged(getState(), this);
|
2012-11-22 23:36:01 +00:00
|
|
|
}
|
|
|
|
|
2015-06-25 10:03:40 +00:00
|
|
|
CSMWorld::UniversalId CSMDoc::Document::verify(const CSMWorld::UniversalId& reportId)
|
2012-11-23 12:15:45 +00:00
|
|
|
{
|
2015-06-25 10:03:40 +00:00
|
|
|
CSMWorld::UniversalId id = mTools.runVerifier(reportId);
|
2012-12-08 16:00:58 +00:00
|
|
|
emit stateChanged(getState(), this);
|
2012-12-11 14:35:47 +00:00
|
|
|
return id;
|
2012-11-23 12:15:45 +00:00
|
|
|
}
|
|
|
|
|
2015-03-17 11:30:47 +00:00
|
|
|
CSMWorld::UniversalId CSMDoc::Document::newSearch()
|
|
|
|
{
|
|
|
|
return mTools.newSearch();
|
|
|
|
}
|
|
|
|
|
2015-03-27 15:33:54 +00:00
|
|
|
void CSMDoc::Document::runSearch(const CSMWorld::UniversalId& searchId, const CSMTools::Search& search)
|
|
|
|
{
|
2015-04-21 12:44:51 +00:00
|
|
|
mTools.runSearch(searchId, search);
|
|
|
|
emit stateChanged(getState(), this);
|
2015-03-27 15:33:54 +00:00
|
|
|
}
|
|
|
|
|
2017-04-28 15:30:26 +00:00
|
|
|
void CSMDoc::Document::runMerge(std::unique_ptr<CSMDoc::Document> target)
|
2015-08-13 10:03:20 +00:00
|
|
|
{
|
2017-04-28 15:30:26 +00:00
|
|
|
mTools.runMerge(std::move(target));
|
2015-08-13 10:03:20 +00:00
|
|
|
emit stateChanged(getState(), this);
|
|
|
|
}
|
|
|
|
|
2012-11-23 11:20:35 +00:00
|
|
|
void CSMDoc::Document::abortOperation(int type)
|
2012-11-22 23:36:01 +00:00
|
|
|
{
|
2012-11-23 11:20:35 +00:00
|
|
|
if (type == State_Saving)
|
2013-09-15 10:48:57 +00:00
|
|
|
mSaving.abort();
|
|
|
|
else
|
|
|
|
mTools.abortOperation(type);
|
2012-11-22 23:36:01 +00:00
|
|
|
}
|
|
|
|
|
2012-11-22 22:42:17 +00:00
|
|
|
void CSMDoc::Document::modificationStateChanged(bool clean)
|
|
|
|
{
|
|
|
|
emit stateChanged(getState(), this);
|
2012-11-22 23:36:01 +00:00
|
|
|
}
|
|
|
|
|
2015-06-20 15:56:42 +00:00
|
|
|
void CSMDoc::Document::reportMessage(const CSMDoc::Message& message, int type)
|
2013-09-27 11:54:51 +00:00
|
|
|
{
|
|
|
|
/// \todo find a better way to get these messages to the user.
|
2018-08-14 16:01:09 +00:00
|
|
|
Log(Debug::Info) << message.mMessage;
|
2013-09-27 11:54:51 +00:00
|
|
|
}
|
2013-03-06 12:41:33 +00:00
|
|
|
|
2018-06-13 13:48:24 +00:00
|
|
|
void CSMDoc::Document::operationDone2(int type, bool failed)
|
2012-12-08 10:59:13 +00:00
|
|
|
{
|
2015-08-23 10:37:45 +00:00
|
|
|
if (type == CSMDoc::State_Saving && !failed)
|
|
|
|
mDirty = false;
|
|
|
|
|
2012-12-08 10:59:13 +00:00
|
|
|
emit stateChanged(getState(), this);
|
|
|
|
}
|
|
|
|
|
2012-11-26 11:29:22 +00:00
|
|
|
const CSMWorld::Data& CSMDoc::Document::getData() const
|
|
|
|
{
|
|
|
|
return mData;
|
|
|
|
}
|
|
|
|
|
|
|
|
CSMWorld::Data& CSMDoc::Document::getData()
|
|
|
|
{
|
|
|
|
return mData;
|
2012-12-08 10:59:13 +00:00
|
|
|
}
|
|
|
|
|
2012-12-11 14:35:47 +00:00
|
|
|
CSMTools::ReportModel* CSMDoc::Document::getReport(const CSMWorld::UniversalId& id)
|
|
|
|
{
|
|
|
|
return mTools.getReport(id);
|
|
|
|
}
|
|
|
|
|
2014-07-21 10:15:21 +00:00
|
|
|
bool CSMDoc::Document::isBlacklisted(const CSMWorld::UniversalId& id) const
|
|
|
|
{
|
|
|
|
return mBlacklist.isBlacklisted(id);
|
|
|
|
}
|
|
|
|
|
2014-09-02 08:21:17 +00:00
|
|
|
void CSMDoc::Document::startRunning(const std::string& profile, const std::string& startupInstruction)
|
|
|
|
{
|
2022-07-02 22:02:29 +00:00
|
|
|
std::vector<std::filesystem::path> contentFiles;
|
2014-09-06 10:24:09 +00:00
|
|
|
|
2022-07-02 22:02:29 +00:00
|
|
|
for (const auto& mContentFile : mContentFiles)
|
|
|
|
{
|
|
|
|
contentFiles.emplace_back(mContentFile.filename());
|
|
|
|
}
|
2014-09-06 10:24:09 +00:00
|
|
|
|
2022-10-06 17:39:46 +00:00
|
|
|
mRunner.configure(getData().getDebugProfiles().getRecord(ESM::RefId::stringRefId(profile)).get(), contentFiles, startupInstruction);
|
2014-09-05 09:03:16 +00:00
|
|
|
|
2014-09-03 17:56:52 +00:00
|
|
|
int state = getState();
|
|
|
|
|
|
|
|
if (state & State_Modified)
|
|
|
|
{
|
|
|
|
// need to save first
|
|
|
|
mRunner.start(true);
|
|
|
|
|
|
|
|
new SaveWatcher(&mRunner, &mSaving); // no, that is not a memory leak. Qt is weird.
|
|
|
|
|
|
|
|
if (!(state & State_Saving))
|
|
|
|
save();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
mRunner.start();
|
2014-09-02 08:21:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CSMDoc::Document::stopRunning()
|
|
|
|
{
|
|
|
|
mRunner.stop();
|
|
|
|
}
|
2014-07-21 10:15:21 +00:00
|
|
|
|
2014-09-05 11:49:34 +00:00
|
|
|
QTextDocument* CSMDoc::Document::getRunLog()
|
|
|
|
{
|
|
|
|
return mRunner.getLog();
|
|
|
|
}
|
|
|
|
|
2014-09-02 09:56:35 +00:00
|
|
|
void CSMDoc::Document::runStateChanged()
|
|
|
|
{
|
|
|
|
emit stateChanged(getState(), this);
|
|
|
|
}
|
2014-07-21 10:15:21 +00:00
|
|
|
|
2012-12-08 10:59:13 +00:00
|
|
|
void CSMDoc::Document::progress(int current, int max, int type)
|
|
|
|
{
|
|
|
|
emit progress(current, max, type, 1, this);
|
2013-02-23 03:53:32 +00:00
|
|
|
}
|
2014-11-30 19:44:12 +00:00
|
|
|
|
2015-05-30 12:40:23 +00:00
|
|
|
CSMWorld::IdCompletionManager& CSMDoc::Document::getIdCompletionManager()
|
|
|
|
{
|
|
|
|
return mIdCompletionManager;
|
2015-05-31 15:18:49 +00:00
|
|
|
}
|
2015-08-23 10:37:45 +00:00
|
|
|
|
|
|
|
void CSMDoc::Document::flagAsDirty()
|
|
|
|
{
|
|
|
|
mDirty = true;
|
|
|
|
}
|