1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-04-04 04:20:51 +00:00
OpenMW/apps/openmw/mwscript/interpretercontext.cpp
fteppe 125b21de20 Initial commit: In ESM structures, replace the string members that are RefIds to other records, to a new strong type
The strong type is actually just a string underneath, but this will help in the future to have a distinction so it's easier to search and replace when we use an integer ID

Slowly going through all the changes to make, still hundreds of errors

a lot of functions/structures use std::string or stringview to designate an ID. So it takes time

Continues slowly replacing ids. There are technically more and more compilation errors

I have good hope that there is a point where the amount of errors will dramatically go down as all the main functions use the ESM::RefId type

Continue moving forward, changes to the stores

slowly moving along

Starting to see the fruit of those changes.

still many many error, but more and more Irun into a situation where a function is sandwiched between two functions that use the RefId type.

More replacements. Things are starting to get easier

I can see more and more often the issue is that the function is awaiting a RefId, but is given a string
there is less need to go down functions and to fix a long list of them.

Still moving forward, and for the first time error count is going down!

Good pace, not sure about topics though, mId and mName are actually the same thing and are used interchangeably

Cells are back to using string for the name, haven't fixed everything yet. Many other changes

Under the bar of 400 compilation errors.

more good progress <100 compile errors!

More progress

Game settings store can use string for find, it was a bit absurd how every use of it required to create refId from string

some more progress on other fronts

Mostly game settings clean

one error opened a lot of other errors. Down to 18, but more will prbably appear

only link errors left??

Fixed link errors

OpenMW compiles, and launches, with some issues, but still!
2022-12-27 19:15:54 +01:00

488 lines
16 KiB
C++

#include "interpretercontext.hpp"
#include <cmath>
#include <sstream>
#include <components/compiler/locals.hpp>
#include <components/esm/records.hpp>
#include "../mwworld/esmstore.hpp"
#include "../mwbase/environment.hpp"
#include "../mwbase/inputmanager.hpp"
#include "../mwbase/luamanager.hpp"
#include "../mwbase/scriptmanager.hpp"
#include "../mwbase/windowmanager.hpp"
#include "../mwbase/world.hpp"
#include "../mwworld/action.hpp"
#include "../mwworld/cellstore.hpp"
#include "../mwworld/class.hpp"
#include "../mwmechanics/npcstats.hpp"
#include "globalscripts.hpp"
#include "locals.hpp"
namespace MWScript
{
const MWWorld::Ptr InterpreterContext::getReferenceImp(const ESM::RefId& id, bool activeOnly, bool doThrow) const
{
if (!id.empty())
{
return MWBase::Environment::get().getWorld()->getPtr(id, activeOnly);
}
else
{
if (mReference.isEmpty() && mGlobalScriptDesc)
mReference = mGlobalScriptDesc->getPtr();
if (mReference.isEmpty() && doThrow)
throw MissingImplicitRefError();
return mReference;
}
}
const Locals& InterpreterContext::getMemberLocals(ESM::RefId& id, bool global) const
{
if (global)
{
return MWBase::Environment::get().getScriptManager()->getGlobalScripts().getLocals(id);
}
else
{
const MWWorld::Ptr ptr = getReferenceImp(id, false);
id = ptr.getClass().getScript(ptr);
ptr.getRefData().setLocals(*MWBase::Environment::get().getWorld()->getStore().get<ESM::Script>().find(id));
return ptr.getRefData().getLocals();
}
}
Locals& InterpreterContext::getMemberLocals(ESM::RefId& id, bool global)
{
if (global)
{
return MWBase::Environment::get().getScriptManager()->getGlobalScripts().getLocals(id);
}
else
{
const MWWorld::Ptr ptr = getReferenceImp(id, false);
id = ptr.getClass().getScript(ptr);
ptr.getRefData().setLocals(*MWBase::Environment::get().getWorld()->getStore().get<ESM::Script>().find(id));
return ptr.getRefData().getLocals();
}
}
MissingImplicitRefError::MissingImplicitRefError()
: std::runtime_error("no implicit reference")
{
}
int InterpreterContext::findLocalVariableIndex(const ESM::RefId& scriptId, std::string_view name, char type) const
{
int index = MWBase::Environment::get().getScriptManager()->getLocals(scriptId).searchIndex(type, name);
if (index != -1)
return index;
std::ostringstream stream;
stream << "Failed to access ";
switch (type)
{
case 's':
stream << "short";
break;
case 'l':
stream << "long";
break;
case 'f':
stream << "float";
break;
}
stream << " member variable " << name << " in script " << scriptId;
throw std::runtime_error(stream.str().c_str());
}
InterpreterContext::InterpreterContext(MWScript::Locals* locals, const MWWorld::Ptr& reference)
: mLocals(locals)
, mReference(reference)
{
}
InterpreterContext::InterpreterContext(std::shared_ptr<GlobalScriptDesc> globalScriptDesc)
: mLocals(&(globalScriptDesc->mLocals))
{
const MWWorld::Ptr* ptr = globalScriptDesc->getPtrIfPresent();
// A nullptr here signifies that the script's target has not yet been resolved after loading the game.
// Script targets are lazily resolved to MWWorld::Ptrs (which can, upon resolution, be empty)
// because scripts started through dialogue often don't use their implicit target.
if (ptr)
mReference = *ptr;
else
mGlobalScriptDesc = globalScriptDesc;
}
const ESM::RefId& InterpreterContext::getTarget() const
{
if (!mReference.isEmpty())
return mReference.mRef->mRef.getRefId();
else if (mGlobalScriptDesc)
return mGlobalScriptDesc->getId();
return ESM::RefId::sEmpty;
}
int InterpreterContext::getLocalShort(int index) const
{
if (!mLocals)
throw std::runtime_error("local variables not available in this context");
return mLocals->mShorts.at(index);
}
int InterpreterContext::getLocalLong(int index) const
{
if (!mLocals)
throw std::runtime_error("local variables not available in this context");
return mLocals->mLongs.at(index);
}
float InterpreterContext::getLocalFloat(int index) const
{
if (!mLocals)
throw std::runtime_error("local variables not available in this context");
return mLocals->mFloats.at(index);
}
void InterpreterContext::setLocalShort(int index, int value)
{
if (!mLocals)
throw std::runtime_error("local variables not available in this context");
mLocals->mShorts.at(index) = value;
}
void InterpreterContext::setLocalLong(int index, int value)
{
if (!mLocals)
throw std::runtime_error("local variables not available in this context");
mLocals->mLongs.at(index) = value;
}
void InterpreterContext::setLocalFloat(int index, float value)
{
if (!mLocals)
throw std::runtime_error("local variables not available in this context");
mLocals->mFloats.at(index) = value;
}
void InterpreterContext::messageBox(std::string_view message, const std::vector<std::string>& buttons)
{
if (buttons.empty())
MWBase::Environment::get().getWindowManager()->messageBox(message);
else
MWBase::Environment::get().getWindowManager()->interactiveMessageBox(message, buttons);
}
void InterpreterContext::report(const std::string& message) {}
int InterpreterContext::getGlobalShort(std::string_view name) const
{
return MWBase::Environment::get().getWorld()->getGlobalInt(name);
}
int InterpreterContext::getGlobalLong(std::string_view name) const
{
// a global long is internally a float.
return MWBase::Environment::get().getWorld()->getGlobalInt(name);
}
float InterpreterContext::getGlobalFloat(std::string_view name) const
{
return MWBase::Environment::get().getWorld()->getGlobalFloat(name);
}
void InterpreterContext::setGlobalShort(std::string_view name, int value)
{
MWBase::Environment::get().getWorld()->setGlobalInt(name, value);
}
void InterpreterContext::setGlobalLong(std::string_view name, int value)
{
MWBase::Environment::get().getWorld()->setGlobalInt(name, value);
}
void InterpreterContext::setGlobalFloat(std::string_view name, float value)
{
MWBase::Environment::get().getWorld()->setGlobalFloat(name, value);
}
std::vector<std::string> InterpreterContext::getGlobals() const
{
const MWWorld::Store<ESM::Global>& globals
= MWBase::Environment::get().getWorld()->getStore().get<ESM::Global>();
std::vector<std::string> ids;
for (const auto& globalVariable : globals)
{
ids.emplace_back(globalVariable.mId.getRefIdString());
}
return ids;
}
char InterpreterContext::getGlobalType(std::string_view name) const
{
MWBase::World* world = MWBase::Environment::get().getWorld();
return world->getGlobalVariableType(name);
}
std::string InterpreterContext::getActionBinding(std::string_view targetAction) const
{
MWBase::InputManager* input = MWBase::Environment::get().getInputManager();
const auto& actions = input->getActionKeySorting();
for (const int action : actions)
{
std::string_view desc = input->getActionDescription(action);
if (desc.empty())
continue;
if (desc == targetAction)
{
if (input->joystickLastUsed())
return input->getActionControllerBindingName(action);
else
return input->getActionKeyBindingName(action);
}
}
return "None";
}
std::string_view InterpreterContext::getActorName() const
{
const MWWorld::Ptr& ptr = getReferenceImp();
if (ptr.getClass().isNpc())
{
const ESM::NPC* npc = ptr.get<ESM::NPC>()->mBase;
return npc->mName;
}
const ESM::Creature* creature = ptr.get<ESM::Creature>()->mBase;
return creature->mName;
}
std::string_view InterpreterContext::getNPCRace() const
{
const ESM::NPC* npc = getReferenceImp().get<ESM::NPC>()->mBase;
const ESM::Race* race = MWBase::Environment::get().getWorld()->getStore().get<ESM::Race>().find(npc->mRace);
return race->mName;
}
std::string_view InterpreterContext::getNPCClass() const
{
const ESM::NPC* npc = getReferenceImp().get<ESM::NPC>()->mBase;
const ESM::Class* class_
= MWBase::Environment::get().getWorld()->getStore().get<ESM::Class>().find(npc->mClass);
return class_->mName;
}
std::string_view InterpreterContext::getNPCFaction() const
{
const ESM::NPC* npc = getReferenceImp().get<ESM::NPC>()->mBase;
const ESM::Faction* faction
= MWBase::Environment::get().getWorld()->getStore().get<ESM::Faction>().find(npc->mFaction);
return faction->mName;
}
std::string_view InterpreterContext::getNPCRank() const
{
const MWWorld::Ptr& ptr = getReferenceImp();
const ESM::RefId& faction = ptr.getClass().getPrimaryFaction(ptr);
if (faction.empty())
throw std::runtime_error("getNPCRank(): NPC is not in a faction");
int rank = ptr.getClass().getPrimaryFactionRank(ptr);
if (rank < 0 || rank > 9)
throw std::runtime_error("getNPCRank(): invalid rank");
MWBase::World* world = MWBase::Environment::get().getWorld();
const MWWorld::ESMStore& store = world->getStore();
const ESM::Faction* fact = store.get<ESM::Faction>().find(faction);
return fact->mRanks[rank];
}
std::string_view InterpreterContext::getPCName() const
{
MWBase::World* world = MWBase::Environment::get().getWorld();
return world->getPlayerPtr().get<ESM::NPC>()->mBase->mName;
}
std::string_view InterpreterContext::getPCRace() const
{
MWBase::World* world = MWBase::Environment::get().getWorld();
const ESM::RefId& race = world->getPlayerPtr().get<ESM::NPC>()->mBase->mRace;
return world->getStore().get<ESM::Race>().find(race)->mName;
}
std::string_view InterpreterContext::getPCClass() const
{
MWBase::World* world = MWBase::Environment::get().getWorld();
const ESM::RefId& class_ = world->getPlayerPtr().get<ESM::NPC>()->mBase->mClass;
return world->getStore().get<ESM::Class>().find(class_)->mName;
}
std::string_view InterpreterContext::getPCRank() const
{
MWBase::World* world = MWBase::Environment::get().getWorld();
MWWorld::Ptr player = world->getPlayerPtr();
const ESM::RefId& factionId = getReferenceImp().getClass().getPrimaryFaction(getReferenceImp());
if (factionId.empty())
throw std::runtime_error("getPCRank(): NPC is not in a faction");
const auto& ranks = player.getClass().getNpcStats(player).getFactionRanks();
auto it = ranks.find(factionId);
int rank = -1;
if (it != ranks.end())
rank = it->second;
// If you are not in the faction, PcRank returns the first rank, for whatever reason.
// This is used by the dialogue when joining the Thieves Guild in Balmora.
if (rank == -1)
rank = 0;
const MWWorld::ESMStore& store = world->getStore();
const ESM::Faction* faction = store.get<ESM::Faction>().find(factionId);
if (rank < 0 || rank > 9) // there are only 10 ranks
return {};
return faction->mRanks[rank];
}
std::string_view InterpreterContext::getPCNextRank() const
{
MWBase::World* world = MWBase::Environment::get().getWorld();
MWWorld::Ptr player = world->getPlayerPtr();
const ESM::RefId& factionId = getReferenceImp().getClass().getPrimaryFaction(getReferenceImp());
if (factionId.empty())
throw std::runtime_error("getPCNextRank(): NPC is not in a faction");
const auto& ranks = player.getClass().getNpcStats(player).getFactionRanks();
auto it = ranks.find(factionId);
int rank = -1;
if (it != ranks.end())
rank = it->second;
++rank; // Next rank
// if we are already at max rank, there is no next rank
if (rank > 9)
rank = 9;
const MWWorld::ESMStore& store = world->getStore();
const ESM::Faction* faction = store.get<ESM::Faction>().find(factionId);
if (rank < 0)
return {};
return faction->mRanks[rank];
}
int InterpreterContext::getPCBounty() const
{
MWBase::World* world = MWBase::Environment::get().getWorld();
MWWorld::Ptr player = world->getPlayerPtr();
return player.getClass().getNpcStats(player).getBounty();
}
std::string_view InterpreterContext::getCurrentCellName() const
{
return MWBase::Environment::get().getWorld()->getCellName();
}
void InterpreterContext::executeActivation(const MWWorld::Ptr& ptr, const MWWorld::Ptr& actor)
{
MWBase::Environment::get().getLuaManager()->objectActivated(ptr, actor);
std::unique_ptr<MWWorld::Action> action = (ptr.getClass().activate(ptr, actor));
action->execute(actor);
if (action->getTarget() != MWWorld::Ptr() && action->getTarget() != ptr)
{
updatePtr(ptr, action->getTarget());
}
}
int InterpreterContext::getMemberShort(ESM::RefId id, std::string_view name, bool global) const
{
const Locals& locals = getMemberLocals(id, global);
return locals.mShorts[findLocalVariableIndex(id, name, 's')];
}
int InterpreterContext::getMemberLong(ESM::RefId id, std::string_view name, bool global) const
{
const Locals& locals = getMemberLocals(id, global);
return locals.mLongs[findLocalVariableIndex(id, name, 'l')];
}
float InterpreterContext::getMemberFloat(ESM::RefId id, std::string_view name, bool global) const
{
const Locals& locals = getMemberLocals(id, global);
return locals.mFloats[findLocalVariableIndex(id, name, 'f')];
}
void InterpreterContext::setMemberShort(ESM::RefId id, std::string_view name, int value, bool global)
{
Locals& locals = getMemberLocals(id, global);
locals.mShorts[findLocalVariableIndex(id, name, 's')] = value;
}
void InterpreterContext::setMemberLong(ESM::RefId id, std::string_view name, int value, bool global)
{
Locals& locals = getMemberLocals(id, global);
locals.mLongs[findLocalVariableIndex(id, name, 'l')] = value;
}
void InterpreterContext::setMemberFloat(ESM::RefId id, std::string_view name, float value, bool global)
{
Locals& locals = getMemberLocals(id, global);
locals.mFloats[findLocalVariableIndex(id, name, 'f')] = value;
}
MWWorld::Ptr InterpreterContext::getReference(bool required) const
{
return getReferenceImp({}, true, required);
}
void InterpreterContext::updatePtr(const MWWorld::Ptr& base, const MWWorld::Ptr& updated)
{
if (!mReference.isEmpty() && base == mReference)
{
mReference = updated;
if (mLocals == &base.getRefData().getLocals())
mLocals = &mReference.getRefData().getLocals();
}
}
}