1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-26 18:35:20 +00:00
OpenMW/apps/opencs/model/doc/document.cpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

550 lines
15 KiB
C++
Raw Normal View History

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