#include "CActorModelParticles.hpp" #include "CStateManager.hpp" #include "GameGlobalObjects.hpp" #include "CSimplePool.hpp" #include "CDependencyGroup.hpp" #include "Particle/CElementGen.hpp" #include "Particle/CParticleElectric.hpp" #include "Particle/CGenDescription.hpp" #include "World/CWorld.hpp" #include "Graphics/CBooRenderer.hpp" #include "Graphics/CSkinnedModel.hpp" #include "World/CScriptPlayerActor.hpp" #include "CPatterned.hpp" namespace urde { static bool IsMediumOrLarge(CActor& act) { if (TCastToConstPtr pat = act) return pat->GetKnockBackController().GetVariant() != EKnockBackVariant::Small; return false; } CActorModelParticles::CItem::CItem(const CEntity& ent, CActorModelParticles& parent) : x0_id(ent.GetUniqueId()), x4_areaId(ent.GetAreaIdAlways()), xdc_ashy(parent.x48_ashy), x128_parent(parent) { x8_thermalHotParticles.resize(8); } u32 GetNextBestPt(s32 start, const std::vector>& vn, CRandom16& rnd) { const zeus::CVector3f& startVec = vn[start].first; u32 ret; float lastMag = 0.f; for (s32 i = 0; i < 10; ++i) { u32 idx = u32(rnd.Range(0, s32(vn.size()) - 1)); const zeus::CVector3f& rndVec = vn[idx].first; float mag = (startVec - rndVec).magSquared(); if (mag > lastMag) { ret = idx; lastMag = mag; } } return ret; } void CActorModelParticles::CItem::GeneratePoints(const std::vector>& vn) { for (std::pair, u32>& pair : x8_thermalHotParticles) { if (pair.first) { CRandom16 rnd(pair.second); const zeus::CVector3f& vec = vn[u32(rnd.Float() * (vn.size() - 1))].first; pair.first->SetTranslation(xec_ * vec); } } if (x84_ > 0) { CRandom16 rnd(x88_seed1); u32 count = (x84_ >= 16 ? 16 : x84_); u32 idx = x80_; for (u32 i = 0; i < count; ++i) { idx = GetNextBestPt(idx, vn, rnd); x78_->SetTranslation(xec_ * vn[idx].first); zeus::CVector3f v = vn[idx].second; if (v.canBeNormalized()) { v.normalize(); x78_->SetOrientation( zeus::CTransform{zeus::CVector3f::skUp.cross(v), v, zeus::CVector3f::skUp, zeus::CVector3f::skZero}); } x78_->ForceParticleCreation(1); } x84_ -= count; x88_seed1 = rnd.GetSeed(); } if (xb0_ != -1) { CRandom16 rnd(xb4_seed2); std::unique_ptr iceGen = x128_parent.MakeIceGen(); iceGen->SetGlobalOrientAndTrans(xf8_); u32 next = GetNextBestPt(xb0_, vn, rnd); iceGen->SetTranslation(xec_ * vn[next].first); iceGen->SetOrientation(zeus::CTransform::MakeRotationsBasedOnY(zeus::CUnitVector3f(vn[next].second))); x8c_thermalColdParticles.push_back(std::move(iceGen)); xb0_ = (x8c_thermalColdParticles.size() == 4 ? -1 : next); } // TODO: Verify behavior if (xc0_particleElectric && xc0_particleElectric->GetParticleEmission()) { CRandom16 rnd(xcc_seed3); u32 end = 1; #if 0 if (4 < 1) end = 4; #endif u32 lastRnd; for (u32 i = 0; i < end; ++i) { xc0_particleElectric->SetOverrideIPos(vn[u32(rnd.Range(0, s32(vn.size()) - 1))].first * xec_); lastRnd = u32(rnd.Range(0, s32(vn.size()) - 1)); xc0_particleElectric->SetOverrideFPos(vn[lastRnd].first * xec_); xc0_particleElectric->ForceParticleCreation(1); } xcc_seed3 = rnd.GetSeed(); xc8_ = lastRnd; } // TODO: Finish #if 0 if (xd4_) { xd4_->sub_8026A5E0(v1, v2, w1); } #endif } void CActorModelParticles::CItem::sub_801e59a8(EDependency i) { if (x134_lockDeps & (1 << int(i))) { // x128_parent.sub_801e3a94(i); x134_lockDeps &= ~(1 << int(i)); } } bool CActorModelParticles::CItem::UpdateOnFire(float dt, CActor* actor, CStateManager& mgr) { bool r30 = false; bool r31 = false; x6c_onFireDelayTimer -= dt; if (x6c_onFireDelayTimer < 0.f) x6c_onFireDelayTimer = 0.f; if (x134_lockDeps & 0x1) { if (x128_parent.xe6_bits2 & 0x1) { if (x70_onFire && actor) { bool r26 = true; if (x78_ || xdc_ashy) { r26 = false; } else if (!IsMediumOrLarge(*actor)) { int r3 = 0; for (const auto& p : x8_thermalHotParticles) if (p.first) ++r3; if (r3 >= 4) r26 = false; } if (r26) { for (auto& p : x8_thermalHotParticles) { if (!p.first) { p.second = mgr.GetActiveRandom()->Next(); p.first = x128_parent.MakeOnFireGen(); x6c_onFireDelayTimer = 0.3f; break; } } } if (!x74_sfx) { x74_sfx = CSfxManager::AddEmitter(SFXsfx0480 + (IsMediumOrLarge(*actor) ? 1 : 0), actor->GetTranslation(), zeus::CVector3f::skZero, true, true, 0x7f, kInvalidAreaId); } x70_onFire = false; } for (auto& p : x8_thermalHotParticles) { if (p.first) { if (p.first->IsSystemDeletable()) p.first.reset(); else if (actor) p.first->SetGlobalOrientAndTrans(actor->GetTransform()); p.first->Update(dt); r30 = true; r31 = true; } } } else { r30 = true; } } if (x74_sfx) { if (r31) { CSfxManager::UpdateEmitter(x74_sfx, xf8_.origin, zeus::CVector3f::skZero, 1.f); } else { CSfxManager::RemoveEmitter(x74_sfx); x74_sfx.reset(); } } if (!r30) { Unlock(EDependency::OnFire); } return r30; } bool CActorModelParticles::CItem::UpdateAsh(float dt, CActor* actor, CStateManager& mgr) { return false; } bool CActorModelParticles::CItem::sub_801e65bc(float dt, CActor* actor, CStateManager& mgr) { return false; } bool CActorModelParticles::CItem::UpdateFirePop(float dt, CActor* actor, CStateManager& mgr) { return false; } bool CActorModelParticles::CItem::UpdateElectric(float dt, CActor* actor, CStateManager& mgr) { return false; } bool CActorModelParticles::CItem::sub_801e69f0(float dt, CActor* actor, CStateManager& mgr) { return false; } bool CActorModelParticles::CItem::sub_801e5e98(float dt, CActor* actor, CStateManager& mgr) { return false; } bool CActorModelParticles::CItem::UpdateIcePop(float dt, CActor* actor, CStateManager& mgr) { return false; } bool CActorModelParticles::CItem::Update(float dt, CStateManager& mgr) { CActor* act = static_cast(mgr.ObjectById(x0_id)); if (act && act->HasModelData() && !act->ModelData()->IsNull()) { xec_ = act->ModelData()->GetScale(); xf8_ = act->GetTransform(); x4_areaId = act->GetAreaIdAlways(); } else { x0_id = kInvalidUniqueId; x84_ = 0; xb0_ = -1; if (xc0_particleElectric) xc0_particleElectric->SetParticleEmission(false); if (x74_sfx) { CSfxManager::RemoveEmitter(x74_sfx); x74_sfx.reset(); } x130_ -= dt; if (x130_ <= 0.f) return false; } bool ret = false; if (UpdateOnFire(dt, act, mgr)) ret = true; if (UpdateAsh(dt, act, mgr)) ret = true; if (sub_801e65bc(dt, act, mgr)) ret = true; if (UpdateFirePop(dt, act, mgr)) ret = true; if (UpdateElectric(dt, act, mgr)) ret = true; if (sub_801e69f0(dt, act, mgr)) ret = true; if (sub_801e5e98(dt, act, mgr)) ret = true; if (UpdateIcePop(dt, act, mgr)) ret = true; return ret; } void CActorModelParticles::CItem::Lock(EDependency d) { if (!(x134_lockDeps & (1 << int(d)))) { x128_parent.IncrementDependency(d); x134_lockDeps |= (1 << int(d)); } } void CActorModelParticles::CItem::Unlock(EDependency d) { if (x134_lockDeps & (1 << int(d))) { x128_parent.DecrementDependency(d); x134_lockDeps &= ~(1 << int(d)); } } void CActorModelParticles::DecrementDependency(EDependency d) { Dependency& dep = x50_dgrps[int(d)]; dep.Decrement(); if (dep.x10_refCount == 0) { xe4_bits &= ~(1 << int(d)); xe6_bits2 &= ~(1 << int(d)); xe5_bits1 &= ~(1 << int(d)); } } void CActorModelParticles::IncrementDependency(EDependency d) { x50_dgrps[int(d)].Increment(); if (!(xe6_bits2 & (1 << int(d)))) xe4_bits |= (1 << int(d)); } static const char* ParticleDGRPs[] = { "Effect_OnFire_DGRP", "Effect_Ash_DGRP", "Effect_IceBreak_DGRP", "Effect_FirePop_DGRP", "Effect_IcePop_DGRP", "Effect_Electric_DGRP", }; CActorModelParticles::Dependency CActorModelParticles::GetParticleDGRPTokens(const char* name) { Dependency ret = {}; TToken dgrp = g_SimplePool->GetObj(name); const auto& vector = dgrp->GetObjectTagVector(); ret.x0_tokens.reserve(vector.size()); for (const SObjectTag& tag : vector) ret.x0_tokens.push_back(g_SimplePool->GetObj(tag)); return ret; } void CActorModelParticles::LoadParticleDGRPs() { for (int i = 0; i < 6; ++i) x50_dgrps.push_back(GetParticleDGRPTokens(ParticleDGRPs[i])); } std::unique_ptr CActorModelParticles::MakeOnFireGen() const { return std::make_unique(x18_onFire); } std::unique_ptr CActorModelParticles::MakeAshGen() const { return std::make_unique(x20_ash); } std::unique_ptr CActorModelParticles::MakeIceGen() const { return std::make_unique(x28_iceBreak); } std::unique_ptr CActorModelParticles::MakeFirePopGen() const { return std::make_unique(x30_firePop); } std::unique_ptr CActorModelParticles::MakeIcePopGen() const { return std::make_unique(x38_icePop); } std::unique_ptr CActorModelParticles::MakeElectricGen() const { return std::make_unique(x40_electric); } CActorModelParticles::CActorModelParticles() { x18_onFire = g_SimplePool->GetObj("Effect_OnFire"); x20_ash = g_SimplePool->GetObj("Effect_Ash"); x28_iceBreak = g_SimplePool->GetObj("Effect_IceBreak"); x30_firePop = g_SimplePool->GetObj("Effect_FirePop"); x38_icePop = g_SimplePool->GetObj("Effect_IcePop"); x40_electric = g_SimplePool->GetObj("Effect_Electric"); x48_ashy = g_SimplePool->GetObj("TXTR_Ashy"); LoadParticleDGRPs(); } void CActorModelParticles::AddStragglersToRenderer(const CStateManager& mgr) { bool isNotCold = mgr.GetThermalDrawFlag() != EThermalDrawFlag::Cold; bool isNotHot = mgr.GetThermalDrawFlag() != EThermalDrawFlag::Hot; for (CItem& item : x0_items) { if (item.x4_areaId != kInvalidAreaId) { const CGameArea* area = mgr.GetWorld()->GetAreaAlways(item.x4_areaId); if (!area->IsPostConstructed()) continue; CGameArea::EOcclusionState occState = area->GetPostConstructed()->x10dc_occlusionState; if (occState == CGameArea::EOcclusionState::Occluded) continue; } if (mgr.GetObjectById(item.x0_id) && ((isNotCold && item.x12c_24_thermalCold) || (isNotHot && item.x12c_25_thermalHot))) { item.x12c_24_thermalCold = false; item.x12c_25_thermalHot = false; continue; } if (isNotCold) { /* Hot Draw */ for (int i = 0; i < 8; ++i) { std::unique_ptr& gen = item.x8_thermalHotParticles[i].first; if (gen) g_Renderer->AddParticleGen(*gen); } if (mgr.GetThermalDrawFlag() != EThermalDrawFlag::Hot && item.x78_) g_Renderer->AddParticleGen(*item.x78_); if (item.xb8_) g_Renderer->AddParticleGen(*item.xb8_); if (item.xc0_particleElectric) g_Renderer->AddParticleGen(*item.xc0_particleElectric); } if (isNotHot) { /* Cold Draw */ for (std::unique_ptr& gen : item.x8c_thermalColdParticles) g_Renderer->AddParticleGen(*gen); if (item.xe4_) g_Renderer->AddParticleGen(*item.xe4_); } if (isNotCold) { /* Thermal Reset */ item.x12c_24_thermalCold = false; item.x12c_25_thermalHot = false; } } } void CActorModelParticles::Update(float dt, CStateManager& mgr) {} void CActorModelParticles::PointGenerator(void* ctx, const std::vector>& vn) { reinterpret_cast(ctx)->GeneratePoints(vn); } void CActorModelParticles::SetupHook(TUniqueId uid) { auto search = FindSystem(uid); if (search != x0_items.cend()) CSkinnedModel::SetPointGeneratorFunc((void*)&*search, PointGenerator); } std::list::const_iterator CActorModelParticles::FindSystem(TUniqueId uid) const { for (auto it = x0_items.cbegin(); it != x0_items.cend(); ++it) if (it->x0_id == uid) return it; return x0_items.cend(); } std::list::iterator CActorModelParticles::FindOrCreateSystem(CActor& act) { if (act.GetPointGeneratorParticles()) { for (auto it = x0_items.begin(); it != x0_items.end(); ++it) if (it->x0_id == act.GetUniqueId()) return it; } act.SetPointGeneratorParticles(true); return x0_items.emplace(x0_items.end(), act, *this); } void CActorModelParticles::StartIce(CActor& act) { auto iter = FindOrCreateSystem(act); iter->Lock(EDependency::Ash); } void CActorModelParticles::StartElectric(CActor& act) { auto iter = FindOrCreateSystem(act); if (iter->xc0_particleElectric && !iter->xc0_particleElectric->GetParticleEmission()) iter->xc0_particleElectric->SetParticleEmission(true); } void CActorModelParticles::StopElectric(CActor& act) { if (act.GetPointGeneratorParticles()) { auto iter = FindSystem(act.GetUniqueId()); if (iter != x0_items.cend() && iter->xc0_particleElectric) iter->xc0_particleElectric->SetParticleEmission(false); } } void CActorModelParticles::sub_801e51d0(CActor& act) { auto iter = FindOrCreateSystem(act); if (!iter->xc0_particleElectric) iter->Lock(EDependency::IcePop); else { if (!iter->xc0_particleElectric->GetParticleEmission()) iter->xc0_particleElectric->SetParticleEmission(true); } } void CActorModelParticles::StopThermalHotParticles(CActor& act) { if (act.GetPointGeneratorParticles()) { auto iter = FindSystem(act.GetUniqueId()); if (iter != x0_items.cend()) { for (auto& part : iter->x8_thermalHotParticles) if (part.first) part.first->SetParticleEmission(false); } } } void CActorModelParticles::StartBurnDeath(CActor& act) { auto iter = FindOrCreateSystem(act); u16 sfx = SFXeff_x_smallburndeath_lp_00 - s16(IsMediumOrLarge(act)); CSfxManager::AddEmitter(sfx, act.GetTranslation(), zeus::CVector3f::skZero, true, false, 0x7f, kInvalidAreaId); iter->xdc_ashy.Lock(); } void CActorModelParticles::EnsureElectricLoaded(CActor& act) { auto iter = FindOrCreateSystem(act); iter->Lock(EDependency::Electric); } void CActorModelParticles::EnsureFirePopLoaded(CActor& act) { auto iter = FindOrCreateSystem(act); iter->Lock(EDependency::FirePop); } void CActorModelParticles::EnsureIceBreakLoaded(CActor& act) { auto iter = FindOrCreateSystem(act); iter->Lock(EDependency::IceBreak); } void CActorModelParticles::LightDudeOnFire(CActor& act) { auto iter = FindOrCreateSystem(act); iter->Lock(EDependency::OnFire); if (iter->x6c_onFireDelayTimer <= 0.f) iter->x70_onFire = true; } const CTexture* CActorModelParticles::GetAshyTexture(const CActor& act) { auto iter = FindSystem(act.GetUniqueId()); if (iter != x0_items.cend() && iter->xdc_ashy && iter->xdc_ashy.IsLoaded()) { iter->xdc_ashy->GetBooTexture()->setClampMode(boo::TextureClampMode::ClampToEdge); return iter->xdc_ashy.GetObj(); } return nullptr; } void CActorModelParticles::AddRainSplashGenerator(CActor& act, CStateManager& mgr, u32 maxSplashes, u32 genRate, float minZ) { auto it = FindOrCreateSystem(act); if (it->xd4_rainSplashGenerator) return; if (act.GetModelData() && !act.GetModelData()->IsNull()) it->xd4_rainSplashGenerator = std::make_unique(act.GetModelData()->GetScale(), maxSplashes, genRate, minZ, 0.1875f); } void CActorModelParticles::RemoveRainSplashGenerator(CActor& act) { auto it = FindOrCreateSystem(act); it->xd4_rainSplashGenerator.reset(); } void CActorModelParticles::Render(const CActor& actor) const {} } // namespace urde