#include "CScriptTemplate.h" #include "CScriptObject.h" #include "CMasterTemplate.h" #include #include #include #include #include CScriptTemplate::CScriptTemplate(CMasterTemplate *pMaster) { mpMaster = pMaster; mVisible = true; mVolumeShape = eNoShape; } CScriptTemplate::~CScriptTemplate() { for (u32 iSet = 0; iSet < mPropertySets.size(); iSet++) delete mPropertySets[iSet].pBaseStruct; } CMasterTemplate* CScriptTemplate::MasterTemplate() { return mpMaster; } std::string CScriptTemplate::TemplateName(s32 propCount) const { // Return original name if there is only one property set // or if caller doesn't want to distinguish between sets if ((NumPropertySets() == 1) || (propCount == -1)) return mTemplateName; // Otherwise we return the template name with the set name appended for (auto it = mPropertySets.begin(); it != mPropertySets.end(); it++) if (it->pBaseStruct->Count() == propCount) return mTemplateName + " (" + it->SetName + ")"; return mTemplateName + " (Invalid)"; } std::string CScriptTemplate::PropertySetNameByCount(s32 propCount) const { for (auto it = mPropertySets.begin(); it != mPropertySets.end(); it++) if (it->pBaseStruct->Count() == propCount) return it->SetName; return ""; } std::string CScriptTemplate::PropertySetNameByIndex(u32 index) const { if (index < NumPropertySets()) return mPropertySets[index].SetName; else return ""; } u32 CScriptTemplate::NumPropertySets() const { return mPropertySets.size(); } CScriptTemplate::ERotationType CScriptTemplate::RotationType() const { return mRotationType; } CScriptTemplate::EScaleType CScriptTemplate::ScaleType() const { return mScaleType; } u32 CScriptTemplate::ObjectID() const { return mObjectID; } void CScriptTemplate::SetVisible(bool visible) { mVisible = visible; } bool CScriptTemplate::IsVisible() const { return mVisible; } void CScriptTemplate::DebugPrintProperties(int propCount) { CStructTemplate *pTemp = BaseStructByCount(propCount); if (pTemp) pTemp->DebugPrintProperties(""); } // ************ PROPERTY FETCHING ************ template t TFetchProperty(CPropertyStruct *pProperties, const TIDString& ID) { if (ID.empty()) return nullptr; CPropertyBase *pProp = pProperties->PropertyByIDString(ID); if (pProp && (pProp->Type() == propType)) return static_cast(pProp); else return nullptr; } CStructTemplate* CScriptTemplate::BaseStructByCount(s32 propCount) { for (u32 iSet = 0; iSet < mPropertySets.size(); iSet++) if (mPropertySets[iSet].pBaseStruct->Count() == propCount) return mPropertySets[iSet].pBaseStruct; return nullptr; } CStructTemplate* CScriptTemplate::BaseStructByIndex(u32 index) { if (index < NumPropertySets()) return mPropertySets[index].pBaseStruct; else return nullptr; } EVolumeShape CScriptTemplate::VolumeShape(CScriptObject *pObj) { if (pObj->Template() != this) { Log::Error(pObj->Template()->TemplateName() + " instance somehow called VolumeShape() on " + TemplateName() + " template"); return eInvalidShape; } if (mVolumeShape == eConditionalShape) { CPropertyBase *pProp = pObj->Properties()->PropertyByIDString(mVolumeConditionIDString); // Get value of the condition test property (only boolean, integral, and enum types supported) int v; switch (pProp->Type()) { case eBoolProperty: v = (static_cast(pProp)->Get() ? 1 : 0); break; case eByteProperty: v = (int) static_cast(pProp)->Get(); break; case eShortProperty: v = (int) static_cast(pProp)->Get(); break; case eLongProperty: case eEnumProperty: v = (int) static_cast(pProp)->Get(); break; } // Test and check whether any of the conditions are true for (auto it = mVolumeConditions.begin(); it != mVolumeConditions.end(); it++) { if (it->Value == v) return it->Shape; } Log::Error(TemplateName() + " instance " + StringUtil::ToHexString(pObj->InstanceID(), true, true, 8) + " has unexpected volume shape value of " + StringUtil::ToHexString((u32) v, true, true)); return eInvalidShape; } else return mVolumeShape; } CStringProperty* CScriptTemplate::FindInstanceName(CPropertyStruct *pProperties) { return TFetchProperty(pProperties, mNameIDString); } CVector3Property* CScriptTemplate::FindPosition(CPropertyStruct *pProperties) { return TFetchProperty(pProperties, mPositionIDString); } CVector3Property* CScriptTemplate::FindRotation(CPropertyStruct *pProperties) { return TFetchProperty(pProperties, mRotationIDString); } CVector3Property* CScriptTemplate::FindScale(CPropertyStruct *pProperties) { return TFetchProperty(pProperties, mScaleIDString); } CBoolProperty* CScriptTemplate::FindActive(CPropertyStruct *pProperties) { return TFetchProperty(pProperties, mActiveIDString); } CPropertyStruct* CScriptTemplate::FindLightParameters(CPropertyStruct *pProperties) { return TFetchProperty(pProperties, mLightParametersIDString); } CModel* CScriptTemplate::FindDisplayModel(CPropertyStruct *pProperties) { for (auto it = mAssets.begin(); it != mAssets.end(); it++) { CResource *pRes = nullptr; int animSetIndex = -1; // File if (it->AssetSource == SEditorAsset::eFile) { std::string path = "../resources/" + it->AssetLocation; pRes = gResCache.GetResource(path); } // Property else { CPropertyBase *pProp = pProperties->PropertyByIDString(it->AssetLocation); if (pProp->Type() == eFileProperty) { CFileProperty *pFile = static_cast(pProp); pRes = pFile->Get(); } else if (pProp->Type() == eStructProperty) { CPropertyStruct *pStruct = static_cast(pProp); // Slightly hacky code to fetch the correct parameters for each game EGame game = mpMaster->GetGame(); if (game <= eCorruption) pRes = static_cast(pStruct->PropertyByIndex(0))->Get(); else pRes = static_cast(pStruct->PropertyByIndex(1))->Get(); if (it->ForceNodeIndex >= 0) animSetIndex = it->ForceNodeIndex; else if (game >= eCorruptionProto) animSetIndex = 0; else animSetIndex = static_cast(pStruct->PropertyByIndex(1))->Get(); } } // Verify resource exists + is correct type if (pRes) { if ((it->AssetType == SEditorAsset::eModel) && (pRes->Type() == eModel)) return static_cast(pRes); if ((it->AssetType == SEditorAsset::eAnimParams) && ((pRes->Type() == eAnimSet))) { CAnimSet *pSet = static_cast(pRes); if (animSetIndex < pSet->getNodeCount()) { CModel *pModel = pSet->getNodeModel(animSetIndex); if (pModel && (pModel->Type() == eModel)) return pModel; } } } } return nullptr; } bool CScriptTemplate::HasPosition() { return (!mPositionIDString.empty()); } // ************ OBJECT TRACKING ************ u32 CScriptTemplate::NumObjects() const { return mObjectList.size(); } const std::list& CScriptTemplate::ObjectList() const { return mObjectList; } void CScriptTemplate::AddObject(CScriptObject *pObject) { mObjectList.push_back(pObject); } void CScriptTemplate::RemoveObject(CScriptObject *pObject) { for (auto it = mObjectList.begin(); it != mObjectList.end(); it++) { if (*it == pObject) { mObjectList.erase(it); break; } } } void CScriptTemplate::SortObjects() { // todo: make this function take layer names into account mObjectList.sort([](CScriptObject *pA, CScriptObject *pB) -> bool { return (pA->InstanceID() < pB->InstanceID()); }); }