#include "CScriptGenerator.hpp" #include "CStateManager.hpp" #include "CWallCrawlerSwarm.hpp" #include "TCastTo.hpp" namespace urde { CScriptGenerator::CScriptGenerator(TUniqueId uid, std::string_view name, const CEntityInfo& info, u32 spawnCount, bool noReuseFollowers, const zeus::CVector3f& vec1, bool noInheritXf, bool active, float minScale, float maxScale) : CEntity(uid, info, active, name) , x34_spawnCount(spawnCount) , x38_24_noReuseFollowers(noReuseFollowers) , x38_25_noInheritTransform(noInheritXf) , x3c_offset(vec1) , x48_minScale(minScale) , x4c_maxScale(maxScale) {} void CScriptGenerator::Accept(IVisitor& visitor) { visitor.Visit(this); } void CScriptGenerator::AcceptScriptMsg(EScriptObjectMessage msg, TUniqueId sender, CStateManager& stateMgr) { switch (msg) { case EScriptObjectMessage::SetToZero: { if (!GetActive()) break; std::vector follows; follows.reserve(x20_conns.size()); for (const SConnection& conn : x20_conns) { if (conn.x0_state != EScriptObjectState::Zero || conn.x4_msg != EScriptObjectMessage::Follow) continue; TUniqueId uid = stateMgr.GetIdForScript(conn.x8_objId); if (stateMgr.GetObjectById(uid) != nullptr) follows.push_back(uid); } if (follows.empty()) follows.push_back(sender); std::vector> activates; activates.reserve(x20_conns.size()); for (const SConnection& conn : x20_conns) { if (conn.x0_state != EScriptObjectState::Zero) continue; TUniqueId uid = stateMgr.GetIdForScript(conn.x8_objId); if (uid == kInvalidUniqueId) continue; if (conn.x4_msg == EScriptObjectMessage::Activate) { if (!stateMgr.GetObjectById(uid)) continue; activates.emplace_back(uid, conn.x8_objId); } stateMgr.SendScriptMsgAlways(GetUniqueId(), uid, conn.x4_msg); } if (activates.empty()) break; for (u32 i = 0; i < x34_spawnCount; ++i) { if (activates.size() == 0 || follows.size() == 0) break; u32 activatesRand = 0.99f * (stateMgr.GetActiveRandom()->Float() * activates.size()); u32 followsRand = 0.99f * (stateMgr.GetActiveRandom()->Float() * follows.size()); for (u32 j = 0; j < activates.size(); ++j) if (TCastToConstPtr(stateMgr.GetObjectById(activates[j].first))) activatesRand = j; std::pair idPair = activates[activatesRand]; CEntity* activate = stateMgr.ObjectById(idPair.first); CEntity* follow = stateMgr.ObjectById(follows[followsRand]); if (!activate || !follow) break; bool oldGeneratingObject = stateMgr.GetIsGeneratingObject(); stateMgr.SetIsGeneratingObject(true); std::pair objId = stateMgr.GenerateObject(idPair.second); stateMgr.SetIsGeneratingObject(oldGeneratingObject); if (objId.second != kInvalidUniqueId) { if (CEntity* genObj = stateMgr.ObjectById(objId.second)) { TCastToPtr activateActor(genObj); TCastToPtr followActor(follow); TCastToPtr wallCrawlerSwarm(follow); if (activateActor && wallCrawlerSwarm) { if (!x38_25_noInheritTransform) activateActor->SetTransform(wallCrawlerSwarm->GetTransform()); activateActor->SetTranslation(wallCrawlerSwarm->GetLastKilledOffset() + x3c_offset); } else if (activateActor && followActor) { if (!x38_25_noInheritTransform) activateActor->SetTransform(followActor->GetTransform()); activateActor->SetTranslation(followActor->GetTranslation() + x3c_offset); } float rnd = stateMgr.GetActiveRandom()->Range(x48_minScale, x4c_maxScale); CModelData* mData = activateActor->ModelData(); if (mData && !mData->IsNull()) mData->SetScale(rnd * mData->GetScale()); stateMgr.SendScriptMsg(genObj, GetUniqueId(), EScriptObjectMessage::Activate); } } activates.erase(std::find(activates.begin(), activates.end(), idPair)); if (x38_24_noReuseFollowers) follows.erase(std::find(follows.begin(), follows.end(), follows[followsRand])); } break; } default: break; } CEntity::AcceptScriptMsg(msg, sender, stateMgr); } } // namespace urde