2020-12-18 22:21:10 +00:00
|
|
|
#include "userdataserializer.hpp"
|
|
|
|
|
|
|
|
#include <components/lua/serialization.hpp>
|
|
|
|
#include <components/misc/endianness.hpp>
|
|
|
|
|
|
|
|
#include "object.hpp"
|
|
|
|
|
|
|
|
namespace MWLua
|
|
|
|
{
|
|
|
|
|
|
|
|
class Serializer final : public LuaUtil::UserdataSerializer
|
|
|
|
{
|
|
|
|
public:
|
2021-01-29 01:38:09 +00:00
|
|
|
explicit Serializer(bool localSerializer, ObjectRegistry* registry, std::map<int, int>* contentFileMapping)
|
|
|
|
: mLocalSerializer(localSerializer), mObjectRegistry(registry), mContentFileMapping(contentFileMapping) {}
|
2020-12-18 22:21:10 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
// Appends serialized sol::userdata to the end of BinaryData.
|
|
|
|
// Returns false if this type of userdata is not supported by this serializer.
|
|
|
|
bool serialize(LuaUtil::BinaryData& out, const sol::userdata& data) const override
|
|
|
|
{
|
|
|
|
if (data.is<GObject>() || data.is<LObject>())
|
|
|
|
{
|
|
|
|
ObjectId id = data.as<Object>().id();
|
|
|
|
static_assert(sizeof(ObjectId) == 8);
|
|
|
|
id.mIndex = Misc::toLittleEndian(id.mIndex);
|
|
|
|
id.mContentFile = Misc::toLittleEndian(id.mContentFile);
|
|
|
|
append(out, "o", &id, sizeof(ObjectId));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Deserializes userdata of type "typeName" from binaryData. Should push the result on stack using sol::stack::push.
|
|
|
|
// Returns false if this type is not supported by this serializer.
|
|
|
|
bool deserialize(std::string_view typeName, std::string_view binaryData, sol::state& lua) const override
|
|
|
|
{
|
|
|
|
if (typeName == "o")
|
|
|
|
{
|
|
|
|
if (binaryData.size() != sizeof(ObjectId))
|
|
|
|
throw std::runtime_error("Incorrect serialization format. Size of ObjectId doesn't match.");
|
|
|
|
ObjectId id;
|
|
|
|
std::memcpy(&id, binaryData.data(), sizeof(ObjectId));
|
|
|
|
id.mIndex = Misc::fromLittleEndian(id.mIndex);
|
|
|
|
id.mContentFile = Misc::fromLittleEndian(id.mContentFile);
|
2021-01-29 01:38:09 +00:00
|
|
|
if (id.hasContentFile() && mContentFileMapping)
|
|
|
|
{
|
|
|
|
auto iter = mContentFileMapping->find(id.mContentFile);
|
|
|
|
if (iter != mContentFileMapping->end())
|
|
|
|
id.mContentFile = iter->second;
|
|
|
|
}
|
2020-12-18 22:21:10 +00:00
|
|
|
if (mLocalSerializer)
|
|
|
|
sol::stack::push<LObject>(lua, LObject(id, mObjectRegistry));
|
|
|
|
else
|
|
|
|
sol::stack::push<GObject>(lua, GObject(id, mObjectRegistry));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool mLocalSerializer;
|
|
|
|
ObjectRegistry* mObjectRegistry;
|
2021-01-29 01:38:09 +00:00
|
|
|
std::map<int, int>* mContentFileMapping;
|
2020-12-18 22:21:10 +00:00
|
|
|
};
|
|
|
|
|
2021-01-29 01:38:09 +00:00
|
|
|
std::unique_ptr<LuaUtil::UserdataSerializer> createUserdataSerializer(
|
|
|
|
bool local, ObjectRegistry* registry, std::map<int, int>* contentFileMapping)
|
2020-12-18 22:21:10 +00:00
|
|
|
{
|
2021-01-29 01:38:09 +00:00
|
|
|
return std::make_unique<Serializer>(local, registry, contentFileMapping);
|
2020-12-18 22:21:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|