#ifndef MWLUA_OBJECT_H #define MWLUA_OBJECT_H #include #include #include #include #include "../mwbase/environment.hpp" #include "../mwbase/world.hpp" #include "../mwworld/ptr.hpp" namespace MWLua { // ObjectId is a unique identifier of a game object. // It can change only if the order of content files was change. using ObjectId = ESM::RefNum; inline const ObjectId& getId(const MWWorld::Ptr& ptr) { return ptr.getCellRef().getRefNum(); } std::string idToString(const ObjectId& id); std::string ptrToString(const MWWorld::Ptr& ptr); bool isMarker(const MWWorld::Ptr& ptr); std::string_view getLuaObjectTypeName(ESM::RecNameInts recordType, std::string_view fallback = "Unknown"); std::string_view getLuaObjectTypeName(const MWWorld::Ptr& ptr); // Each script has a set of flags that controls to which objects the script should be // automatically attached. This function maps each object types to one of the flags. ESM::LuaScriptCfg::Flags getLuaScriptFlag(const MWWorld::Ptr& ptr); // Holds a mapping ObjectId -> MWWord::Ptr. class ObjectRegistry { public: ObjectRegistry() { mLastAssignedId.unset(); } void update(); // Should be called every frame. void clear(); // Should be called before starting or loading a new game. ObjectId registerPtr(const MWWorld::Ptr& ptr); ObjectId deregisterPtr(const MWWorld::Ptr& ptr); // Returns Ptr by id. If object is not found, returns empty Ptr. // If local = true, returns non-empty ptr only if it can be used in local scripts // (i.e. is active or was active in the previous frame). MWWorld::Ptr getPtr(ObjectId id, bool local); // Needed only for saving/loading. const ObjectId& getLastAssignedId() const { return mLastAssignedId; } void setLastAssignedId(ObjectId id) { mLastAssignedId = id; } private: friend class Object; friend class LuaManager; bool mChanged = false; int64_t mUpdateCounter = 0; std::map mObjectMapping; ObjectId mLastAssignedId; }; // Lua scripts can't use MWWorld::Ptr directly, because lifetime of a script can be longer than lifetime of Ptr. // `GObject` and `LObject` are intended to be passed to Lua as a userdata. // It automatically updates the underlying Ptr when needed. class Object { public: Object(ObjectId id, ObjectRegistry* reg) : mId(id), mObjectRegistry(reg) {} virtual ~Object() {} ObjectId id() const { return mId; } std::string toString() const; std::string_view type() const { return getLuaObjectTypeName(ptr()); } // Updates and returns the underlying Ptr. Throws an exception if object is not available. const MWWorld::Ptr& ptr() const; // Returns `true` if calling `ptr()` is safe. bool isValid() const; protected: virtual void updatePtr() const = 0; const ObjectId mId; ObjectRegistry* mObjectRegistry; mutable MWWorld::Ptr mPtr; mutable int64_t mLastUpdate = -1; }; // Used only in local scripts class LObject : public Object { using Object::Object; void updatePtr() const final { mPtr = mObjectRegistry->getPtr(mId, true); } }; // Used only in global scripts class GObject : public Object { using Object::Object; void updatePtr() const final { mPtr = mObjectRegistry->getPtr(mId, false); } }; using ObjectIdList = std::shared_ptr>; template struct ObjectList { ObjectIdList mIds; }; using GObjectList = ObjectList; using LObjectList = ObjectList; } #endif // MWLUA_OBJECT_H