#include "userdataserializer.hpp" #include #include #include "object.hpp" namespace MWLua { class Serializer final : public LuaUtil::UserdataSerializer { public: explicit Serializer(bool localSerializer, ObjectRegistry* registry, std::map* contentFileMapping) : mLocalSerializer(localSerializer), mObjectRegistry(registry), mContentFileMapping(contentFileMapping) {} 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() || data.is()) { ObjectId id = data.as().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); if (id.hasContentFile() && mContentFileMapping) { auto iter = mContentFileMapping->find(id.mContentFile); if (iter != mContentFileMapping->end()) id.mContentFile = iter->second; } if (mLocalSerializer) sol::stack::push(lua, LObject(id, mObjectRegistry)); else sol::stack::push(lua, GObject(id, mObjectRegistry)); return true; } return false; } bool mLocalSerializer; ObjectRegistry* mObjectRegistry; std::map* mContentFileMapping; }; std::unique_ptr createUserdataSerializer( bool local, ObjectRegistry* registry, std::map* contentFileMapping) { return std::make_unique(local, registry, contentFileMapping); } }