#include "BsManagedSerializableList.h" #include "BsManagedSerializableListRTTI.h" #include "BsMonoManager.h" #include "BsScriptAssemblyManager.h" #include "BsManagedSerializableField.h" #include "BsMonoClass.h" #include "BsMonoMethod.h" #include "BsMonoProperty.h" #include "BsMonoUtil.h" namespace BansheeEngine { ManagedSerializableList::ManagedSerializableList(const ConstructPrivately& dummy) :mManagedInstance(nullptr), mNumElements(0), mItemProp(nullptr), mCountProp(nullptr), mAddMethod(nullptr), mAddRangeMethod(nullptr), mCopyToMethod(nullptr), mClearMethod(nullptr) { } ManagedSerializableList::ManagedSerializableList(const ConstructPrivately& dummy, const ManagedSerializableTypeInfoListPtr& typeInfo, MonoObject* managedInstance) :mListTypeInfo(typeInfo), mManagedInstance(managedInstance), mNumElements(0), mItemProp(nullptr), mCountProp(nullptr), mAddMethod(nullptr), mAddRangeMethod(nullptr), mCopyToMethod(nullptr), mClearMethod(nullptr) { MonoClass* listClass = MonoManager::instance().findClass(mono_object_get_class(managedInstance)); if(listClass == nullptr) return; initMonoObjects(listClass); mNumElements = getLengthInternal(); } ManagedSerializableListPtr ManagedSerializableList::createFromExisting(MonoObject* managedInstance, const ManagedSerializableTypeInfoListPtr& typeInfo) { if(managedInstance == nullptr) return nullptr; String elementNs; String elementTypeName; MonoUtil::getClassName(managedInstance, elementNs, elementTypeName); String fullName = elementNs + "." + elementTypeName; if(ScriptAssemblyManager::instance().getSystemGenericListClass()->getFullName() != fullName) return nullptr; return bs_shared_ptr_new(ConstructPrivately(), typeInfo, managedInstance); } ManagedSerializableListPtr ManagedSerializableList::createNew(const ManagedSerializableTypeInfoListPtr& typeInfo, UINT32 size) { return bs_shared_ptr_new(ConstructPrivately(), typeInfo, createManagedInstance(typeInfo, size)); } MonoObject* ManagedSerializableList::createManagedInstance(const ManagedSerializableTypeInfoListPtr& typeInfo, UINT32 size) { if (!typeInfo->isTypeLoaded()) return nullptr; ::MonoClass* listMonoClass = typeInfo->getMonoClass(); MonoClass* listClass = MonoManager::instance().findClass(listMonoClass); if (listClass == nullptr) return nullptr; void* params[1] = { &size }; MonoObject* instance = listClass->createInstance("int", params); ScriptArray tempArray(typeInfo->mElementType->getMonoClass(), size); params[0] = tempArray.getInternal(); MonoMethod* addRangeMethod = listClass->getMethod("AddRange", 1); addRangeMethod->invoke(instance, params); return instance; } ManagedSerializableListPtr ManagedSerializableList::createEmpty() { return bs_shared_ptr_new(ConstructPrivately()); } void ManagedSerializableList::setFieldData(UINT32 arrayIdx, const ManagedSerializableFieldDataPtr& val) { if (mManagedInstance != nullptr) mItemProp->setIndexed(mManagedInstance, arrayIdx, val->getValue(mListTypeInfo->mElementType)); else mCachedEntries[arrayIdx] = val; } void ManagedSerializableList::addFieldDataInternal(const ManagedSerializableFieldDataPtr& val) { void* params[1]; params[0] = val->getValue(mListTypeInfo->mElementType); mAddMethod->invoke(mManagedInstance, params); } ManagedSerializableFieldDataPtr ManagedSerializableList::getFieldData(UINT32 arrayIdx) { if (mManagedInstance != nullptr) { MonoObject* obj = mItemProp->getIndexed(mManagedInstance, arrayIdx); return ManagedSerializableFieldData::create(mListTypeInfo->mElementType, obj); } else return mCachedEntries[arrayIdx]; } void ManagedSerializableList::resize(UINT32 newSize) { if (mManagedInstance != nullptr) { ScriptArray tempArray(mListTypeInfo->mElementType->getMonoClass(), newSize); UINT32 minSize = std::min(mNumElements, newSize); UINT32 dummy = 0; void* params[4]; params[0] = &dummy;; params[1] = tempArray.getInternal(); params[2] = &dummy; params[3] = &minSize; mCopyToMethod->invoke(getManagedInstance(), params); mClearMethod->invoke(getManagedInstance(), nullptr); params[0] = tempArray.getInternal(); mAddRangeMethod->invoke(getManagedInstance(), params); } else { mCachedEntries.resize(newSize); } mNumElements = newSize; } void ManagedSerializableList::serialize() { if (mManagedInstance == nullptr) return; mNumElements = getLengthInternal(); mCachedEntries = Vector(mNumElements); for (UINT32 i = 0; i < mNumElements; i++) mCachedEntries[i] = getFieldData(i); // Serialize children for (auto& fieldEntry : mCachedEntries) fieldEntry->serialize(); mManagedInstance = nullptr; } void ManagedSerializableList::deserialize() { mManagedInstance = createManagedInstance(mListTypeInfo, mNumElements); if (mManagedInstance == nullptr) { mCachedEntries.clear(); return; } MonoClass* listClass = MonoManager::instance().findClass(mListTypeInfo->getMonoClass()); initMonoObjects(listClass); // Deserialize children for (auto& fieldEntry : mCachedEntries) fieldEntry->deserialize(); UINT32 idx = 0; for (auto& entry : mCachedEntries) { setFieldData(idx, entry); idx++; } mCachedEntries.clear(); } UINT32 ManagedSerializableList::getLengthInternal() const { MonoObject* length = mCountProp->get(mManagedInstance); if(length == nullptr) return 0; return *(UINT32*)mono_object_unbox(length); } void ManagedSerializableList::initMonoObjects(MonoClass* listClass) { mItemProp = &listClass->getProperty("Item"); mCountProp = &listClass->getProperty("Count"); mAddMethod = listClass->getMethod("Add", 1); mAddRangeMethod = listClass->getMethod("AddRange", 1); mClearMethod = listClass->getMethod("Clear"); mCopyToMethod = listClass->getMethod("CopyTo", 4); } RTTITypeBase* ManagedSerializableList::getRTTIStatic() { return ManagedSerializableListRTTI::instance(); } RTTITypeBase* ManagedSerializableList::getRTTI() const { return ManagedSerializableList::getRTTIStatic(); } }