1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-18 13:12:50 +00:00
OpenMW/apps/openmw/mwscript/locals.cpp
2023-07-08 11:28:56 +02:00

248 lines
7.6 KiB
C++

#include "locals.hpp"
#include "globalscripts.hpp"
#include <components/compiler/locals.hpp>
#include <components/debug/debuglog.hpp>
#include <components/esm3/loadscpt.hpp>
#include <components/esm3/locals.hpp>
#include <components/esm3/variant.hpp>
#include "../mwbase/environment.hpp"
#include "../mwbase/scriptmanager.hpp"
#include "../mwworld/esmstore.hpp"
namespace MWScript
{
void Locals::ensure(const ESM::RefId& scriptName)
{
if (!mInitialised)
{
const ESM::Script* script = MWBase::Environment::get().getESMStore()->get<ESM::Script>().find(scriptName);
configure(*script);
}
}
Locals::Locals()
: mInitialised(false)
{
}
bool Locals::configure(const ESM::Script& script)
{
if (mInitialised)
return false;
const GlobalScriptDesc* global
= MWBase::Environment::get().getScriptManager()->getGlobalScripts().getScriptIfPresent(script.mId);
if (global)
{
mShorts = global->mLocals.mShorts;
mLongs = global->mLocals.mLongs;
mFloats = global->mLocals.mFloats;
}
else
{
const Compiler::Locals& locals = MWBase::Environment::get().getScriptManager()->getLocals(script.mId);
mShorts.clear();
mShorts.resize(locals.get('s').size(), 0);
mLongs.clear();
mLongs.resize(locals.get('l').size(), 0);
mFloats.clear();
mFloats.resize(locals.get('f').size(), 0);
}
mScriptId = script.mId;
mInitialised = true;
return true;
}
bool Locals::isEmpty() const
{
return (mShorts.empty() && mLongs.empty() && mFloats.empty());
}
bool Locals::hasVar(const ESM::RefId& script, std::string_view var)
{
ensure(script);
const Compiler::Locals& locals = MWBase::Environment::get().getScriptManager()->getLocals(script);
int index = locals.getIndex(var);
return (index != -1);
}
double Locals::getVarAsDouble(const ESM::RefId& script, std::string_view var)
{
ensure(script);
const Compiler::Locals& locals = MWBase::Environment::get().getScriptManager()->getLocals(script);
int index = locals.getIndex(var);
if (index == -1)
return 0;
switch (locals.getType(var))
{
case 's':
return mShorts.at(index);
case 'l':
return mLongs.at(index);
case 'f':
return mFloats.at(index);
default:
return 0;
}
}
bool Locals::setVar(const ESM::RefId& script, std::string_view var, double val)
{
ensure(script);
const Compiler::Locals& locals = MWBase::Environment::get().getScriptManager()->getLocals(script);
int index = locals.getIndex(var);
if (index == -1)
return false;
switch (locals.getType(var))
{
case 's':
mShorts.at(index) = static_cast<Interpreter::Type_Short>(val);
break;
case 'l':
mLongs.at(index) = static_cast<Interpreter::Type_Integer>(val);
break;
case 'f':
mFloats.at(index) = static_cast<Interpreter::Type_Float>(val);
break;
}
return true;
}
bool Locals::write(ESM::Locals& locals, const ESM::RefId& script) const
{
if (!mInitialised)
return false;
const Compiler::Locals& declarations = MWBase::Environment::get().getScriptManager()->getLocals(script);
for (int i = 0; i < 3; ++i)
{
char type = 0;
switch (i)
{
case 0:
type = 's';
break;
case 1:
type = 'l';
break;
case 2:
type = 'f';
break;
}
const std::vector<std::string>& names = declarations.get(type);
for (int i2 = 0; i2 < static_cast<int>(names.size()); ++i2)
{
ESM::Variant value;
switch (i)
{
case 0:
value.setType(ESM::VT_Int);
value.setInteger(mShorts.at(i2));
break;
case 1:
value.setType(ESM::VT_Int);
value.setInteger(mLongs.at(i2));
break;
case 2:
value.setType(ESM::VT_Float);
value.setFloat(mFloats.at(i2));
break;
}
locals.mVariables.emplace_back(names[i2], value);
}
}
return true;
}
void Locals::read(const ESM::Locals& locals, const ESM::RefId& script)
{
ensure(script);
const Compiler::Locals& declarations = MWBase::Environment::get().getScriptManager()->getLocals(script);
int index = 0, numshorts = 0, numlongs = 0;
for (unsigned int v = 0; v < locals.mVariables.size(); ++v)
{
ESM::VarType type = locals.mVariables[v].second.getType();
if (type == ESM::VT_Short)
++numshorts;
else if (type == ESM::VT_Int)
++numlongs;
}
for (std::vector<std::pair<std::string, ESM::Variant>>::const_iterator iter = locals.mVariables.begin();
iter != locals.mVariables.end(); ++iter, ++index)
{
if (iter->first.empty())
{
// no variable names available (this will happen for legacy, i.e. ESS-imported savegames only)
try
{
if (index >= numshorts + numlongs)
mFloats.at(index - (numshorts + numlongs)) = iter->second.getFloat();
else if (index >= numshorts)
mLongs.at(index - numshorts) = iter->second.getInteger();
else
mShorts.at(index) = iter->second.getInteger();
}
catch (std::exception& e)
{
Log(Debug::Error) << "Failed to read local variable state for script '" << script
<< "' (legacy format): " << e.what() << "\nNum shorts: " << numshorts << " / "
<< mShorts.size() << " Num longs: " << numlongs << " / " << mLongs.size();
}
}
else
{
char type = declarations.getType(iter->first);
int index2 = declarations.getIndex(iter->first);
// silently ignore locals that don't exist anymore
if (type == ' ' || index2 == -1)
continue;
try
{
switch (type)
{
case 's':
mShorts.at(index2) = iter->second.getInteger();
break;
case 'l':
mLongs.at(index2) = iter->second.getInteger();
break;
case 'f':
mFloats.at(index2) = iter->second.getFloat();
break;
}
}
catch (...)
{
// ignore type changes
/// \todo write to log
}
}
}
}
}