2023-03-19 21:32:16 +01:00
|
|
|
#include "luaevents.hpp"
|
|
|
|
|
|
|
|
#include <components/debug/debuglog.hpp>
|
|
|
|
|
|
|
|
#include <components/esm/luascripts.hpp>
|
|
|
|
#include <components/esm3/esmreader.hpp>
|
|
|
|
#include <components/esm3/esmwriter.hpp>
|
|
|
|
|
|
|
|
#include <components/lua/serialization.hpp>
|
|
|
|
|
|
|
|
#include "../mwbase/environment.hpp"
|
|
|
|
#include "../mwworld/worldmodel.hpp"
|
|
|
|
|
|
|
|
#include "globalscripts.hpp"
|
|
|
|
#include "localscripts.hpp"
|
|
|
|
|
|
|
|
namespace MWLua
|
|
|
|
{
|
|
|
|
|
|
|
|
void LuaEvents::clear()
|
|
|
|
{
|
|
|
|
mGlobalEventBatch.clear();
|
|
|
|
mLocalEventBatch.clear();
|
|
|
|
mNewGlobalEventBatch.clear();
|
|
|
|
mNewLocalEventBatch.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void LuaEvents::finalizeEventBatch()
|
|
|
|
{
|
|
|
|
mNewGlobalEventBatch.swap(mGlobalEventBatch);
|
|
|
|
mNewLocalEventBatch.swap(mLocalEventBatch);
|
|
|
|
mNewGlobalEventBatch.clear();
|
|
|
|
mNewLocalEventBatch.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void LuaEvents::callEventHandlers()
|
|
|
|
{
|
2023-03-25 18:14:33 +01:00
|
|
|
for (const Global& e : mGlobalEventBatch)
|
|
|
|
mGlobalScripts.receiveEvent(e.mEventName, e.mEventData);
|
2023-03-19 21:32:16 +01:00
|
|
|
mGlobalEventBatch.clear();
|
2023-03-25 18:14:33 +01:00
|
|
|
for (const Local& e : mLocalEventBatch)
|
2023-03-19 21:32:16 +01:00
|
|
|
{
|
2023-05-30 21:44:54 +02:00
|
|
|
MWWorld::Ptr ptr = MWBase::Environment::get().getWorldModel()->getPtr(e.mDest);
|
2023-03-19 21:32:16 +01:00
|
|
|
LocalScripts* scripts = ptr.isEmpty() ? nullptr : ptr.getRefData().getLuaScripts();
|
|
|
|
if (scripts)
|
|
|
|
scripts->receiveEvent(e.mEventName, e.mEventData);
|
|
|
|
else
|
|
|
|
Log(Debug::Debug) << "Ignored event " << e.mEventName << " to L" << e.mDest.toString()
|
|
|
|
<< ". Object not found or has no attached scripts";
|
|
|
|
}
|
|
|
|
mLocalEventBatch.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename Event>
|
|
|
|
static void saveEvent(ESM::ESMWriter& esm, const ESM::RefNum& dest, const Event& event)
|
|
|
|
{
|
|
|
|
esm.writeHNString("LUAE", event.mEventName);
|
2023-04-07 02:14:32 +02:00
|
|
|
esm.writeFormId(dest, true);
|
2023-03-19 21:32:16 +01:00
|
|
|
if (!event.mEventData.empty())
|
|
|
|
saveLuaBinaryData(esm, event.mEventData);
|
|
|
|
}
|
|
|
|
|
|
|
|
void LuaEvents::load(lua_State* lua, ESM::ESMReader& esm, const std::map<int, int>& contentFileMapping,
|
|
|
|
const LuaUtil::UserdataSerializer* serializer)
|
|
|
|
{
|
|
|
|
clear();
|
|
|
|
while (esm.isNextSub("LUAE"))
|
|
|
|
{
|
|
|
|
std::string name = esm.getHString();
|
2023-04-07 02:14:32 +02:00
|
|
|
ESM::RefNum dest = esm.getFormId(true);
|
2023-03-19 21:32:16 +01:00
|
|
|
std::string data = loadLuaBinaryData(esm);
|
|
|
|
try
|
|
|
|
{
|
|
|
|
data = LuaUtil::serialize(LuaUtil::deserialize(lua, data, serializer), serializer);
|
|
|
|
}
|
|
|
|
catch (std::exception& e)
|
|
|
|
{
|
|
|
|
Log(Debug::Error) << "loadEvent: invalid event data: " << e.what();
|
|
|
|
}
|
|
|
|
if (dest.isSet())
|
|
|
|
{
|
|
|
|
auto it = contentFileMapping.find(dest.mContentFile);
|
|
|
|
if (it != contentFileMapping.end())
|
|
|
|
dest.mContentFile = it->second;
|
|
|
|
mLocalEventBatch.push_back({ dest, std::move(name), std::move(data) });
|
|
|
|
}
|
|
|
|
else
|
|
|
|
mGlobalEventBatch.push_back({ std::move(name), std::move(data) });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void LuaEvents::save(ESM::ESMWriter& esm) const
|
|
|
|
{
|
|
|
|
// Used as a marker of a global event.
|
|
|
|
constexpr ESM::RefNum globalId;
|
|
|
|
|
|
|
|
for (const Global& e : mGlobalEventBatch)
|
|
|
|
saveEvent(esm, globalId, e);
|
|
|
|
for (const Global& e : mNewGlobalEventBatch)
|
|
|
|
saveEvent(esm, globalId, e);
|
|
|
|
for (const Local& e : mLocalEventBatch)
|
|
|
|
saveEvent(esm, e.mDest, e);
|
|
|
|
for (const Local& e : mNewLocalEventBatch)
|
|
|
|
saveEvent(esm, e.mDest, e);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|