mirror of
				https://github.com/AxioDL/metaforce.git
				synced 2025-10-26 21:30:25 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			1839 lines
		
	
	
		
			69 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1839 lines
		
	
	
		
			69 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "Runtime/World/CPatterned.hpp"
 | |
| 
 | |
| #include "Runtime/Camera/CFirstPersonCamera.hpp"
 | |
| #include "Runtime/Character/CAnimData.hpp"
 | |
| #include "Runtime/Character/CPASAnimParmData.hpp"
 | |
| #include "Runtime/GameGlobalObjects.hpp"
 | |
| #include "Runtime/Graphics/CSkinnedModel.hpp"
 | |
| #include "Runtime/MP1/World/CMetroid.hpp"
 | |
| #include "Runtime/MP1/World/CSpacePirate.hpp"
 | |
| #include "Runtime/Weapon/CEnergyProjectile.hpp"
 | |
| #include "Runtime/Weapon/CGameProjectile.hpp"
 | |
| #include "Runtime/World/CActorParameters.hpp"
 | |
| #include "Runtime/World/CExplosion.hpp"
 | |
| #include "Runtime/World/CPathFindSearch.hpp"
 | |
| #include "Runtime/World/CPatternedInfo.hpp"
 | |
| #include "Runtime/World/CPlayer.hpp"
 | |
| #include "Runtime/World/CScriptActorKeyframe.hpp"
 | |
| #include "Runtime/World/CScriptCoverPoint.hpp"
 | |
| #include "Runtime/World/CScriptWaypoint.hpp"
 | |
| #include "Runtime/World/CStateMachine.hpp"
 | |
| 
 | |
| #include <hecl/CVarManager.hpp>
 | |
| 
 | |
| #include "TCastTo.hpp" // Generated file, do not modify include path
 | |
| #include <cmath>
 | |
| 
 | |
| namespace metaforce {
 | |
| namespace {
 | |
| hecl::CVar* cv_disableAi = nullptr;
 | |
| } // namespace
 | |
| 
 | |
| constexpr CMaterialList skPatternedGroundMaterialList(EMaterialTypes::Character, EMaterialTypes::Solid,
 | |
|                                                       EMaterialTypes::Orbit, EMaterialTypes::GroundCollider,
 | |
|                                                       EMaterialTypes::Target);
 | |
| constexpr CMaterialList skPatternedFlyerMaterialList(EMaterialTypes::Character, EMaterialTypes::Solid,
 | |
|                                                      EMaterialTypes::Orbit, EMaterialTypes::Target);
 | |
| 
 | |
| CPatterned::CPatterned(ECharacter character, TUniqueId uid, std::string_view name, CPatterned::EFlavorType flavor,
 | |
|                        const CEntityInfo& info, const zeus::CTransform& xf, CModelData&& mData,
 | |
|                        const CPatternedInfo& pInfo, CPatterned::EMovementType moveType,
 | |
|                        CPatterned::EColliderType colliderType, EBodyType bodyType, const CActorParameters& actorParms,
 | |
|                        EKnockBackVariant kbVariant)
 | |
| : CAi(uid, pInfo.xf8_active, name, info, xf, std::move(mData),
 | |
|       zeus::CAABox(pInfo.xcc_bodyOrigin - zeus::CVector3f{pInfo.xc4_halfExtent, pInfo.xc4_halfExtent, 0.f},
 | |
|                    pInfo.xcc_bodyOrigin +
 | |
|                        zeus::CVector3f{pInfo.xc4_halfExtent, pInfo.xc4_halfExtent, pInfo.xc8_height}),
 | |
|       pInfo.x0_mass, pInfo.x54_healthInfo, pInfo.x5c_damageVulnerability,
 | |
|       moveType == EMovementType::Flyer ? skPatternedFlyerMaterialList : skPatternedGroundMaterialList,
 | |
|       pInfo.xfc_stateMachineId, actorParms, pInfo.xd8_stepUpHeight, 0.8f)
 | |
| , x2fc_minAttackRange(pInfo.x18_minAttackRange)
 | |
| , x300_maxAttackRange(pInfo.x1c_maxAttackRange)
 | |
| , x304_averageAttackTime(pInfo.x20_averageAttackTime)
 | |
| , x308_attackTimeVariation(pInfo.x24_attackTimeVariation)
 | |
| , x328_25_verticalMovement(moveType == EMovementType::Flyer)
 | |
| , x328_27_onGround(moveType != EMovementType::Flyer)
 | |
| , x34c_character(character)
 | |
| , x388_anim(pInfo.GetAnimationParameters().GetInitialAnimation())
 | |
| , x3b4_speed(pInfo.x4_speed)
 | |
| , x3b8_turnSpeed(pInfo.x8_turnSpeed)
 | |
| , x3bc_detectionRange(pInfo.xc_detectionRange)
 | |
| , x3c0_detectionHeightRange(pInfo.x10_detectionHeightRange)
 | |
| , x3c4_detectionAngle(std::cos(zeus::degToRad(pInfo.x14_dectectionAngle)))
 | |
| , x3c8_leashRadius(pInfo.x28_leashRadius)
 | |
| , x3cc_playerLeashRadius(pInfo.x2c_playerLeashRadius)
 | |
| , x3d0_playerLeashTime(pInfo.x30_playerLeashTime)
 | |
| , x3d8_xDamageThreshold(pInfo.xdc_xDamage)
 | |
| , x3dc_frozenXDamageThreshold(pInfo.xe0_frozenXDamage)
 | |
| , x3e0_xDamageDelay(pInfo.xe4_xDamageDelay)
 | |
| , x3fc_flavor(flavor)
 | |
| , x400_31_isFlyer(moveType == CPatterned::EMovementType::Flyer)
 | |
| , x402_30_updateThermalFrozenState(x402_31_thawed = actorParms.HasThermalHeat())
 | |
| , x460_knockBackController(kbVariant) {
 | |
|   x404_contactDamage = pInfo.x34_contactDamageInfo;
 | |
|   x424_damageWaitTime = pInfo.x50_damageWaitTime;
 | |
|   x454_deathSfx = pInfo.xe8_deathSfx;
 | |
|   x458_iceShatterSfx = pInfo.x134_iceShatterSfx;
 | |
|   x4f4_intoFreezeDur = pInfo.x100_intoFreezeDur;
 | |
|   x4f8_outofFreezeDur = pInfo.x104_outofFreezeDur;
 | |
|   x4fc_freezeDur = pInfo.x108_freezeDur;
 | |
|   x508_colliderType = colliderType;
 | |
|   x50c_baseDamageMag = actorParms.GetThermalMag();
 | |
|   x514_deathExplosionOffset = pInfo.x110_particle1Scale;
 | |
|   x540_iceDeathExplosionOffset = pInfo.x124_particle2Scale;
 | |
| 
 | |
|   if (pInfo.x11c_particle1.IsValid()) {
 | |
|     x520_deathExplosionParticle = {g_SimplePool->GetObj({FOURCC('PART'), pInfo.x11c_particle1})};
 | |
|   }
 | |
| 
 | |
|   if (pInfo.x120_electric.IsValid()) {
 | |
|     x530_deathExplosionElectric = {g_SimplePool->GetObj({FOURCC('ELSC'), pInfo.x120_electric})};
 | |
|   }
 | |
| 
 | |
|   if (pInfo.x130_particle2.IsValid()) {
 | |
|     x54c_iceDeathExplosionParticle = {g_SimplePool->GetObj({FOURCC('PART'), pInfo.x130_particle2})};
 | |
|   }
 | |
| 
 | |
|   if (x404_contactDamage.GetRadius() > 0.f) {
 | |
|     x404_contactDamage.SetRadius(0.f);
 | |
|   }
 | |
| 
 | |
|   xe6_29_renderParticleDBInside = false;
 | |
|   if (x64_modelData) {
 | |
|     x402_27_noXrayModel = !x64_modelData->HasModel(CModelData::EWhichModel::XRay);
 | |
|     BuildBodyController(bodyType);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CPatterned::Accept(metaforce::IVisitor& visitor) { visitor.Visit(this); }
 | |
| 
 | |
| void CPatterned::AcceptScriptMsg(EScriptObjectMessage msg, TUniqueId uid, CStateManager& mgr) {
 | |
|   CAi::AcceptScriptMsg(msg, uid, mgr);
 | |
| 
 | |
|   switch (msg) {
 | |
|   case EScriptObjectMessage::Registered: {
 | |
|     if (x508_colliderType != EColliderType::One) {
 | |
|       CMaterialList include = GetMaterialFilter().GetIncludeList();
 | |
|       CMaterialList exclude = GetMaterialFilter().GetExcludeList();
 | |
|       include.Remove(EMaterialTypes::Character);
 | |
|       exclude.Add(EMaterialTypes::Character);
 | |
|       SetMaterialFilter(CMaterialFilter::MakeIncludeExclude(include, exclude));
 | |
|     }
 | |
| 
 | |
|     if (HasModelData() && GetModelData()->HasAnimData() && GetModelData()->GetAnimationData()->GetIceModel()) {
 | |
|       const auto& baseAABB = GetBaseBoundingBox();
 | |
|       float diagExtent = (baseAABB.max - baseAABB.min).magnitude() * 0.5f;
 | |
|       x510_vertexMorph = std::make_shared<CVertexMorphEffect>(zeus::skRight, zeus::CVector3f{}, diagExtent, 0.f,
 | |
|                                                               *mgr.GetActiveRandom());
 | |
|     }
 | |
| 
 | |
|     xf8_25_angularEnabled = true;
 | |
|     break;
 | |
|   }
 | |
|   case EScriptObjectMessage::OnFloor: {
 | |
|     if (!x328_25_verticalMovement) {
 | |
|       x150_momentum = {};
 | |
|       AddMaterial(EMaterialTypes::GroundCollider, mgr);
 | |
|     }
 | |
|     x328_27_onGround = true;
 | |
|     break;
 | |
|   }
 | |
|   case EScriptObjectMessage::Falling: {
 | |
|     if (!x328_25_verticalMovement) {
 | |
|       if (x450_bodyController->GetPercentageFrozen() == 0.f) {
 | |
|         x150_momentum = {0.f, 0.f, -GetWeight()};
 | |
|         RemoveMaterial(EMaterialTypes::GroundCollider, mgr);
 | |
|       }
 | |
|     }
 | |
|     x328_27_onGround = false;
 | |
|     break;
 | |
|   }
 | |
|   case EScriptObjectMessage::Activate:
 | |
|     x3a0_latestLeashPosition = GetTranslation();
 | |
|     break;
 | |
|   case EScriptObjectMessage::Deleted:
 | |
|     if (x330_stateMachineState.GetActorState() != nullptr) {
 | |
|       x330_stateMachineState.GetActorState()->CallFunc(mgr, *this, EStateMsg::Deactivate, 0.f);
 | |
|     }
 | |
|     break;
 | |
|   case EScriptObjectMessage::Damage: {
 | |
|     if (TCastToConstPtr<CGameProjectile> proj = mgr.GetObjectById(uid)) {
 | |
|       const CDamageInfo& info = proj->GetDamageInfo();
 | |
|       if (info.GetWeaponMode().GetType() == EWeaponType::Wave) {
 | |
|         if (x460_knockBackController.x81_26_enableShock && info.GetWeaponMode().IsComboed() &&
 | |
|             HealthInfo(mgr) != nullptr) {
 | |
|           x401_31_nextPendingShock = true;
 | |
|           KnockBack(GetTransform().frontVector(), mgr, info, EKnockBackType::Direct, false, info.GetKnockBackPower());
 | |
|           x460_knockBackController.DeferKnockBack(EWeaponType::Wave);
 | |
|         }
 | |
|       } else if (info.GetWeaponMode().GetType() == EWeaponType::Plasma) {
 | |
|         if (x460_knockBackController.x81_27_enableBurn && info.GetWeaponMode().IsCharged() &&
 | |
|             HealthInfo(mgr) != nullptr) {
 | |
|           KnockBack(GetTransform().frontVector(), mgr, info, EKnockBackType::Direct, false, info.GetKnockBackPower());
 | |
|           x460_knockBackController.DeferKnockBack(EWeaponType::Plasma);
 | |
|         }
 | |
|       }
 | |
|       if (mgr.GetPlayer().GetUniqueId() == proj->GetOwnerId()) {
 | |
|         x400_24_hitByPlayerProjectile = true;
 | |
|       }
 | |
|     }
 | |
|     break;
 | |
|   }
 | |
|   case EScriptObjectMessage::InvulnDamage: {
 | |
|     if (TCastToConstPtr<CGameProjectile> proj = mgr.GetObjectById(uid)) {
 | |
|       if (proj->GetOwnerId() == mgr.GetPlayer().GetUniqueId()) {
 | |
|         x400_24_hitByPlayerProjectile = true;
 | |
|       }
 | |
|     }
 | |
|     break;
 | |
|   }
 | |
|   default:
 | |
|     break;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CPatterned::MakeThermalColdAndHot() {
 | |
|   x403_24_keepThermalVisorState = true;
 | |
|   xe6_27_thermalVisorFlags = 3;
 | |
| }
 | |
| 
 | |
| void CPatterned::UpdateThermalFrozenState(bool thawed) {
 | |
|   x402_31_thawed = thawed;
 | |
|   if (x403_24_keepThermalVisorState) {
 | |
|     return;
 | |
|   }
 | |
|   xe6_27_thermalVisorFlags = u8(thawed ? 2 : 1);
 | |
| }
 | |
| 
 | |
| void CPatterned::Think(float dt, CStateManager& mgr) {
 | |
|   if (!GetActive() || (cv_disableAi != nullptr && cv_disableAi->toBoolean())) {
 | |
|     Stop();
 | |
|     ClearForcesAndTorques();
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (x402_30_updateThermalFrozenState) {
 | |
|     UpdateThermalFrozenState(x450_bodyController->GetPercentageFrozen() == 0.f);
 | |
|   }
 | |
| 
 | |
|   if (x64_modelData->GetAnimationData()->GetIceModel()) {
 | |
|     x510_vertexMorph->Update(dt);
 | |
|   }
 | |
| 
 | |
|   if (x402_26_dieIf80PercFrozen) {
 | |
|     float froz = x450_bodyController->GetPercentageFrozen();
 | |
|     if (froz > 0.8f) {
 | |
|       x400_29_pendingMassiveFrozenDeath = true;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (!x400_25_alive) {
 | |
|     if ((x400_28_pendingMassiveDeath || x400_29_pendingMassiveFrozenDeath) && x3e0_xDamageDelay <= 0.f) {
 | |
|       if (x400_29_pendingMassiveFrozenDeath) {
 | |
|         SendScriptMsgs(EScriptObjectState::AboutToMassivelyDie, mgr, EScriptObjectMessage::None);
 | |
|         MassiveFrozenDeath(mgr);
 | |
|       } else {
 | |
|         SendScriptMsgs(EScriptObjectState::AboutToMassivelyDie, mgr, EScriptObjectMessage::None);
 | |
|         MassiveDeath(mgr);
 | |
|       }
 | |
|     } else {
 | |
|       x3e0_xDamageDelay -= dt;
 | |
|       if (x403_26_stateControlledMassiveDeath && x330_stateMachineState.GetName() != nullptr) {
 | |
|         bool isDead = x330_stateMachineState.GetName() == "Dead"sv;
 | |
|         if (isDead && x330_stateMachineState.x8_time > 15.f) {
 | |
|           MassiveDeath(mgr);
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   UpdateAlphaDelta(dt, mgr);
 | |
| 
 | |
|   x3e4_lastHP = HealthInfo(mgr)->GetHP();
 | |
|   if (x330_stateMachineState.x4_state == nullptr) {
 | |
|     x330_stateMachineState.SetState(mgr, *this, GetStateMachine(), "Start"sv);
 | |
|   }
 | |
| 
 | |
|   zeus::CVector3f diffVec = x4e4_latestPredictedTranslation - GetTranslation();
 | |
|   if (!x328_25_verticalMovement) {
 | |
|     diffVec.z() = 0.f;
 | |
|   }
 | |
| 
 | |
|   if (diffVec.magSquared() > (0.1f * dt)) {
 | |
|     x4f0_predictedLeashTime += dt;
 | |
|   } else {
 | |
|     x4f0_predictedLeashTime = 0.f;
 | |
|   }
 | |
| 
 | |
|   if (x460_knockBackController.x81_26_enableShock) {
 | |
|     /* Shock on logical falling edge */
 | |
|     if (!x401_31_nextPendingShock && x402_24_pendingShock) {
 | |
|       Shock(mgr, 0.5f + mgr.GetActiveRandom()->Range(0.f, 0.5f), 0.2f);
 | |
|     }
 | |
|     x402_24_pendingShock = x401_31_nextPendingShock;
 | |
|     x401_31_nextPendingShock = false;
 | |
| 
 | |
|     if (x450_bodyController->IsElectrocuting()) {
 | |
|       mgr.GetActorModelParticles()->StartElectric(*this);
 | |
|       if (x3f0_pendingShockDamage > 0.f && x400_25_alive) {
 | |
|         const CDamageInfo dInfo(CDamageInfo{CWeaponMode{EWeaponType::Wave}, x3f0_pendingShockDamage, 0.f, 0.f}, dt);
 | |
|         mgr.ApplyDamage(kInvalidUniqueId, GetUniqueId(), kInvalidUniqueId, dInfo,
 | |
|                         CMaterialFilter::MakeIncludeExclude({EMaterialTypes::Solid}, {}), {});
 | |
|       }
 | |
|     } else {
 | |
|       if (x3f0_pendingShockDamage != 0.f) {
 | |
|         x450_bodyController->StopElectrocution();
 | |
|         mgr.GetActorModelParticles()->StopElectric(*this);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (x450_bodyController->IsOnFire()) {
 | |
|     if (x400_25_alive) {
 | |
|       mgr.GetActorModelParticles()->LightDudeOnFire(*this);
 | |
|       const CDamageInfo dInfo(CDamageInfo{CWeaponMode{EWeaponType::Plasma}, x3ec_pendingFireDamage, 0.f, 0.f}, dt);
 | |
|       mgr.ApplyDamage(kInvalidUniqueId, GetUniqueId(), kInvalidUniqueId, dInfo,
 | |
|                       CMaterialFilter::MakeIncludeExclude({EMaterialTypes::Solid}, {}), {});
 | |
|     }
 | |
|   } else {
 | |
|     if (x3ec_pendingFireDamage > 0.f) {
 | |
|       x3ec_pendingFireDamage = 0.f;
 | |
|     }
 | |
|     if (x450_bodyController->IsFrozen()) {
 | |
|       mgr.GetActorModelParticles()->StopThermalHotParticles(*this);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (x401_27_phazingOut || x401_28_burning) {
 | |
|     x3e8_alphaDelta = -0.33333334f;
 | |
|   }
 | |
| 
 | |
|   if (x401_30_pendingDeath) {
 | |
|     x401_30_pendingDeath = false;
 | |
|     Death(mgr, GetTransform().frontVector(), EScriptObjectState::DeathRattle);
 | |
|   }
 | |
| 
 | |
|   float thinkDt = (x400_25_alive ? dt : dt * CalcDyingThinkRate());
 | |
| 
 | |
|   x450_bodyController->Update(thinkDt, mgr);
 | |
|   x450_bodyController->MultiplyPlaybackRate(x3b4_speed);
 | |
|   SAdvancementDeltas deltas = UpdateAnimation(thinkDt, mgr, !x450_bodyController->IsFrozen());
 | |
|   x434_posDelta = deltas.x0_posDelta;
 | |
|   x440_rotDelta = deltas.xc_rotDelta;
 | |
| 
 | |
|   if (x403_25_enableStateMachine && x450_bodyController->GetPercentageFrozen() < 1.f) {
 | |
|     x330_stateMachineState.Update(mgr, *this, thinkDt);
 | |
|   }
 | |
| 
 | |
|   ThinkAboutMove(thinkDt);
 | |
| 
 | |
|   x460_knockBackController.Update(thinkDt, mgr, *this);
 | |
|   x4e4_latestPredictedTranslation = GetTranslation() + PredictMotion(thinkDt).x0_translation;
 | |
|   x328_26_solidCollision = false;
 | |
|   if (x420_curDamageRemTime > 0.f) {
 | |
|     x420_curDamageRemTime -= dt;
 | |
|   }
 | |
| 
 | |
|   if (x401_28_burning && x3f4_burnThinkRateTimer > dt) {
 | |
|     x3f4_burnThinkRateTimer -= dt;
 | |
|   }
 | |
| 
 | |
|   xd0_damageMag = x50c_baseDamageMag;
 | |
|   UpdateDamageColor(dt);
 | |
| 
 | |
|   if (!x450_bodyController->IsFrozen()) {
 | |
|     if (x3a0_latestLeashPosition == zeus::CVector3f()) {
 | |
|       x3a0_latestLeashPosition = GetTranslation();
 | |
|     }
 | |
| 
 | |
|     if (x3cc_playerLeashRadius != 0.f) {
 | |
|       if ((GetTranslation() - mgr.GetPlayer().GetTranslation()).magSquared() >
 | |
|           x3cc_playerLeashRadius * x3cc_playerLeashRadius) {
 | |
|         x3d4_curPlayerLeashTime += dt;
 | |
|       } else {
 | |
|         x3d4_curPlayerLeashTime = 0.f;
 | |
|       }
 | |
|     }
 | |
|   } else {
 | |
|     RemoveEmitter();
 | |
|   }
 | |
| 
 | |
|   if (x2f8_waypointPauseRemTime > 0.f) {
 | |
|     x2f8_waypointPauseRemTime -= dt;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CPatterned::CollidedWith(TUniqueId other, const CCollisionInfoList& list, CStateManager& mgr) {
 | |
|   if (x420_curDamageRemTime <= 0.f) {
 | |
|     if (TCastToPtr<CPlayer> player = mgr.ObjectById(other)) {
 | |
|       bool jumpOnHead =
 | |
|           player->GetTimeSinceJump() < 5.f && list.GetCount() != 0 && list.Front().GetNormalLeft().z() > 0.707f;
 | |
|       if (x400_25_alive || jumpOnHead) {
 | |
|         CDamageInfo cDamage = GetContactDamage();
 | |
|         if (!x400_25_alive || x450_bodyController->IsFrozen()) {
 | |
|           cDamage.SetDamage(0.f);
 | |
|         }
 | |
|         if (jumpOnHead) {
 | |
|           mgr.ApplyDamage(GetUniqueId(), player->GetUniqueId(), GetUniqueId(), cDamage,
 | |
|                           CMaterialFilter::skPassEverything, -player->GetVelocity());
 | |
|           player->ResetTimeSinceJump();
 | |
|         } else if (x400_25_alive && !x450_bodyController->IsFrozen()) {
 | |
|           mgr.ApplyDamage(GetUniqueId(), player->GetUniqueId(), GetUniqueId(), cDamage,
 | |
|                           CMaterialFilter::MakeIncludeExclude({EMaterialTypes::Solid}, {}), zeus::skZero3f);
 | |
|         }
 | |
|         x420_curDamageRemTime = x424_damageWaitTime;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   static constexpr CMaterialList testList(EMaterialTypes::Solid, EMaterialTypes::Ceiling, EMaterialTypes::Wall,
 | |
|                                           EMaterialTypes::Floor, EMaterialTypes::Character);
 | |
|   for (const CCollisionInfo& info : list) {
 | |
|     if (info.GetMaterialLeft().Intersection(testList) != 0u) {
 | |
|       if (!info.GetMaterialLeft().HasMaterial(EMaterialTypes::Floor)) {
 | |
|         if (!x310_moveVec.isZero() && info.GetNormalLeft().dot(x310_moveVec) >= 0.f) {
 | |
|           continue;
 | |
|         }
 | |
|       } else if (!x400_31_isFlyer) {
 | |
|         continue;
 | |
|       }
 | |
|       x328_26_solidCollision = true;
 | |
|       return;
 | |
|     }
 | |
|   }
 | |
|   CPhysicsActor::CollidedWith(other, list, mgr);
 | |
| }
 | |
| 
 | |
| void CPatterned::Touch(CActor& act, CStateManager& mgr) {
 | |
|   if (!x400_25_alive) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (TCastToPtr<CGameProjectile> proj = act) {
 | |
|     if (mgr.GetPlayer().GetUniqueId() == proj->GetOwnerId()) {
 | |
|       x400_24_hitByPlayerProjectile = true;
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| std::optional<zeus::CAABox> CPatterned::GetTouchBounds() const { return GetBoundingBox(); }
 | |
| 
 | |
| bool CPatterned::CanRenderUnsorted(const metaforce::CStateManager& mgr) const {
 | |
|   return x64_modelData->GetAnimationData()->GetParticleDB().AreAnySystemsDrawnWithModel()
 | |
|              ? false
 | |
|              : CActor::CanRenderUnsorted(mgr);
 | |
| }
 | |
| 
 | |
| zeus::CVector3f CPatterned::GetAimPosition(const metaforce::CStateManager& mgr, float dt) const {
 | |
|   zeus::CVector3f offset;
 | |
|   if (dt > 0.f) {
 | |
|     offset = PredictMotion(dt).x0_translation;
 | |
|   }
 | |
| 
 | |
|   const CSegId segId = GetModelData()->GetAnimationData()->GetLocatorSegId("lockon_target_LCTR"sv);
 | |
|   if (segId.IsValid()) {
 | |
|     const zeus::CTransform xf = GetModelData()->GetAnimationData()->GetLocatorTransform(segId, nullptr);
 | |
|     const zeus::CVector3f scaledOrigin = GetModelData()->GetScale() * xf.origin;
 | |
| 
 | |
|     if (const auto tb = GetTouchBounds()) {
 | |
|       return offset + tb->clampToBox(x34_transform * scaledOrigin);
 | |
|     }
 | |
| 
 | |
|     const zeus::CAABox aabox = GetBaseBoundingBox();
 | |
|     const zeus::CAABox primBox(aabox.min + GetPrimitiveOffset(), aabox.max + GetPrimitiveOffset());
 | |
| 
 | |
|     return offset + (x34_transform * primBox.clampToBox(scaledOrigin));
 | |
|   }
 | |
| 
 | |
|   return offset + GetBoundingBox().center();
 | |
| }
 | |
| 
 | |
| zeus::CTransform CPatterned::GetLctrTransform(std::string_view name) const {
 | |
|   return x34_transform * GetScaledLocatorTransform(name);
 | |
| }
 | |
| 
 | |
| zeus::CTransform CPatterned::GetLctrTransform(CSegId id) const {
 | |
|   zeus::CTransform xf = x64_modelData->GetAnimationData()->GetLocatorTransform(id, nullptr);
 | |
|   return x34_transform * zeus::CTransform(xf.buildMatrix3f(), x64_modelData->GetScale() * xf.origin);
 | |
| }
 | |
| 
 | |
| void CPatterned::DeathDelete(CStateManager& mgr) {
 | |
|   SendScriptMsgs(EScriptObjectState::Dead, mgr, EScriptObjectMessage::None);
 | |
|   if (x450_bodyController->IsElectrocuting()) {
 | |
|     x3f0_pendingShockDamage = 0.f;
 | |
|     x450_bodyController->StopElectrocution();
 | |
|     mgr.GetActorModelParticles()->StopElectric(*this);
 | |
|   }
 | |
|   mgr.FreeScriptObject(GetUniqueId());
 | |
| }
 | |
| 
 | |
| void CPatterned::Death(CStateManager& mgr, const zeus::CVector3f& dir, EScriptObjectState state) {
 | |
|   if (x400_25_alive) {
 | |
|     if (!x450_bodyController->IsOnFire()) {
 | |
|       x402_25_lostMassiveFrozenHP = (x3e4_lastHP - HealthInfo(mgr)->GetHP()) >= x3dc_frozenXDamageThreshold;
 | |
|       if (x402_25_lostMassiveFrozenHP && x54c_iceDeathExplosionParticle &&
 | |
|           x450_bodyController->GetPercentageFrozen() > 0.8f) {
 | |
|         x400_29_pendingMassiveFrozenDeath = true;
 | |
|       } else if ((x3e4_lastHP - HealthInfo(mgr)->GetHP()) >= x3d8_xDamageThreshold) {
 | |
|         x400_28_pendingMassiveDeath = true;
 | |
|       }
 | |
|     }
 | |
|     if (x400_28_pendingMassiveDeath || x400_29_pendingMassiveFrozenDeath) {
 | |
|       if (x328_30_lookAtDeathDir && x3e0_xDamageDelay <= 0.f && dir != zeus::skZero3f) {
 | |
|         SetTransform(zeus::lookAt(GetTranslation(), GetTranslation() - dir) *
 | |
|                      zeus::CTransform::RotateX(zeus::degToRad(45.f)));
 | |
|       }
 | |
|     } else {
 | |
|       x330_stateMachineState.SetState(mgr, *this, GetStateMachine(), "Dead"sv);
 | |
|       RemoveMaterial(EMaterialTypes::GroundCollider, mgr);
 | |
|       x328_25_verticalMovement = false;
 | |
|     }
 | |
|     x400_25_alive = false;
 | |
|     if (x450_bodyController->HasBodyState(pas::EAnimationState::Hurled) &&
 | |
|         x450_bodyController->GetBodyType() == EBodyType::Flyer) {
 | |
|       x450_bodyController->GetCommandMgr().DeliverCmd(CBCHurledCmd(-dir, zeus::skZero3f));
 | |
|     } else if (x450_bodyController->HasBodyState(pas::EAnimationState::Fall)) {
 | |
|       x450_bodyController->GetCommandMgr().DeliverCmd(CBCKnockDownCmd(-dir, pas::ESeverity::One));
 | |
|     }
 | |
|     if (state != EScriptObjectState::Any) {
 | |
|       SendScriptMsgs(state, mgr, EScriptObjectMessage::None);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CPatterned::KnockBack(const zeus::CVector3f& backVec, CStateManager& mgr, const CDamageInfo& info,
 | |
|                            EKnockBackType type, bool inDeferred, float magnitude) {
 | |
|   CHealthInfo* hInfo = HealthInfo(mgr);
 | |
|   if (!x401_27_phazingOut && !x401_28_burning && hInfo != nullptr) {
 | |
|     x460_knockBackController.KnockBack(backVec, mgr, *this, info, type, magnitude);
 | |
|     if (x450_bodyController->IsFrozen() && x460_knockBackController.GetActiveParms().xc_intoFreezeDur >= 0.f) {
 | |
|       x450_bodyController->FrozenBreakout();
 | |
|     }
 | |
|     switch (x460_knockBackController.GetActiveParms().x4_animFollowup) {
 | |
|     case EKnockBackAnimationFollowUp::Freeze:
 | |
|       Freeze(mgr, zeus::skZero3f, zeus::CUnitVector3f(x34_transform.transposeRotate(backVec)),
 | |
|              x460_knockBackController.GetActiveParms().x8_followupDuration);
 | |
|       break;
 | |
|     case EKnockBackAnimationFollowUp::PhazeOut:
 | |
|       PhazeOut(mgr);
 | |
|       break;
 | |
|     case EKnockBackAnimationFollowUp::Shock:
 | |
|       Shock(mgr, x460_knockBackController.GetActiveParms().x8_followupDuration, -1.f);
 | |
|       break;
 | |
|     case EKnockBackAnimationFollowUp::Burn:
 | |
|       Burn(x460_knockBackController.GetActiveParms().x8_followupDuration, 0.25f);
 | |
|       break;
 | |
|     case EKnockBackAnimationFollowUp::LaggedBurnDeath:
 | |
|       x401_29_laggedBurnDeath = true;
 | |
|       [[fallthrough]];
 | |
|     case EKnockBackAnimationFollowUp::BurnDeath:
 | |
|       Burn(x460_knockBackController.GetActiveParms().x8_followupDuration, -1.f);
 | |
|       Death(mgr, zeus::skZero3f, EScriptObjectState::DeathRattle);
 | |
|       x400_28_pendingMassiveDeath = x400_29_pendingMassiveFrozenDeath = false;
 | |
|       x400_27_fadeToDeath = x401_28_burning = true;
 | |
|       x3f4_burnThinkRateTimer = 1.5f;
 | |
|       x402_29_drawParticles = false;
 | |
|       x450_bodyController->DouseFlames();
 | |
|       mgr.GetActorModelParticles()->StopThermalHotParticles(*this);
 | |
|       mgr.GetActorModelParticles()->StartBurnDeath(*this);
 | |
|       if (!x401_29_laggedBurnDeath) {
 | |
|         mgr.GetActorModelParticles()->EnsureFirePopLoaded(*this);
 | |
|         mgr.GetActorModelParticles()->EnsureIceBreakLoaded(*this);
 | |
|       }
 | |
|       break;
 | |
|     case EKnockBackAnimationFollowUp::Death:
 | |
|       Death(mgr, zeus::skZero3f, EScriptObjectState::DeathRattle);
 | |
|       break;
 | |
|     case EKnockBackAnimationFollowUp::ExplodeDeath:
 | |
|       Death(mgr, zeus::skZero3f, EScriptObjectState::DeathRattle);
 | |
|       if (GetDeathExplosionParticle() || x530_deathExplosionElectric) {
 | |
|         MassiveDeath(mgr);
 | |
|       } else if (x450_bodyController->IsFrozen()) {
 | |
|         x450_bodyController->FrozenBreakout();
 | |
|       }
 | |
|       break;
 | |
|     case EKnockBackAnimationFollowUp::IceDeath:
 | |
|       Death(mgr, zeus::skZero3f, EScriptObjectState::DeathRattle);
 | |
|       if (x54c_iceDeathExplosionParticle) {
 | |
|         MassiveFrozenDeath(mgr);
 | |
|       } else if (x450_bodyController->IsFrozen()) {
 | |
|         x450_bodyController->FrozenBreakout();
 | |
|       }
 | |
|       break;
 | |
|     default:
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CPatterned::TakeDamage(const zeus::CVector3f&, float arg) { x428_damageCooldownTimer = 0.33f; }
 | |
| 
 | |
| bool CPatterned::FixedRandom(CStateManager&, float arg) {
 | |
|   return x330_stateMachineState.GetRandom() < x330_stateMachineState.x14_;
 | |
| }
 | |
| 
 | |
| bool CPatterned::Random(CStateManager&, float arg) { return x330_stateMachineState.GetRandom() < arg; }
 | |
| 
 | |
| bool CPatterned::CodeTrigger(CStateManager&, float arg) { return x330_stateMachineState.x18_24_codeTrigger; }
 | |
| 
 | |
| bool CPatterned::FixedDelay(CStateManager&, float arg) {
 | |
|   return x330_stateMachineState.GetTime() > x330_stateMachineState.GetDelay();
 | |
| }
 | |
| 
 | |
| bool CPatterned::RandomDelay(CStateManager&, float arg) {
 | |
|   return x330_stateMachineState.GetTime() > arg * x330_stateMachineState.GetRandom();
 | |
| }
 | |
| 
 | |
| bool CPatterned::Delay(CStateManager&, float arg) { return x330_stateMachineState.GetTime() > arg; }
 | |
| 
 | |
| bool CPatterned::PatrolPathOver(CStateManager&, float arg) { return x2dc_destObj == kInvalidUniqueId; }
 | |
| 
 | |
| bool CPatterned::Stuck(CStateManager&, float arg) { return x4f0_predictedLeashTime > 0.2f; }
 | |
| 
 | |
| bool CPatterned::AnimOver(CStateManager&, float arg) { return x32c_animState == EAnimState::Over; }
 | |
| 
 | |
| bool CPatterned::InPosition(CStateManager&, float arg) { return x328_24_inPosition; }
 | |
| 
 | |
| bool CPatterned::HasPatrolPath(CStateManager& mgr, float arg) {
 | |
|   return GetWaypointForState(mgr, EScriptObjectState::Patrol, EScriptObjectMessage::Follow) != kInvalidUniqueId;
 | |
| }
 | |
| 
 | |
| bool CPatterned::Attacked(CStateManager&, float arg) { return x400_24_hitByPlayerProjectile; }
 | |
| 
 | |
| bool CPatterned::PatternShagged(CStateManager&, float arg) { return x400_30_patternShagged; }
 | |
| 
 | |
| bool CPatterned::PatternOver(CStateManager&, float arg) { return x38c_patterns.size() <= x39c_curPattern; }
 | |
| 
 | |
| bool CPatterned::HasRetreatPattern(CStateManager& mgr, float arg) {
 | |
|   return GetWaypointForState(mgr, EScriptObjectState::Retreat, EScriptObjectMessage::Follow) != kInvalidUniqueId;
 | |
| }
 | |
| 
 | |
| bool CPatterned::HasAttackPattern(CStateManager& mgr, float arg) {
 | |
|   return GetWaypointForState(mgr, EScriptObjectState::Attack, EScriptObjectMessage::Follow) != kInvalidUniqueId;
 | |
| }
 | |
| 
 | |
| bool CPatterned::NoPathNodes(CStateManager&, float arg) {
 | |
|   if (CPathFindSearch* search = GetSearchPath()) {
 | |
|     return search->OnPath(GetTranslation()) != CPathFindSearch::EResult::Success;
 | |
|   }
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| constexpr float skActorApproachDistance = 3.f;
 | |
| 
 | |
| bool CPatterned::PathShagged(CStateManager&, float arg) {
 | |
|   if (CPathFindSearch* search = GetSearchPath()) {
 | |
|     if (search->IsShagged()) {
 | |
|       return true;
 | |
|     }
 | |
|     if (search->GetCurrentWaypoint() > 0 && x401_24_pathOverCount == 0) {
 | |
|       zeus::CVector3f origPoint = GetTranslation() + 0.3f * zeus::skUp;
 | |
|       zeus::CVector3f point = origPoint;
 | |
|       search->GetSplinePoint(point, GetTranslation());
 | |
|       return (point - origPoint).magSquared() > 4.f * skActorApproachDistance * skActorApproachDistance;
 | |
|     }
 | |
|   }
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| bool CPatterned::PathFound(CStateManager&, float arg) {
 | |
|   if (CPathFindSearch* search = GetSearchPath()) {
 | |
|     if (!search->IsShagged()) {
 | |
|       return true;
 | |
|     }
 | |
|   }
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| bool CPatterned::PathOver(CStateManager&, float arg) {
 | |
|   if (CPathFindSearch* search = GetSearchPath()) {
 | |
|     if (x328_25_verticalMovement || x328_27_onGround) {
 | |
|       if (!search->IsShagged() && search->IsOver()) {
 | |
|         return true;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| bool CPatterned::Landed(CStateManager&, float arg) {
 | |
|   bool ret = x328_27_onGround && !x328_28_prevOnGround;
 | |
|   x328_28_prevOnGround = x328_27_onGround;
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| bool CPatterned::PlayerSpot(CStateManager& mgr, float arg) {
 | |
|   if (mgr.GetPlayer().GetMorphballTransitionState() == CPlayer::EPlayerMorphBallState::Unmorphed) {
 | |
|     zeus::CVector3f aimPosition = mgr.GetPlayer().GetAimPosition(mgr, 0.f);
 | |
|     zeus::CVector3f center = GetBoundingBox().center();
 | |
|     zeus::CVector3f aimToCenter = center - aimPosition;
 | |
|     float aimToCenterMag = aimToCenter.magnitude();
 | |
|     zeus::CVector3f aimToCenterNorm = aimToCenter * (1.f / aimToCenterMag);
 | |
|     zeus::CVector3f screenSpace = mgr.GetCameraManager()->GetFirstPersonCamera()->ConvertToScreenSpace(center);
 | |
|     if (screenSpace.z() > 0.f && screenSpace.x() * screenSpace.x() < 1.f && screenSpace.y() * screenSpace.y() < 1.f) {
 | |
|       CRayCastResult res = mgr.RayStaticIntersection(aimPosition, aimToCenterNorm, aimToCenterMag,
 | |
|                                                      CMaterialFilter::MakeInclude(EMaterialTypes::Solid));
 | |
|       return res.IsInvalid();
 | |
|     }
 | |
|   }
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| bool CPatterned::SpotPlayer(CStateManager& mgr, float arg) {
 | |
|   zeus::CVector3f gunToPlayer = mgr.GetPlayer().GetAimPosition(mgr, 0.f) - GetGunEyePos();
 | |
|   float lookDot = gunToPlayer.dot(x34_transform.basis[1]);
 | |
|   if (lookDot > 0.f) {
 | |
|     return lookDot * lookDot > gunToPlayer.magSquared() * x3c4_detectionAngle;
 | |
|   }
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| bool CPatterned::Leash(CStateManager&, float arg) {
 | |
|   bool ret = x3d4_curPlayerLeashTime > x3d0_playerLeashTime;
 | |
|   if (ret) {
 | |
|     float posToLeashMagSq = (x3a0_latestLeashPosition - GetTranslation()).magSquared();
 | |
|     return posToLeashMagSq > x3c8_leashRadius * x3c8_leashRadius;
 | |
|   }
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| bool CPatterned::InDetectionRange(CStateManager& mgr, float arg) {
 | |
|   zeus::CVector3f delta = GetTranslation() - mgr.GetPlayer().GetTranslation();
 | |
|   const float maxRange = x3bc_detectionRange * x3bc_detectionRange;
 | |
|   const float dist = delta.magSquared();
 | |
|   if (dist < maxRange) {
 | |
|     if (x3c0_detectionHeightRange > 0.f) {
 | |
|       return delta.z() * delta.z() < x3c0_detectionHeightRange * x3c0_detectionHeightRange;
 | |
|     }
 | |
|     return true;
 | |
|   }
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| bool CPatterned::InMaxRange(CStateManager& mgr, float arg) {
 | |
|   return (mgr.GetPlayer().GetTranslation() - GetTranslation()).magSquared() < x300_maxAttackRange * x300_maxAttackRange;
 | |
| }
 | |
| 
 | |
| bool CPatterned::TooClose(CStateManager& mgr, float arg) {
 | |
|   return (mgr.GetPlayer().GetTranslation() - GetTranslation()).magSquared() < x2fc_minAttackRange * x2fc_minAttackRange;
 | |
| }
 | |
| 
 | |
| bool CPatterned::InRange(CStateManager& mgr, float arg) {
 | |
|   float range = 0.5f * (x2fc_minAttackRange + x300_maxAttackRange);
 | |
|   return (mgr.GetPlayer().GetTranslation() - GetTranslation()).magSquared() < range * range;
 | |
| }
 | |
| 
 | |
| bool CPatterned::OffLine(CStateManager&, float arg) {
 | |
|   zeus::CVector3f curLine = GetTranslation() - x2ec_reflectedDestPos;
 | |
|   zeus::CVector3f pathLine = x2e0_destPos - x2ec_reflectedDestPos;
 | |
|   float distSq = 0.f;
 | |
|   if (curLine.dot(pathLine) <= 0.f) {
 | |
|     distSq = curLine.magSquared();
 | |
|   } else {
 | |
|     pathLine.normalize();
 | |
|     distSq = (curLine - pathLine.dot(curLine) * pathLine).magSquared();
 | |
|     zeus::CVector3f delta = GetTranslation() - x2e0_destPos;
 | |
|     if (pathLine.dot(delta) > 0.f) {
 | |
|       distSq = delta.magSquared();
 | |
|     }
 | |
|   }
 | |
|   return distSq > arg * arg;
 | |
| }
 | |
| 
 | |
| void CPatterned::PathFind(CStateManager& mgr, EStateMsg msg, float dt) {
 | |
|   if (CPathFindSearch* search = GetSearchPath()) {
 | |
|     switch (msg) {
 | |
|     case EStateMsg::Activate: {
 | |
|       if (search->Search(GetTranslation(), x2e0_destPos) == CPathFindSearch::EResult::Success) {
 | |
|         x2ec_reflectedDestPos = GetTranslation();
 | |
|         zeus::CVector3f destPos;
 | |
|         if (search->GetCurrentWaypoint() + 1 < search->GetWaypoints().size()) {
 | |
|           destPos = search->GetWaypoints()[search->GetCurrentWaypoint() + 1];
 | |
|         } else {
 | |
|           destPos = search->GetWaypoints()[search->GetCurrentWaypoint()];
 | |
|         }
 | |
|         SetDestPos(destPos);
 | |
|         x328_24_inPosition = false;
 | |
|         ApproachDest(mgr);
 | |
|       }
 | |
|       break;
 | |
|     }
 | |
|     case EStateMsg::Update: {
 | |
|       if (search->GetCurrentWaypoint() < search->GetWaypoints().size() - 1) {
 | |
|         if (x328_25_verticalMovement || x328_27_onGround) {
 | |
|           x401_24_pathOverCount += 1;
 | |
|         }
 | |
|         zeus::CVector3f biasedPos = GetTranslation() + 0.3f * zeus::skUp;
 | |
|         x2ec_reflectedDestPos = biasedPos - (x2e0_destPos - biasedPos);
 | |
|         ApproachDest(mgr);
 | |
|         zeus::CVector3f biasedForward = x34_transform.basis[1] * x64_modelData->GetScale().y() + biasedPos;
 | |
|         search->GetSplinePointWithLookahead(biasedForward, biasedPos, 3.f * x64_modelData->GetScale().y());
 | |
|         SetDestPos(biasedForward);
 | |
|         if (search->SegmentOver(biasedPos)) {
 | |
|           search->SetCurrentWaypoint(search->GetCurrentWaypoint() + 1);
 | |
|         }
 | |
|       }
 | |
|       break;
 | |
|     }
 | |
|     default:
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CPatterned::Dead(CStateManager& mgr, EStateMsg msg, float dt) {
 | |
|   switch (msg) {
 | |
|   case EStateMsg::Activate:
 | |
|     x31c_faceVec = zeus::skZero3f;
 | |
|     break;
 | |
|   case EStateMsg::Update:
 | |
|     x450_bodyController->GetCommandMgr().DeliverCmd(CBodyStateCmd(EBodyStateCmd::Die));
 | |
|     if (!x400_27_fadeToDeath) {
 | |
|       if (x450_bodyController->GetBodyStateInfo().GetCurrentState()->IsDead()) {
 | |
|         x400_27_fadeToDeath = true;
 | |
|         x3e8_alphaDelta = -0.333333f;
 | |
|         RemoveMaterial(EMaterialTypes::Character, EMaterialTypes::Solid, EMaterialTypes::Target, EMaterialTypes::Orbit,
 | |
|                        mgr);
 | |
|         AddMaterial(EMaterialTypes::ProjectilePassthrough, mgr);
 | |
|       }
 | |
|     }
 | |
|     break;
 | |
|   default:
 | |
|     break;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CPatterned::TargetPlayer(CStateManager& mgr, EStateMsg msg, float dt) {
 | |
|   if (msg == EStateMsg::Activate) {
 | |
|     x2dc_destObj = mgr.GetPlayer().GetUniqueId();
 | |
|     SetDestPos(mgr.GetPlayer().GetTranslation());
 | |
|     x2ec_reflectedDestPos = GetTranslation();
 | |
|     x328_24_inPosition = false;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CPatterned::TargetPatrol(CStateManager& mgr, EStateMsg msg, float dt) {
 | |
|   if (msg == EStateMsg::Activate) {
 | |
|     x2dc_destObj = GetWaypointForState(mgr, EScriptObjectState::Patrol, EScriptObjectMessage::Follow);
 | |
|     if (TCastToConstPtr<CActor> act = mgr.GetObjectById(x2dc_destObj)) {
 | |
|       SetDestPos(act->GetTranslation());
 | |
|     }
 | |
|     x2ec_reflectedDestPos = GetTranslation();
 | |
|     x328_24_inPosition = false;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CPatterned::FollowPattern(CStateManager& mgr, EStateMsg msg, float dt) {
 | |
|   switch (msg) {
 | |
|   case EStateMsg::Activate:
 | |
|     SetupPattern(mgr);
 | |
|     if (x328_29_noPatternShagging || !IsPatternObstructed(mgr, GetTranslation(), x2e0_destPos)) {
 | |
|       ApproachDest(mgr);
 | |
|     } else {
 | |
|       x39c_curPattern = x38c_patterns.size();
 | |
|       x400_30_patternShagged = true;
 | |
|     }
 | |
|     break;
 | |
|   case EStateMsg::Update:
 | |
|     if (x328_24_inPosition) {
 | |
|       x39c_curPattern += 1;
 | |
|       UpdatePatternDestPos(mgr);
 | |
|       if (!x328_29_noPatternShagging && IsPatternObstructed(mgr, GetTranslation(), x2e0_destPos)) {
 | |
|         x39c_curPattern = x38c_patterns.size();
 | |
|         x400_30_patternShagged = true;
 | |
|       } else if (x39c_curPattern < x38c_patterns.size()) {
 | |
|         x2ec_reflectedDestPos = GetTranslation();
 | |
|         x328_24_inPosition = false;
 | |
|         x3b0_moveSpeed = x38c_patterns[x39c_curPattern].GetSpeed();
 | |
|         x380_behaviour = EBehaviour(x38c_patterns[x39c_curPattern].GetBehaviour());
 | |
|         x30c_behaviourOrient = EBehaviourOrient(x38c_patterns[x39c_curPattern].GetBehaviourOrient());
 | |
|         x384_behaviourModifiers = EBehaviourModifiers(x38c_patterns[x39c_curPattern].GetBehaviourModifiers());
 | |
|       }
 | |
|     } else {
 | |
|       UpdatePatternDestPos(mgr);
 | |
|     }
 | |
|     ApproachDest(mgr);
 | |
|     break;
 | |
|   case EStateMsg::Deactivate:
 | |
|     x38c_patterns.clear();
 | |
|     x400_30_patternShagged = false;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CPatterned::Patrol(CStateManager& mgr, EStateMsg msg, float dt) {
 | |
|   switch (msg) {
 | |
|   case EStateMsg::Activate:
 | |
|     if (x3ac_lastPatrolDest == kInvalidUniqueId) {
 | |
|       x2dc_destObj = GetWaypointForState(mgr, EScriptObjectState::Patrol, EScriptObjectMessage::Follow);
 | |
|       x30c_behaviourOrient = EBehaviourOrient::MoveDir;
 | |
|       x3b0_moveSpeed = 1.f;
 | |
|       if (x2dc_destObj != kInvalidUniqueId) {
 | |
|         if (TCastToConstPtr<CScriptWaypoint> wp = mgr.GetObjectById(x2dc_destObj)) {
 | |
|           x30c_behaviourOrient = EBehaviourOrient(wp->GetBehaviourOrient());
 | |
|           x3b0_moveSpeed = wp->GetSpeed();
 | |
|         }
 | |
|       }
 | |
|     } else {
 | |
|       x2dc_destObj = x3ac_lastPatrolDest;
 | |
|     }
 | |
|     x2ec_reflectedDestPos = GetTranslation();
 | |
|     x328_24_inPosition = false;
 | |
|     x2d8_patrolState = EPatrolState::Patrol;
 | |
|     x2f8_waypointPauseRemTime = 0.f;
 | |
|     break;
 | |
|   case EStateMsg::Update:
 | |
|     switch (x2d8_patrolState) {
 | |
|     case EPatrolState::Patrol:
 | |
|       if (x328_24_inPosition && x2dc_destObj != kInvalidUniqueId) {
 | |
|         if (TCastToConstPtr<CScriptWaypoint> wp = mgr.GetObjectById(x2dc_destObj)) {
 | |
|           if (wp->GetPause() > 0.f) {
 | |
|             x2f8_waypointPauseRemTime = wp->GetPause();
 | |
|             x2d8_patrolState = EPatrolState::Pause;
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|       if (x2dc_destObj == kInvalidUniqueId) {
 | |
|         x2d8_patrolState = EPatrolState::Done;
 | |
|       }
 | |
|       UpdateDest(mgr);
 | |
|       ApproachDest(mgr);
 | |
|       break;
 | |
|     case EPatrolState::Pause:
 | |
|       if (x2f8_waypointPauseRemTime <= 0.f) {
 | |
|         x2d8_patrolState = EPatrolState::Patrol;
 | |
|       }
 | |
|       break;
 | |
|     case EPatrolState::Done:
 | |
|       if (x2dc_destObj != kInvalidUniqueId) {
 | |
|         x2d8_patrolState = EPatrolState::Patrol;
 | |
|       }
 | |
|       break;
 | |
|     default:
 | |
|       break;
 | |
|     }
 | |
|     break;
 | |
|   case EStateMsg::Deactivate:
 | |
|     x3ac_lastPatrolDest = x2dc_destObj;
 | |
|     x2d8_patrolState = EPatrolState::Invalid;
 | |
|     break;
 | |
|   default:
 | |
|     break;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CPatterned::TryCommand(CStateManager& mgr, pas::EAnimationState state, CPatternedTryFunc func, int arg) {
 | |
|   if (state == x450_bodyController->GetCurrentStateId()) {
 | |
|     x32c_animState = EAnimState::Repeat;
 | |
|   } else if (x32c_animState == EAnimState::Ready) {
 | |
|     (this->*func)(mgr, arg);
 | |
|   } else {
 | |
|     x32c_animState = EAnimState::Over;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CPatterned::TryLoopReaction(CStateManager& mgr, int arg) {
 | |
|   x450_bodyController->GetCommandMgr().DeliverCmd(CBCLoopReactionCmd(pas::EReactionType(arg)));
 | |
| }
 | |
| 
 | |
| void CPatterned::TryProjectileAttack(CStateManager&, int arg) {
 | |
|   x450_bodyController->GetCommandMgr().DeliverCmd(CBCProjectileAttackCmd(pas::ESeverity(arg), x2e0_destPos, false));
 | |
| }
 | |
| 
 | |
| void CPatterned::TryMeleeAttack_TargetPos(CStateManager& mgr, int arg) {
 | |
|   x450_bodyController->GetCommandMgr().DeliverCmd(CBCMeleeAttackCmd(pas::ESeverity(arg), x2e0_destPos));
 | |
| }
 | |
| 
 | |
| void CPatterned::TryMeleeAttack(CStateManager& mgr, int arg) {
 | |
|   x450_bodyController->GetCommandMgr().DeliverCmd(CBCMeleeAttackCmd(pas::ESeverity(arg)));
 | |
| }
 | |
| 
 | |
| void CPatterned::TryGenerateNoXf(CStateManager& mgr, int arg) {
 | |
|   x450_bodyController->GetCommandMgr().DeliverCmd(CBCGenerateCmd(pas::EGenerateType(arg), x2e0_destPos, false));
 | |
| }
 | |
| 
 | |
| void CPatterned::TryGenerate(CStateManager& mgr, int arg) {
 | |
|   x450_bodyController->GetCommandMgr().DeliverCmd(CBCGenerateCmd(pas::EGenerateType::Zero, x2e0_destPos, true));
 | |
| }
 | |
| 
 | |
| void CPatterned::TryJump(CStateManager& mgr, int arg) {
 | |
|   x450_bodyController->GetCommandMgr().DeliverCmd(CBCJumpCmd(x2e0_destPos, pas::EJumpType(arg)));
 | |
| }
 | |
| 
 | |
| void CPatterned::TryTurn(CStateManager& mgr, int arg) {
 | |
|   x450_bodyController->GetCommandMgr().DeliverCmd(
 | |
|       CBCLocomotionCmd(zeus::skZero3f, (x2e0_destPos - GetTranslation()).normalized(), 1.f));
 | |
| }
 | |
| 
 | |
| void CPatterned::TryGetUp(CStateManager& mgr, int arg) {
 | |
|   x450_bodyController->GetCommandMgr().DeliverCmd(CBCGetupCmd(pas::EGetupType(arg)));
 | |
| }
 | |
| 
 | |
| void CPatterned::TryTaunt(CStateManager& mgr, int arg) {
 | |
|   x450_bodyController->GetCommandMgr().DeliverCmd(CBCTauntCmd(pas::ETauntType(arg)));
 | |
| }
 | |
| 
 | |
| void CPatterned::TryJumpInLoop(CStateManager& mgr, int arg) {
 | |
|   x450_bodyController->GetCommandMgr().DeliverCmd(CBCJumpCmd(x2e0_destPos, pas::EJumpType(arg), true));
 | |
| }
 | |
| 
 | |
| void CPatterned::TryDodge(CStateManager& mgr, int arg) {
 | |
|   x450_bodyController->GetCommandMgr().DeliverCmd(CBCStepCmd(pas::EStepDirection(arg), pas::EStepType::Dodge));
 | |
| }
 | |
| 
 | |
| void CPatterned::TryRollingDodge(CStateManager& mgr, int arg) {
 | |
|   x450_bodyController->GetCommandMgr().DeliverCmd(CBCStepCmd(pas::EStepDirection(arg), pas::EStepType::RollDodge));
 | |
| }
 | |
| 
 | |
| void CPatterned::TryBreakDodge(CStateManager& mgr, int arg) {
 | |
|   x450_bodyController->GetCommandMgr().DeliverCmd(CBCStepCmd(pas::EStepDirection(arg), pas::EStepType::BreakDodge));
 | |
| }
 | |
| 
 | |
| void CPatterned::TryCover(CStateManager& mgr, int arg) {
 | |
|   if (CScriptCoverPoint* cp = GetCoverPoint(mgr, x2dc_destObj)) {
 | |
|     x450_bodyController->GetCommandMgr().DeliverCmd(
 | |
|         CBCCoverCmd(pas::ECoverDirection(arg), cp->GetTranslation(), -cp->GetTransform().basis[1]));
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CPatterned::TryWallHang(CStateManager& mgr, int arg) {
 | |
|   x450_bodyController->GetCommandMgr().DeliverCmd(CBCWallHangCmd(x2dc_destObj));
 | |
| }
 | |
| 
 | |
| void CPatterned::TryKnockBack(CStateManager& mgr, int arg) {
 | |
|   x450_bodyController->GetCommandMgr().DeliverCmd(CBCKnockBackCmd(GetTranslation(), pas::ESeverity(arg)));
 | |
| }
 | |
| 
 | |
| void CPatterned::TryKnockBack_Front(CStateManager& mgr, int arg) {
 | |
|   x450_bodyController->GetCommandMgr().DeliverCmd(CBCKnockBackCmd(GetTransform().frontVector(), pas::ESeverity(arg)));
 | |
| }
 | |
| 
 | |
| void CPatterned::TryGenerateDeactivate(metaforce::CStateManager& mgr, int arg) {
 | |
|   x450_bodyController->GetCommandMgr().DeliverCmd(CBCGenerateCmd(pas::EGenerateType(arg), zeus::skZero3f));
 | |
| }
 | |
| 
 | |
| void CPatterned::TryStep(CStateManager& mgr, int arg) {
 | |
|   x450_bodyController->GetCommandMgr().DeliverCmd(CBCStepCmd(pas::EStepDirection(arg), pas::EStepType::Normal));
 | |
| }
 | |
| 
 | |
| void CPatterned::TryScripted(CStateManager& mgr, int arg) {
 | |
|   x450_bodyController->GetCommandMgr().DeliverCmd(CBCScriptedCmd(arg, false, false, 0.f));
 | |
| }
 | |
| 
 | |
| void CPatterned::BuildBodyController(EBodyType bodyType) {
 | |
|   if (x450_bodyController) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   x450_bodyController = std::make_unique<CBodyController>(*this, x3b8_turnSpeed, bodyType);
 | |
|   auto anim = x450_bodyController->GetPASDatabase().FindBestAnimation(
 | |
|       CPASAnimParmData(pas::EAnimationState::AdditiveReaction, CPASAnimParm::FromEnum(0)), -1);
 | |
|   x460_knockBackController.x81_26_enableShock = anim.first > 0.f;
 | |
| }
 | |
| 
 | |
| void CPatterned::GenerateDeathExplosion(CStateManager& mgr) {
 | |
|   if (auto particle = GetDeathExplosionParticle()) {
 | |
|     zeus::CTransform xf(GetTransform());
 | |
|     xf.origin = GetTransform() * (x64_modelData->GetScale() * x514_deathExplosionOffset);
 | |
|     auto* explo = new CExplosion(*particle, mgr.AllocateUniqueId(), true,
 | |
|                                  CEntityInfo(GetAreaIdAlways(), CEntity::NullConnectionList), "", xf, 1, zeus::skOne3f,
 | |
|                                  zeus::skWhite);
 | |
|     mgr.AddObject(explo);
 | |
|   }
 | |
|   if (x530_deathExplosionElectric) {
 | |
|     zeus::CTransform xf(GetTransform());
 | |
|     xf.origin = GetTransform() * (x64_modelData->GetScale() * x514_deathExplosionOffset);
 | |
|     auto* explo = new CExplosion(*x530_deathExplosionElectric, mgr.AllocateUniqueId(), true,
 | |
|                                  CEntityInfo(GetAreaIdAlways(), CEntity::NullConnectionList), "", xf, 1, zeus::skOne3f,
 | |
|                                  zeus::skWhite);
 | |
|     mgr.AddObject(explo);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CPatterned::MassiveDeath(CStateManager& mgr) {
 | |
|   CSfxManager::AddEmitter(x454_deathSfx, GetTranslation(), zeus::skZero3f, true, false, 0x7f, kInvalidAreaId);
 | |
|   if (!x401_28_burning) {
 | |
|     SendScriptMsgs(EScriptObjectState::MassiveDeath, mgr, EScriptObjectMessage::None);
 | |
|     GenerateDeathExplosion(mgr);
 | |
|   }
 | |
|   DeathDelete(mgr);
 | |
|   x400_28_pendingMassiveDeath = x400_29_pendingMassiveFrozenDeath = false;
 | |
| }
 | |
| 
 | |
| void CPatterned::GenerateIceDeathExplosion(CStateManager& mgr) {
 | |
|   if (x54c_iceDeathExplosionParticle) {
 | |
|     zeus::CTransform xf(GetTransform());
 | |
|     xf.origin = GetTransform() * (x64_modelData->GetScale() * x540_iceDeathExplosionOffset);
 | |
|     auto* explo = new CExplosion(*x54c_iceDeathExplosionParticle, mgr.AllocateUniqueId(), true,
 | |
|                                  CEntityInfo(GetAreaIdAlways(), CEntity::NullConnectionList), "", xf, 1, zeus::skOne3f,
 | |
|                                  zeus::skWhite);
 | |
|     mgr.AddObject(explo);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CPatterned::MassiveFrozenDeath(CStateManager& mgr) {
 | |
|   if (x458_iceShatterSfx == 0xffff) {
 | |
|     x458_iceShatterSfx = x454_deathSfx;
 | |
|   }
 | |
|   CSfxManager::AddEmitter(x458_iceShatterSfx, GetTranslation(), zeus::skZero3f, true, false, 0x7f, kInvalidAreaId);
 | |
|   SendScriptMsgs(EScriptObjectState::MassiveFrozenDeath, mgr, EScriptObjectMessage::None);
 | |
|   GenerateIceDeathExplosion(mgr);
 | |
|   float toPlayerDist = (mgr.GetPlayer().GetTranslation() - GetTranslation()).magnitude();
 | |
|   if (toPlayerDist < 40.f) {
 | |
|     mgr.GetCameraManager()->AddCameraShaker(
 | |
|         CCameraShakeData::BuildPatternedExplodeShakeData(GetTranslation(), 0.25f, 0.3f, 40.f), true);
 | |
|   }
 | |
|   DeathDelete(mgr);
 | |
|   x400_28_pendingMassiveDeath = x400_29_pendingMassiveFrozenDeath = false;
 | |
| }
 | |
| 
 | |
| void CPatterned::Burn(float duration, float damage) {
 | |
|   switch (GetDamageVulnerability()->GetVulnerability(CWeaponMode(EWeaponType::Plasma), false)) {
 | |
|   case EVulnerability::Weak:
 | |
|     x450_bodyController->SetOnFire(1.5f * duration);
 | |
|     x3ec_pendingFireDamage = 1.5f * damage;
 | |
|     break;
 | |
|   case EVulnerability::Normal:
 | |
|     x450_bodyController->SetOnFire(duration);
 | |
|     x3ec_pendingFireDamage = damage;
 | |
|     break;
 | |
|   default:
 | |
|     break;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CPatterned::Shock(CStateManager& mgr, float duration, float damage) {
 | |
|   switch (GetDamageVulnerability()->GetVulnerability(CWeaponMode(EWeaponType::Wave), false)) {
 | |
|   case EVulnerability::Weak:
 | |
|     x450_bodyController->SetElectrocuting(1.5f * duration);
 | |
|     x3f0_pendingShockDamage = 1.5f * damage;
 | |
|     break;
 | |
|   case EVulnerability::Normal:
 | |
|     x450_bodyController->SetElectrocuting(duration);
 | |
|     x3f0_pendingShockDamage = damage;
 | |
|     break;
 | |
|   default:
 | |
|     break;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CPatterned::Freeze(CStateManager& mgr, const zeus::CVector3f& pos, const zeus::CUnitVector3f& dir,
 | |
|                         float frozenDur) {
 | |
|   if (x402_25_lostMassiveFrozenHP) {
 | |
|     x402_26_dieIf80PercFrozen = true;
 | |
|   }
 | |
|   bool playSfx = false;
 | |
|   if (x450_bodyController->IsFrozen()) {
 | |
|     x450_bodyController->Freeze(x460_knockBackController.GetActiveParms().xc_intoFreezeDur, frozenDur,
 | |
|                                 x4f8_outofFreezeDur);
 | |
|     mgr.GetActorModelParticles()->EnsureElectricLoaded(*this);
 | |
|     playSfx = true;
 | |
|   } else if (!x450_bodyController->IsElectrocuting() && !x450_bodyController->IsOnFire()) {
 | |
|     x450_bodyController->Freeze(x4f4_intoFreezeDur, frozenDur, x4f8_outofFreezeDur);
 | |
|     if (x510_vertexMorph) {
 | |
|       x510_vertexMorph->Reset(dir, pos, x4f4_intoFreezeDur);
 | |
|     }
 | |
|     playSfx = true;
 | |
|   }
 | |
| 
 | |
|   if (playSfx) {
 | |
|     u16 sfx = (x460_knockBackController.GetVariant() != EKnockBackVariant::Small &&
 | |
|                CPatterned::CastTo<MP1::CMetroid>(mgr.GetObjectById(GetUniqueId())) != nullptr)
 | |
|                   ? SFXsfx0701
 | |
|                   : SFXsfx0708;
 | |
|     CSfxManager::AddEmitter(sfx, GetTranslation(), zeus::skZero3f, true, false, 0x7f, kInvalidAreaId);
 | |
|   }
 | |
| }
 | |
| 
 | |
| zeus::CVector3f CPatterned::GetGunEyePos() const {
 | |
|   zeus::CVector3f origin = GetTranslation();
 | |
|   zeus::CAABox baseBox = GetBaseBoundingBox();
 | |
|   origin.z() = 0.6f * (baseBox.max.z() - baseBox.min.z()) + origin.z();
 | |
|   return origin;
 | |
| }
 | |
| 
 | |
| void CPatterned::SetupPlayerCollision(bool v) {
 | |
|   CMaterialList include = GetMaterialFilter().GetIncludeList();
 | |
|   CMaterialList exclude = GetMaterialFilter().GetExcludeList();
 | |
|   CMaterialList* modList = (v ? &exclude : &include);
 | |
|   modList->Add(EMaterialTypes::Player);
 | |
|   SetMaterialFilter(CMaterialFilter::MakeIncludeExclude(include, exclude));
 | |
| }
 | |
| 
 | |
| CGameProjectile* CPatterned::LaunchProjectile(const zeus::CTransform& gunXf, CStateManager& mgr, int maxAllowed,
 | |
|                                               EProjectileAttrib attrib, bool playerHoming,
 | |
|                                               const std::optional<TLockedToken<CGenDescription>>& visorParticle,
 | |
|                                               u16 visorSfx, bool sendCollideMsg, const zeus::CVector3f& scale) {
 | |
|   CProjectileInfo* pInfo = GetProjectileInfo();
 | |
|   if (pInfo && pInfo->Token().IsLoaded()) {
 | |
|     if (mgr.CanCreateProjectile(GetUniqueId(), EWeaponType::AI, maxAllowed)) {
 | |
|       TUniqueId homingId = playerHoming ? mgr.GetPlayer().GetUniqueId() : kInvalidUniqueId;
 | |
|       auto* newProjectile =
 | |
|           new CEnergyProjectile(true, pInfo->Token(), EWeaponType::AI, gunXf, EMaterialTypes::Character,
 | |
|                                 pInfo->GetDamage(), mgr.AllocateUniqueId(), GetAreaIdAlways(), GetUniqueId(), homingId,
 | |
|                                 attrib, false, scale, visorParticle, visorSfx, sendCollideMsg);
 | |
|       mgr.AddObject(newProjectile);
 | |
|       return newProjectile;
 | |
|     }
 | |
|   }
 | |
|   return nullptr;
 | |
| }
 | |
| 
 | |
| void CPatterned::DoUserAnimEvent(CStateManager& mgr, const CInt32POINode& node, EUserEventType type, float dt) {
 | |
|   switch (type) {
 | |
|   case EUserEventType::Projectile: {
 | |
|     zeus::CTransform lctrXf = GetLctrTransform(node.GetLocatorName());
 | |
|     zeus::CVector3f aimPos = mgr.GetPlayer().GetAimPosition(mgr, 0.f);
 | |
|     if ((aimPos - lctrXf.origin).normalized().dot(lctrXf.basis[1]) > 0.f) {
 | |
|       zeus::CTransform gunXf = zeus::lookAt(lctrXf.origin, aimPos);
 | |
|       LaunchProjectile(gunXf, mgr, 1, EProjectileAttrib::None, false, {}, 0xffff, false, zeus::skOne3f);
 | |
|     } else {
 | |
|       LaunchProjectile(lctrXf, mgr, 1, EProjectileAttrib::None, false, {}, 0xffff, false, zeus::skOne3f);
 | |
|     }
 | |
|     break;
 | |
|   }
 | |
|   case EUserEventType::DamageOn: {
 | |
|     zeus::CTransform lctrXf = GetLctrTransform(node.GetLocatorName());
 | |
|     zeus::CVector3f xfOrigin = x34_transform * (x64_modelData->GetScale() * lctrXf.origin);
 | |
|     zeus::CVector3f margin = zeus::CVector3f(1.f, 1.f, 0.5f) * x64_modelData->GetScale();
 | |
|     if (zeus::CAABox(xfOrigin - margin, xfOrigin + margin).intersects(mgr.GetPlayer().GetBoundingBox())) {
 | |
|       mgr.ApplyDamage(GetUniqueId(), mgr.GetPlayer().GetUniqueId(), GetUniqueId(), GetContactDamage(),
 | |
|                       CMaterialFilter::MakeIncludeExclude({EMaterialTypes::Solid}, {}), zeus::skZero3f);
 | |
|     }
 | |
|     break;
 | |
|   }
 | |
|   case EUserEventType::Delete: {
 | |
|     if (!x400_25_alive) {
 | |
|       if (!x400_27_fadeToDeath) {
 | |
|         x3e8_alphaDelta = -0.333333f;
 | |
|         x400_27_fadeToDeath = true;
 | |
|       }
 | |
|       RemoveMaterial(EMaterialTypes::Character, EMaterialTypes::Solid, EMaterialTypes::Target, EMaterialTypes::Orbit,
 | |
|                      mgr);
 | |
|       AddMaterial(EMaterialTypes::ProjectilePassthrough, mgr);
 | |
|     } else {
 | |
|       DeathDelete(mgr);
 | |
|     }
 | |
|     break;
 | |
|   }
 | |
|   case EUserEventType::BreakLockOn: {
 | |
|     RemoveMaterial(EMaterialTypes::Character, EMaterialTypes::Target, EMaterialTypes::Orbit, mgr);
 | |
|     break;
 | |
|   }
 | |
|   case EUserEventType::BecomeShootThrough: {
 | |
|     AddMaterial(EMaterialTypes::ProjectilePassthrough, mgr);
 | |
|     break;
 | |
|   }
 | |
|   case EUserEventType::RemoveCollision: {
 | |
|     RemoveMaterial(EMaterialTypes::Solid, mgr);
 | |
|     break;
 | |
|   }
 | |
|   default:
 | |
|     break;
 | |
|   }
 | |
|   CActor::DoUserAnimEvent(mgr, node, type, dt);
 | |
| }
 | |
| 
 | |
| void CPatterned::UpdateAlphaDelta(float dt, CStateManager& mgr) {
 | |
|   if (x3e8_alphaDelta == 0.f) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   float alpha = dt * x3e8_alphaDelta + x42c_color.a();
 | |
|   if (alpha > 1.f) {
 | |
|     alpha = 1.f;
 | |
|     x3e8_alphaDelta = 0.f;
 | |
|   } else if (alpha < 0.f) {
 | |
|     alpha = 0.f;
 | |
|     x3e8_alphaDelta = 0.f;
 | |
|     if (x400_27_fadeToDeath) {
 | |
|       DeathDelete(mgr);
 | |
|     }
 | |
|   }
 | |
|   x94_simpleShadow->SetUserAlpha(alpha);
 | |
|   SetModelAlpha(alpha);
 | |
|   x64_modelData->GetAnimationData()->GetParticleDB().SetModulationColorAllActiveEffects(zeus::CColor(1.f, alpha));
 | |
| }
 | |
| 
 | |
| float CPatterned::CalcDyingThinkRate() const {
 | |
|   float f0 = (x401_28_burning ? (x3f4_burnThinkRateTimer / 1.5f) : 1.f);
 | |
|   return zeus::max(0.1f, f0);
 | |
| }
 | |
| 
 | |
| void CPatterned::UpdateDamageColor(float dt) {
 | |
|   if (x428_damageCooldownTimer > 0.f) {
 | |
|     x428_damageCooldownTimer = std::max(0.f, x428_damageCooldownTimer - dt);
 | |
|     float alpha = x42c_color.a();
 | |
|     x42c_color = zeus::CColor::lerp(zeus::skBlack, x430_damageColor, std::min(x428_damageCooldownTimer / 0.33f, 1.f));
 | |
|     x42c_color.a() = alpha;
 | |
|     if (!x450_bodyController->IsFrozen()) {
 | |
|       xd0_damageMag = x50c_baseDamageMag + x428_damageCooldownTimer;
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| CScriptCoverPoint* CPatterned::GetCoverPoint(CStateManager& mgr, TUniqueId id) const {
 | |
|   if (id != kInvalidUniqueId) {
 | |
|     if (TCastToPtr<CScriptCoverPoint> cp = mgr.ObjectById(id)) {
 | |
|       return cp.GetPtr();
 | |
|     }
 | |
|   }
 | |
|   return nullptr;
 | |
| }
 | |
| 
 | |
| void CPatterned::SetCoverPoint(CScriptCoverPoint* cp, TUniqueId& id) {
 | |
|   cp->SetInUse(true);
 | |
|   id = cp->GetUniqueId();
 | |
| }
 | |
| 
 | |
| void CPatterned::ReleaseCoverPoint(CStateManager& mgr, TUniqueId& id) const {
 | |
|   if (CScriptCoverPoint* cp = GetCoverPoint(mgr, id)) {
 | |
|     cp->SetInUse(false);
 | |
|     id = kInvalidUniqueId;
 | |
|   }
 | |
| }
 | |
| 
 | |
| TUniqueId CPatterned::GetWaypointForState(CStateManager& mgr, EScriptObjectState state,
 | |
|                                           EScriptObjectMessage msg) const {
 | |
|   rstl::reserved_vector<TUniqueId, 12> ids;
 | |
|   for (const auto& conn : GetConnectionList()) {
 | |
|     if (conn.x0_state == state && conn.x4_msg == msg) {
 | |
|       TUniqueId id = mgr.GetIdForScript(conn.x8_objId);
 | |
|       if (const CEntity* ent = mgr.GetObjectById(id)) {
 | |
|         if (ent->GetActive()) {
 | |
|           ids.push_back(id);
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (!ids.empty()) {
 | |
|     return ids[mgr.GetActiveRandom()->Next() % ids.size()];
 | |
|   }
 | |
| 
 | |
|   return kInvalidUniqueId;
 | |
| }
 | |
| 
 | |
| void CPatterned::UpdateActorKeyframe(CStateManager& mgr) const {
 | |
|   if (TCastToConstPtr<CScriptWaypoint> wp = mgr.GetObjectById(x2dc_destObj)) {
 | |
|     for (const auto& conn : wp->GetConnectionList()) {
 | |
|       if (conn.x0_state == EScriptObjectState::Arrived && conn.x4_msg == EScriptObjectMessage::Action) {
 | |
|         if (TCastToPtr<CScriptActorKeyframe> kf = mgr.ObjectById(mgr.GetIdForScript(conn.x8_objId))) {
 | |
|           if (kf->GetActive() && kf->IsPassive()) {
 | |
|             kf->UpdateEntity(GetUniqueId(), mgr);
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| pas::EStepDirection CPatterned::GetStepDirection(const zeus::CVector3f& moveVec) const {
 | |
|   zeus::CVector3f localMove = x34_transform.transposeRotate(moveVec);
 | |
|   float angle = zeus::CVector3f::getAngleDiff(localMove, zeus::skForward);
 | |
|   if (angle < zeus::degToRad(45.f)) {
 | |
|     return pas::EStepDirection::Forward;
 | |
|   }
 | |
|   if (angle > zeus::degToRad(135.f)) {
 | |
|     return pas::EStepDirection::Backward;
 | |
|   }
 | |
|   if (localMove.dot(zeus::skRight) > 0.f) {
 | |
|     return pas::EStepDirection::Right;
 | |
|   }
 | |
|   return pas::EStepDirection::Left;
 | |
| }
 | |
| 
 | |
| bool CPatterned::IsPatternObstructed(CStateManager& mgr, const zeus::CVector3f& p0, const zeus::CVector3f& p1) const {
 | |
|   CMaterialFilter filter = CMaterialFilter::MakeInclude(EMaterialTypes::Character);
 | |
|   zeus::CVector3f delta = p1 - p0;
 | |
|   rstl::reserved_vector<TUniqueId, 1024> nearList;
 | |
|   bool ret = false;
 | |
|   if (delta.canBeNormalized()) {
 | |
|     zeus::CVector3f deltaNorm = delta.normalized();
 | |
|     float deltaMag = delta.magnitude();
 | |
|     mgr.BuildNearList(nearList, p0, deltaNorm, deltaMag, filter, this);
 | |
|     TUniqueId bestId = kInvalidUniqueId;
 | |
|     CRayCastResult res = mgr.RayWorldIntersection(bestId, p0, deltaNorm, deltaMag,
 | |
|                                                   CMaterialFilter::MakeInclude(EMaterialTypes::Solid), nearList);
 | |
|     ret = res.IsValid();
 | |
|   }
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| void CPatterned::UpdateDest(CStateManager& mgr) {
 | |
|   if (x328_24_inPosition && x2dc_destObj != kInvalidUniqueId) {
 | |
|     if (TCastToPtr<CScriptWaypoint> wp = mgr.ObjectById(x2dc_destObj)) {
 | |
|       UpdateActorKeyframe(mgr);
 | |
|       x2dc_destObj = wp->NextWaypoint(mgr);
 | |
|       if (x2dc_destObj != kInvalidUniqueId) {
 | |
|         x2ec_reflectedDestPos = GetTranslation();
 | |
|         x328_24_inPosition = false;
 | |
|         if (TCastToConstPtr<CScriptWaypoint> wp2 = mgr.GetObjectById(x2dc_destObj)) {
 | |
|           x3b0_moveSpeed = wp->GetSpeed();
 | |
|           x30c_behaviourOrient = EBehaviourOrient(wp->GetBehaviourOrient());
 | |
|           if ((wp->GetBehaviourModifiers() & 0x2) != 0) {
 | |
|             x450_bodyController->GetCommandMgr().DeliverCmd(CBCJumpCmd(wp2->GetTranslation(), pas::EJumpType::Normal));
 | |
|           } else if ((wp->GetBehaviourModifiers() & 0x4) != 0) {
 | |
|             TUniqueId wp3Id = wp2->NextWaypoint(mgr);
 | |
|             if (wp3Id != kInvalidUniqueId) {
 | |
|               if (TCastToConstPtr<CScriptWaypoint> wp3 = mgr.GetObjectById(wp3Id)) {
 | |
|                 x450_bodyController->GetCommandMgr().DeliverCmd(
 | |
|                     CBCJumpCmd(wp2->GetTranslation(), wp3->GetTranslation(), pas::EJumpType::Normal));
 | |
|               }
 | |
|             }
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|       mgr.SendScriptMsg(wp.GetPtr(), GetUniqueId(), EScriptObjectMessage::Arrived);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (x2dc_destObj != kInvalidUniqueId) {
 | |
|     if (TCastToConstPtr<CActor> act = mgr.GetObjectById(x2dc_destObj)) {
 | |
|       SetDestPos(act->GetTranslation());
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CPatterned::ApproachDest(CStateManager& mgr) {
 | |
|   zeus::CVector3f faceVec = mgr.GetPlayer().GetTranslation() - GetTranslation();
 | |
|   zeus::CVector3f moveVec = x2e0_destPos - GetTranslation();
 | |
|   if (!x328_25_verticalMovement) {
 | |
|     moveVec.z() = 0.f;
 | |
|     faceVec.z() = 0.f;
 | |
|   }
 | |
|   zeus::CVector3f pathLine = x2e0_destPos - x2ec_reflectedDestPos;
 | |
|   if (pathLine.dot(moveVec) <= 0.f) {
 | |
|     x328_24_inPosition = true;
 | |
|   } else if (moveVec.magSquared() < 3.f * 3.f) {
 | |
|     moveVec = pathLine;
 | |
|   }
 | |
|   if (!x328_24_inPosition) {
 | |
|     if (moveVec.canBeNormalized()) {
 | |
|       moveVec.normalize();
 | |
|     }
 | |
|     switch (x30c_behaviourOrient) {
 | |
|     case EBehaviourOrient::MoveDir:
 | |
|       faceVec = moveVec;
 | |
|       break;
 | |
|     case EBehaviourOrient::Destination:
 | |
|       if (x39c_curPattern != 0u && x39c_curPattern < x38c_patterns.size()) {
 | |
|         faceVec = x38c_patterns[x39c_curPattern].GetForward();
 | |
|       } else if (x2dc_destObj != kInvalidUniqueId) {
 | |
|         if (TCastToConstPtr<CScriptWaypoint> wp = mgr.GetObjectById(x2dc_destObj)) {
 | |
|           faceVec = wp->GetTransform().basis[1];
 | |
|         }
 | |
|       }
 | |
|       break;
 | |
|     default:
 | |
|       break;
 | |
|     }
 | |
|     x31c_faceVec = faceVec;
 | |
|     x310_moveVec = x3b0_moveSpeed * moveVec;
 | |
|     if (!KnockbackWhenFrozen()) {
 | |
|       x450_bodyController->GetCommandMgr().DeliverCmd(CBCLocomotionCmd(x310_moveVec, x31c_faceVec, 1.f));
 | |
|     } else if (x30c_behaviourOrient == EBehaviourOrient::MoveDir ||
 | |
|                !x450_bodyController->HasBodyState(pas::EAnimationState::Step)) {
 | |
|       x450_bodyController->GetCommandMgr().DeliverCmd(CBCLocomotionCmd(x310_moveVec, zeus::skZero3f, 1.f));
 | |
|     } else {
 | |
|       pas::EStepDirection stepDir = GetStepDirection(x310_moveVec);
 | |
|       if (stepDir == pas::EStepDirection::Forward) {
 | |
|         x450_bodyController->GetCommandMgr().DeliverCmd(CBCLocomotionCmd(x310_moveVec, zeus::skZero3f, 1.f));
 | |
|       } else {
 | |
|         x450_bodyController->GetCommandMgr().DeliverCmd(CBCStepCmd(stepDir, pas::EStepType::Normal));
 | |
|       }
 | |
|       x450_bodyController->GetCommandMgr().DeliverTargetVector(x31c_faceVec);
 | |
|     }
 | |
|   } else if (x450_bodyController->GetBodyStateInfo().GetMaxSpeed() > FLT_EPSILON) {
 | |
|     x450_bodyController->GetCommandMgr().DeliverCmd(CBCLocomotionCmd(
 | |
|         (x138_velocity.magnitude() / x450_bodyController->GetBodyStateInfo().GetMaxSpeed()) * x34_transform.basis[1],
 | |
|         zeus::skZero3f, 1.f));
 | |
|   }
 | |
| }
 | |
| 
 | |
| std::pair<CScriptWaypoint*, CScriptWaypoint*> CPatterned::GetDestWaypoints(CStateManager& mgr) const {
 | |
|   std::pair<CScriptWaypoint*, CScriptWaypoint*> ret{};
 | |
|   if (TCastToPtr<CScriptWaypoint> wp = mgr.ObjectById(x2dc_destObj)) {
 | |
|     ret.first = wp.GetPtr();
 | |
|     ret.second = TCastToPtr<CScriptWaypoint>(mgr.ObjectById(wp->FollowWaypoint(mgr))).GetPtr();
 | |
|   }
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| zeus::CQuaternion CPatterned::FindPatternRotation(const zeus::CVector3f& dir) const {
 | |
|   zeus::CVector3f wpDeltaFlat = x368_destWPDelta;
 | |
|   wpDeltaFlat.z() = 0.f;
 | |
|   wpDeltaFlat.normalize();
 | |
|   zeus::CVector3f dirFlat = dir;
 | |
|   dirFlat.z() = 0.f;
 | |
|   dirFlat.normalize();
 | |
| 
 | |
|   zeus::CQuaternion q;
 | |
|   if ((wpDeltaFlat - dirFlat).magSquared() > 3.99f) {
 | |
|     q.rotateZ(zeus::degToRad(180.f));
 | |
|   } else {
 | |
|     q = zeus::CQuaternion::shortestRotationArc(wpDeltaFlat, dirFlat);
 | |
|   }
 | |
| 
 | |
|   if (x328_25_verticalMovement) {
 | |
|     q = zeus::CQuaternion::shortestRotationArc(
 | |
|             (q * zeus::CQuaternion(0.f, x368_destWPDelta) * q.inverse()).getImaginary().normalized(),
 | |
|             dir.normalized()) *
 | |
|         q;
 | |
|   }
 | |
| 
 | |
|   return q;
 | |
| }
 | |
| 
 | |
| zeus::CVector3f CPatterned::FindPatternDir(CStateManager& mgr) const {
 | |
|   zeus::CVector3f ret;
 | |
|   switch (x378_patternOrient) {
 | |
|   case EPatternOrient::StartToPlayerStart:
 | |
|     ret = x35c_patternStartPlayerPos - x350_patternStartPos;
 | |
|     break;
 | |
|   case EPatternOrient::StartToPlayer:
 | |
|     ret = mgr.GetPlayer().GetTranslation() - x350_patternStartPos;
 | |
|     break;
 | |
|   case EPatternOrient::ReversePlayerForward:
 | |
|     ret = -mgr.GetPlayer().GetTransform().basis[1];
 | |
|     break;
 | |
|   case EPatternOrient::Forward:
 | |
|     ret = GetTransform().basis[1];
 | |
|     break;
 | |
|   default:
 | |
|     break;
 | |
|   }
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| void CPatterned::UpdatePatternDestPos(CStateManager& mgr) {
 | |
|   if (x39c_curPattern < x38c_patterns.size()) {
 | |
|     if (x368_destWPDelta != zeus::skZero3f) {
 | |
|       zeus::CVector3f patternDir = FindPatternDir(mgr);
 | |
|       SetDestPos(FindPatternRotation(patternDir).transform(x38c_patterns[x39c_curPattern].GetPos()));
 | |
|       if (x37c_patternFit == EPatternFit::Zero) {
 | |
|         float magSq = x328_25_verticalMovement
 | |
|                           ? patternDir.magSquared() / x368_destWPDelta.magSquared()
 | |
|                           : patternDir.toVec2f().magSquared() / x368_destWPDelta.toVec2f().magSquared();
 | |
|         SetDestPos(std::sqrt(magSq) * x2e0_destPos);
 | |
|       }
 | |
|     } else {
 | |
|       SetDestPos(x38c_patterns[x39c_curPattern].GetPos());
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   switch (x374_patternTranslate) {
 | |
|   case EPatternTranslate::RelativeStart:
 | |
|     SetDestPos(x2e0_destPos + x350_patternStartPos);
 | |
|     break;
 | |
|   case EPatternTranslate::RelativePlayerStart:
 | |
|     SetDestPos(x2e0_destPos + x35c_patternStartPlayerPos);
 | |
|     break;
 | |
|   case EPatternTranslate::RelativePlayer:
 | |
|     SetDestPos(x2e0_destPos + mgr.GetPlayer().GetTranslation());
 | |
|     break;
 | |
|   default:
 | |
|     break;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CPatterned::SetupPattern(CStateManager& mgr) {
 | |
|   EScriptObjectState state = GetDesiredAttackState(mgr);
 | |
|   x2dc_destObj = GetWaypointForState(mgr, state, EScriptObjectMessage::Follow);
 | |
|   if (x2dc_destObj == kInvalidUniqueId && state != EScriptObjectState::Attack) {
 | |
|     x2dc_destObj = GetWaypointForState(mgr, EScriptObjectState::Attack, EScriptObjectMessage::Follow);
 | |
|   }
 | |
|   x38c_patterns.clear();
 | |
|   if (x2dc_destObj != kInvalidUniqueId) {
 | |
|     x350_patternStartPos = GetTranslation();
 | |
|     x35c_patternStartPlayerPos = mgr.GetPlayer().GetTranslation();
 | |
|     auto destWPs = GetDestWaypoints(mgr);
 | |
|     if (destWPs.first != nullptr) {
 | |
|       x374_patternTranslate = EPatternTranslate(destWPs.first->GetPatternTranslate());
 | |
|       x378_patternOrient = EPatternOrient(destWPs.first->GetPatternOrient());
 | |
|       x37c_patternFit = EPatternFit(destWPs.first->GetPatternFit());
 | |
|       if (destWPs.second != nullptr) {
 | |
|         x368_destWPDelta = destWPs.second->GetTranslation() - destWPs.first->GetTranslation();
 | |
|       } else {
 | |
|         x368_destWPDelta = zeus::skZero3f;
 | |
|       }
 | |
| 
 | |
|       int numPatterns = 0;
 | |
|       CScriptWaypoint* curWp = destWPs.first;
 | |
|       do {
 | |
|         ++numPatterns;
 | |
|         curWp = TCastToPtr<CScriptWaypoint>(mgr.ObjectById(curWp->NextWaypoint(mgr))).GetPtr();
 | |
|         if (curWp == nullptr) {
 | |
|           break;
 | |
|         }
 | |
|       } while (curWp->GetUniqueId() != destWPs.first->GetUniqueId());
 | |
|       x38c_patterns.reserve(numPatterns);
 | |
| 
 | |
|       zeus::CVector3f basePos;
 | |
|       switch (x374_patternTranslate) {
 | |
|       case EPatternTranslate::RelativePlayerStart:
 | |
|         if (destWPs.second != nullptr) {
 | |
|           basePos = destWPs.second->GetTranslation();
 | |
|         }
 | |
|         break;
 | |
|       case EPatternTranslate::Absolute:
 | |
|         break;
 | |
|       default:
 | |
|         basePos = destWPs.first->GetTranslation();
 | |
|         break;
 | |
|       }
 | |
| 
 | |
|       curWp = destWPs.first;
 | |
|       do {
 | |
|         zeus::CVector3f wpForward = curWp->GetTransform().basis[1];
 | |
|         if (x368_destWPDelta != zeus::skZero3f) {
 | |
|           wpForward = FindPatternRotation(FindPatternDir(mgr)).transform(wpForward);
 | |
|         }
 | |
|         x38c_patterns.emplace_back(curWp->GetTranslation() - basePos, wpForward, curWp->GetSpeed(),
 | |
|                                    curWp->GetBehaviour(), curWp->GetBehaviourOrient(), curWp->GetBehaviourModifiers(),
 | |
|                                    curWp->GetAnimation());
 | |
|         curWp = TCastToPtr<CScriptWaypoint>(mgr.ObjectById(curWp->NextWaypoint(mgr))).GetPtr();
 | |
|         if (curWp == nullptr) {
 | |
|           break;
 | |
|         }
 | |
|       } while (curWp->GetUniqueId() != destWPs.first->GetUniqueId());
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   x400_30_patternShagged = false;
 | |
|   x39c_curPattern = 0;
 | |
|   x328_24_inPosition = false;
 | |
|   x2ec_reflectedDestPos = GetTranslation();
 | |
|   if (!x38c_patterns.empty()) {
 | |
|     x3b0_moveSpeed = x38c_patterns.front().GetSpeed();
 | |
|     x380_behaviour = EBehaviour(x38c_patterns.front().GetBehaviour());
 | |
|     x30c_behaviourOrient = EBehaviourOrient(x38c_patterns.front().GetBehaviourOrient());
 | |
|     x384_behaviourModifiers = EBehaviourModifiers(x38c_patterns.front().GetBehaviourModifiers());
 | |
|   }
 | |
| }
 | |
| 
 | |
| EScriptObjectState CPatterned::GetDesiredAttackState(CStateManager& mgr) const {
 | |
|   float deltaMagSq = (GetTranslation() - mgr.GetPlayer().GetTranslation()).magSquared();
 | |
|   if (deltaMagSq < x2fc_minAttackRange * x2fc_minAttackRange) {
 | |
|     return EScriptObjectState::Retreat;
 | |
|   }
 | |
|   if (deltaMagSq > x300_maxAttackRange * x300_maxAttackRange) {
 | |
|     return EScriptObjectState::CloseIn;
 | |
|   }
 | |
|   return EScriptObjectState::Attack;
 | |
| }
 | |
| 
 | |
| float CPatterned::GetAnimationDistance(const CPASAnimParmData& data) const {
 | |
|   auto bestAnim = x64_modelData->GetAnimationData()->GetCharacterInfo().GetPASDatabase().FindBestAnimation(data, -1);
 | |
|   float dist = 1.f;
 | |
|   if (bestAnim.first > FLT_EPSILON) {
 | |
|     dist = x64_modelData->GetAnimationData()->GetAnimationDuration(bestAnim.second) *
 | |
|            x64_modelData->GetAnimationData()->GetAverageVelocity(bestAnim.second);
 | |
|   }
 | |
|   return dist;
 | |
| }
 | |
| 
 | |
| void CPatterned::PreRender(CStateManager& mgr, const zeus::CFrustum& frustum) {
 | |
|   if (mgr.GetPlayerState()->GetActiveVisor(mgr) == CPlayerState::EPlayerVisor::Thermal) {
 | |
|     SetCalculateLighting(false);
 | |
|     x90_actorLights->BuildConstantAmbientLighting(zeus::skWhite);
 | |
|   } else {
 | |
|     SetCalculateLighting(true);
 | |
|   }
 | |
| 
 | |
|   zeus::CColor col = x42c_color;
 | |
|   u8 alpha = GetModelAlphau8(mgr);
 | |
|   if (x402_27_noXrayModel && mgr.GetPlayerState()->GetActiveVisor(mgr) == CPlayerState::EPlayerVisor::XRay) {
 | |
|     alpha = 76;
 | |
|   }
 | |
| 
 | |
|   if (alpha < 255) {
 | |
|     if (col.r() == 0.f && col.g() == 0.f && col.b() == 0.f) {
 | |
|       col = zeus::skWhite; /* Not being damaged */
 | |
|     }
 | |
| 
 | |
|     if (x401_29_laggedBurnDeath) {
 | |
|       int stripedAlpha = 255;
 | |
|       if (alpha > 127) {
 | |
|         stripedAlpha = (alpha - 128) * 2;
 | |
|       }
 | |
|       xb4_drawFlags = CModelFlags(3, 0, 3, zeus::CColor(0.f, float(stripedAlpha * stripedAlpha) / 65025.f));
 | |
|     } else if (x401_28_burning) {
 | |
|       xb4_drawFlags = CModelFlags(5, 0, 3, zeus::CColor(0.f, 1.f));
 | |
|     } else {
 | |
|       zeus::CColor col2 = col;
 | |
|       col2.a() = float(alpha) / 255.f;
 | |
|       xb4_drawFlags = CModelFlags(5, 0, 3, col2);
 | |
|     }
 | |
|   } else {
 | |
|     if (col.r() != 0.f || col.g() != 0.f || col.b() != 0.f) {
 | |
|       /* Being damaged */
 | |
|       zeus::CColor col2 = col;
 | |
|       col2.a() = float(alpha) / 255.f;
 | |
|       xb4_drawFlags = CModelFlags(2, 0, 3, zeus::skWhite);
 | |
|       /* Make color additive */
 | |
|       xb4_drawFlags.addColor = col2;
 | |
|     } else {
 | |
|       xb4_drawFlags = CModelFlags(0, 0, 3, zeus::skWhite);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   CActor::PreRender(mgr, frustum);
 | |
| }
 | |
| 
 | |
| void CPatterned::AddToRenderer(const zeus::CFrustum& frustum, CStateManager& mgr) {
 | |
|   if (x402_29_drawParticles) {
 | |
|     if (x64_modelData && !x64_modelData->IsNull()) {
 | |
|       int mask = 0;
 | |
|       int target = 0;
 | |
|       mgr.GetCharacterRenderMaskAndTarget(x402_31_thawed, mask, target);
 | |
|       if (CAnimData* aData = x64_modelData->GetAnimationData()) {
 | |
|         aData->GetParticleDB().AddToRendererClippedMasked(frustum, mask, target);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   CActor::AddToRenderer(frustum, mgr);
 | |
| }
 | |
| 
 | |
| void CPatterned::RenderIceModelWithFlags(const CModelFlags& flags) const {
 | |
|   CModelFlags useFlags = flags;
 | |
|   useFlags.x1_matSetIdx = 0;
 | |
|   CAnimData* animData = x64_modelData->GetAnimationData();
 | |
|   if (CMorphableSkinnedModel* iceModel = animData->GetIceModel().GetObj()) {
 | |
|     animData->Render(*iceModel, useFlags, {*x510_vertexMorph}, iceModel->GetMorphMagnitudes());
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CPatterned::Render(CStateManager& mgr) {
 | |
|   int mask = 0;
 | |
|   int target = 0;
 | |
|   if (x402_29_drawParticles) {
 | |
|     mgr.GetCharacterRenderMaskAndTarget(x402_31_thawed, mask, target);
 | |
|     x64_modelData->GetAnimationData()->GetParticleDB().RenderSystemsToBeDrawnFirstMasked(mask, target);
 | |
|   }
 | |
|   if ((mgr.GetThermalDrawFlag() == EThermalDrawFlag::Cold && !x402_31_thawed) ||
 | |
|       (mgr.GetThermalDrawFlag() == EThermalDrawFlag::Hot && x402_31_thawed) ||
 | |
|       mgr.GetThermalDrawFlag() == EThermalDrawFlag::Bypass) {
 | |
|     if (x401_28_burning) {
 | |
|       const CTexture* ashy = mgr.GetActorModelParticles()->GetAshyTexture(*this);
 | |
|       u8 alpha = GetModelAlphau8(mgr);
 | |
|       if (ashy != nullptr && ((!x401_29_laggedBurnDeath && alpha <= 255) || alpha <= 127)) {
 | |
|         if (xe5_31_pointGeneratorParticles) {
 | |
|           mgr.SetupParticleHook(*this);
 | |
|         }
 | |
|         zeus::CColor addColor;
 | |
|         if (x401_29_laggedBurnDeath) {
 | |
|           addColor = zeus::skClear;
 | |
|         } else {
 | |
|           addColor = mgr.GetThermalDrawFlag() == EThermalDrawFlag::Hot ? zeus::skWhite : zeus::skBlack;
 | |
|         }
 | |
|         x64_modelData->DisintegrateDraw(mgr, GetTransform(), *ashy, addColor,
 | |
|                                         float(alpha) * (x401_29_laggedBurnDeath ? 0.00787402f : 0.00392157f));
 | |
|         if (xe5_31_pointGeneratorParticles) {
 | |
|           CSkinnedModel::ClearPointGeneratorFunc();
 | |
|           mgr.GetActorModelParticles()->Render(mgr, *this);
 | |
|         }
 | |
|       } else {
 | |
|         CPhysicsActor::Render(mgr);
 | |
|       }
 | |
|     } else {
 | |
|       CPhysicsActor::Render(mgr);
 | |
|     }
 | |
| 
 | |
|     if (x450_bodyController->IsFrozen() && !x401_28_burning) {
 | |
|       RenderIceModelWithFlags(CModelFlags(0, 0, 3, zeus::skWhite));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (x402_29_drawParticles) {
 | |
|     x64_modelData->GetAnimationData()->GetParticleDB().RenderSystemsToBeDrawnLastMasked(mask, target);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CPatterned::ThinkAboutMove(float dt) {
 | |
|   bool doMove = true;
 | |
|   if (!x328_25_verticalMovement && !x328_27_onGround) {
 | |
|     x310_moveVec.zeroOut();
 | |
|     doMove = false;
 | |
|   }
 | |
| 
 | |
|   if (doMove && x39c_curPattern < x38c_patterns.size()) {
 | |
|     zeus::CVector3f frontVec = GetTransform().frontVector();
 | |
|     zeus::CVector3f x31cCpy = x31c_faceVec;
 | |
|     if (x31c_faceVec.magSquared() > 0.1f) {
 | |
|       x31cCpy.normalize();
 | |
|     }
 | |
|     float mag = frontVec.dot(x31cCpy);
 | |
| 
 | |
|     switch (x3f8_moveState) {
 | |
|     case EMoveState::Zero:
 | |
|       if (!x328_26_solidCollision) {
 | |
|         break;
 | |
|       }
 | |
|       [[fallthrough]];
 | |
|     case EMoveState::One:
 | |
|       doMove = false;
 | |
|       if (mag > 0.85f) {
 | |
|         doMove = true;
 | |
|         x3f8_moveState = EMoveState::Two;
 | |
|         break;
 | |
|       }
 | |
|       x3f8_moveState = EMoveState::One;
 | |
|       break;
 | |
|     case EMoveState::Two:
 | |
|       x3f8_moveState = EMoveState::Three;
 | |
|       [[fallthrough]];
 | |
|     case EMoveState::Three:
 | |
|       doMove = true;
 | |
|       if (!x328_26_solidCollision) {
 | |
|         x3f8_moveState = EMoveState::Zero;
 | |
|         break;
 | |
|       }
 | |
|       if (mag > 0.9f) {
 | |
|         x3f8_moveState = EMoveState::Four;
 | |
|       }
 | |
|       break;
 | |
|     case EMoveState::Four:
 | |
|       x328_24_inPosition = true;
 | |
|       doMove = false;
 | |
|       x3f8_moveState = EMoveState::Zero;
 | |
|       break;
 | |
|     default:
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (!x401_26_disableMove && doMove) {
 | |
|     const CBodyState* state = x450_bodyController->GetBodyStateInfo().GetCurrentState();
 | |
|     if (state->ApplyAnimationDeltas() && !zeus::close_enough(x2e0_destPos - GetTranslation(), {})) {
 | |
|       MoveToOR((x64_modelData->GetScale() * x434_posDelta) * x55c_moveScale, dt);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   RotateToOR(x440_rotDelta, dt);
 | |
| }
 | |
| 
 | |
| void CPatterned::PhazeOut(CStateManager& mgr) {
 | |
|   if (!x400_27_fadeToDeath) {
 | |
|     SendScriptMsgs(EScriptObjectState::DeathRattle, mgr, EScriptObjectMessage::None);
 | |
|   }
 | |
|   x401_27_phazingOut = true;
 | |
|   x450_bodyController->SetPlaybackRate(0.f);
 | |
|   x64_modelData->GetAnimationData()->GetParticleDB().SetUpdatesEnabled(false);
 | |
| }
 | |
| 
 | |
| bool CPatterned::ApplyBoneTracking() const {
 | |
|   if (x400_25_alive) {
 | |
|     return x460_knockBackController.GetFlinchRemTime() <= 0.f;
 | |
|   }
 | |
| 
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| void CPatterned::Initialize() {
 | |
|   if (cv_disableAi == nullptr) {
 | |
|     cv_disableAi = hecl::CVarManager::instance()->findOrMakeCVar("disableAi"sv, "Disables AI state machines", false,
 | |
|                                                                  hecl::CVar::EFlags::Cheat | hecl::CVar::EFlags::Game);
 | |
|   }
 | |
| }
 | |
| 
 | |
| } // namespace metaforce
 |