mirror of
				https://github.com/AxioDL/metaforce.git
				synced 2025-10-24 22:50:24 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			1188 lines
		
	
	
		
			45 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1188 lines
		
	
	
		
			45 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "Runtime/MP1/World/CDrone.hpp"
 | |
| 
 | |
| #include "Runtime/Audio/CSfxManager.hpp"
 | |
| #include "Runtime/Collision/CGameCollision.hpp"
 | |
| #include "Runtime/CSimplePool.hpp"
 | |
| #include "Runtime/CStateManager.hpp"
 | |
| #include "Runtime/GameGlobalObjects.hpp"
 | |
| #include "Runtime/MP1/World/CDroneLaser.hpp"
 | |
| #include "Runtime/Particle/CWeaponDescription.hpp"
 | |
| #include "Runtime/Weapon/CGameProjectile.hpp"
 | |
| #include "Runtime/Weapon/CWeapon.hpp"
 | |
| #include "Runtime/World/CGameLight.hpp"
 | |
| #include "Runtime/World/CPatternedInfo.hpp"
 | |
| #include "Runtime/World/CPlayer.hpp"
 | |
| #include "Runtime/World/CScriptVisorFlare.hpp"
 | |
| #include "Runtime/World/CScriptWater.hpp"
 | |
| #include "Runtime/World/CTeamAiMgr.hpp"
 | |
| #include "Runtime/World/CWorld.hpp"
 | |
| 
 | |
| #include "DataSpec/DNAMP1/SFX/Drones.h"
 | |
| 
 | |
| #include "TCastTo.hpp" // Generated file, do not modify include path
 | |
| 
 | |
| #include <algorithm>
 | |
| 
 | |
| namespace urde::MP1 {
 | |
| CDrone::CDrone(TUniqueId uid, std::string_view name, EFlavorType flavor, const CEntityInfo& info,
 | |
|                const zeus::CTransform& xf, float f1, CModelData&& mData, const CPatternedInfo& pInfo,
 | |
|                const CActorParameters& actParms, EMovementType movement, EColliderType colliderType, EBodyType bodyType,
 | |
|                const CDamageInfo& dInfo1, CAssetId aId1, const CDamageInfo& dInfo2, CAssetId aId2,
 | |
|                std::vector<CVisorFlare::CFlareDef> flares, float f2, float f3, float f4, float f5, float f6, float f7,
 | |
|                float f8, float f9, float f10, float f11, float f12, float f13, float f14, float f15, float f16,
 | |
|                float f17, float f18, float f19, float f20, CAssetId crscId, float f21, float f22, float f23, float f24,
 | |
|                s32 sId, bool b1)
 | |
| : CPatterned(ECharacter::Drone, uid, name, flavor, info, xf, std::move(mData), pInfo, movement, colliderType, bodyType,
 | |
|              actParms, flavor == EFlavorType::Zero ? EKnockBackVariant::Medium : EKnockBackVariant::Large)
 | |
| , x568_laserParticlesId(aId1)
 | |
| , x56c_(g_SimplePool->GetObj({SBIG('CRSC'), crscId}))
 | |
| , x57c_flares(std::move(flares))
 | |
| , x590_(dInfo1)
 | |
| , x5ac_(dInfo2)
 | |
| , x5e4_(f23)
 | |
| , x5ec_turnSpeed(f1)
 | |
| , x5f0_(f2)
 | |
| , x5f4_(f3)
 | |
| , x5f8_(f4)
 | |
| , x5fc_(f5)
 | |
| , x600_(f11)
 | |
| , x608_(f6)
 | |
| , x60c_(f7)
 | |
| , x610_(f8)
 | |
| , x614_(f9)
 | |
| , x618_(f10)
 | |
| , x61c_(f12)
 | |
| , x620_(f20)
 | |
| , x63c_(f13)
 | |
| , x640_(f14)
 | |
| , x648_(f15)
 | |
| , x64c_(f16)
 | |
| , x650_(f17)
 | |
| , x654_(f18)
 | |
| , x658_(f19)
 | |
| , x65c_(f21)
 | |
| , x660_(f22)
 | |
| , x664_(f24)
 | |
| , x690_colSphere(zeus::CSphere({0.f, 0.f, 1.8f}, 1.1f), CActor::GetMaterialList())
 | |
| , x6b0_pathFind(nullptr, 3 + int(b1) /* TODO double check */, pInfo.GetPathfindingIndex(), 1.f, 2.4f)
 | |
| , x7cc_laserSfx(CSfxManager::TranslateSFXID(sId))
 | |
| , x82c_shieldModel(std::make_unique<CModelData>(CStaticRes{aId2, zeus::skOne3f}))
 | |
| , x835_25_(b1) {
 | |
|   UpdateTouchBounds(pInfo.GetHalfExtent());
 | |
|   x460_knockBackController.SetEnableShock(true);
 | |
|   x460_knockBackController.SetAvailableState(EKnockBackAnimationState::Hurled, false);
 | |
|   x460_knockBackController.SetLocomotionDuringElectrocution(true);
 | |
|   MakeThermalColdAndHot();
 | |
|   CreateShadow(flavor != EFlavorType::One);
 | |
| }
 | |
| 
 | |
| void CDrone::Accept(IVisitor& visitor) { visitor.Visit(this); }
 | |
| 
 | |
| void CDrone::Think(float dt, CStateManager& mgr) {
 | |
|   if (x3fc_flavor == EFlavorType::One) {
 | |
|     if (mgr.GetPlayerState()->GetActiveVisor(mgr) == CPlayerState::EPlayerVisor::XRay) {
 | |
|       x42c_color.a() = 1.f;
 | |
|     } else {
 | |
|       x42c_color.a() = std::max(0.f, x428_damageCooldownTimer / 0.33f);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   x403_25_enableStateMachine = !GetBodyController()->IsElectrocuting();
 | |
|   if (GetBodyController()->IsElectrocuting() && (x824_activeLasers[0] || x824_activeLasers[1])) {
 | |
|     x824_activeLasers[0] = false;
 | |
|     x824_activeLasers[1] = false;
 | |
|     UpdateLaser(mgr, 0, false);
 | |
|     UpdateLaser(mgr, 1, false);
 | |
|     SetVisorFlareEnabled(mgr, false);
 | |
|   }
 | |
|   CPatterned::Think(dt, mgr);
 | |
| 
 | |
|   if (!GetActive())
 | |
|     return;
 | |
| 
 | |
|   x5c8_ -= dt;
 | |
|   if (x7c4_ > 0.f) {
 | |
|     x7c4_ -= dt;
 | |
|   }
 | |
| 
 | |
|   if (x5d0_ > 0.f) {
 | |
|     x5d0_ -= (mgr.GetPlayer().GetMorphballTransitionState() == CPlayer::EPlayerMorphBallState::Morphed ? 3.f * dt : dt);
 | |
|   }
 | |
| 
 | |
|   if (x624_ > 0.f) {
 | |
|     x624_ -= dt;
 | |
|   }
 | |
| 
 | |
|   if (x644_ > 0.f) {
 | |
|     x644_ -= dt;
 | |
|   }
 | |
| 
 | |
|   if (x824_activeLasers[0] || (x824_activeLasers[1] && IsAlive())) {
 | |
|     UpdateLasers(mgr, dt);
 | |
|     UpdateVisorFlare(mgr);
 | |
|   }
 | |
| 
 | |
|   if (x834_25_ && IsAlive()) {
 | |
|     UpdateScanner(mgr, dt);
 | |
|   }
 | |
| 
 | |
|   const float dist = (mgr.GetPlayer().GetTranslation() - GetTranslation()).magSquared();
 | |
|   if (x834_28_ && dist < x60c_ * x60c_) {
 | |
|     mgr.GetPlayerState()->GetStaticInterference().RemoveSource(GetUniqueId());
 | |
|     mgr.GetPlayerState()->GetStaticInterference().AddSource(
 | |
|         GetUniqueId(), std::max(0.f, x608_ - mgr.GetPlayerState()->GetStaticInterference().GetTotalInterference()),
 | |
|         0.2f);
 | |
|   }
 | |
| 
 | |
|   if (!x834_28_ && dist < x614_ * x614_) {
 | |
|     mgr.GetPlayerState()->GetStaticInterference().RemoveSource(GetUniqueId());
 | |
|     mgr.GetPlayerState()->GetStaticInterference().AddSource(
 | |
|         GetUniqueId(), std::max(0.f, x610_ - mgr.GetPlayerState()->GetStaticInterference().GetTotalInterference()),
 | |
|         0.2f);
 | |
|   }
 | |
| 
 | |
|   if (!x834_28_ && IsAlive() && !x835_25_) {
 | |
|     x5e0_ -= dt;
 | |
|     if (x5e0_ < 0.f) {
 | |
|       sub_801633a8(mgr);
 | |
|       x5e0_ = 0.1f;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   const float healthDiff = x604_ - HealthInfo(mgr)->GetHP();
 | |
|   if (!zeus::close_enough(x600_, 0.f)) {
 | |
|     x5d0_ -= healthDiff / x600_;
 | |
|     x624_ -= healthDiff / x600_;
 | |
|   }
 | |
|   x604_ = HealthInfo(mgr)->GetHP();
 | |
|   if (x3fc_flavor == EFlavorType::One) {
 | |
|     if (!x834_30_visible) {
 | |
|       x5dc_ = zeus::max(0.f, x5dc_ - (3.f * dt));
 | |
|     } else {
 | |
|       x5dc_ = zeus::min(1.f, x5dc_ + (3.f * dt));
 | |
|     }
 | |
|     x5e8_shieldTime = zeus::max(0.f, x5e8_shieldTime - dt);
 | |
| 
 | |
|     if (zeus::close_enough(x5dc_, 0.f)) {
 | |
|       if (x7d0_) {
 | |
|         CSfxManager::RemoveEmitter(x7d0_);
 | |
|         x7d0_.reset();
 | |
|       }
 | |
|     } else if (!x7d0_ && IsAlive()) {
 | |
|       x7d0_ = CSfxManager::AddEmitter(SFXsfx00DD, GetTranslation(), zeus::skZero3f, true, true, 127, GetAreaIdAlways());
 | |
|     }
 | |
|   }
 | |
|   sub_8015f25c(dt, mgr);
 | |
|   sub_8015f158(dt);
 | |
| 
 | |
|   if (!x835_25_) {
 | |
|     CGameCollision::AvoidStaticCollisionWithinRadius(mgr, *this, 8, dt, 0.25f, 3.5f * GetModelData()->GetScale().y(),
 | |
|                                                      3000.f, 0.5f);
 | |
|   }
 | |
|   if (x66c_ > 0.f) {
 | |
|     x66c_ -= dt;
 | |
|   } else {
 | |
|     x668_elevation = mgr.RayStaticIntersection(GetTranslation(), zeus::skDown, 10000.f,
 | |
|                                                CMaterialFilter::MakeInclude({EMaterialTypes::Solid}))
 | |
|                          .GetT();
 | |
|     x66c_ = 0.f;
 | |
|   }
 | |
| 
 | |
|   if (IsAlive() && x835_25_) {
 | |
|     zeus::CAABox box = GetBoundingBox();
 | |
|     box.accumulateBounds(GetTranslation() + 20.f * zeus::skDown);
 | |
|     rstl::reserved_vector<TUniqueId, 1024> nearList;
 | |
|     mgr.BuildNearList(nearList, GetBoundingBox(), CMaterialFilter::MakeInclude({EMaterialTypes::Trigger}), this);
 | |
|     for (TUniqueId id : nearList) {
 | |
|       if (const TCastToConstPtr<CScriptWater> water = mgr.GetObjectById(id)) {
 | |
|         zeus::CAABox waterBox = water->GetTriggerBoundsWR();
 | |
|         if (waterBox.max.z() - GetTranslation().z() < 3.f) {
 | |
|           float z = 20.f;
 | |
|           if (waterBox.max.z() - GetTranslation().z() < 1.5f) {
 | |
|             z = 60.f;
 | |
|           }
 | |
|           ApplyImpulseWR(GetMoveToORImpulseWR(GetTransform().transposeRotate(z * (dt * zeus::skDown)), dt),
 | |
|                          zeus::CAxisAngle());
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   if (IsAlive() && x668_elevation < x664_) {
 | |
|     ApplyImpulseWR(GetMoveToORImpulseWR(GetTransform().transposeRotate(dt * (1.f * zeus::skUp)), dt),
 | |
|                    zeus::CAxisAngle());
 | |
|   }
 | |
|   xe7_31_targetable = IsAlive();
 | |
| }
 | |
| 
 | |
| void CDrone::AcceptScriptMsg(EScriptObjectMessage msg, TUniqueId sender, CStateManager& mgr) {
 | |
|   CPatterned::AcceptScriptMsg(msg, sender, mgr);
 | |
| 
 | |
|   switch (msg) {
 | |
|   case EScriptObjectMessage::Activate: {
 | |
|     SetLightEnabled(mgr, true);
 | |
|     AddToTeam(mgr);
 | |
|     break;
 | |
|   }
 | |
|   case EScriptObjectMessage::Deactivate:
 | |
|   case EScriptObjectMessage::Deleted: {
 | |
|     for (TUniqueId& unkId : x7d8_laserIds) {
 | |
|       if (unkId != kInvalidUniqueId) {
 | |
|         mgr.FreeScriptObject(unkId);
 | |
|         unkId = kInvalidUniqueId;
 | |
|       }
 | |
|     }
 | |
|     RemoveFromTeam(mgr);
 | |
|     mgr.GetPlayerState()->GetStaticInterference().RemoveSource(GetUniqueId());
 | |
|     if (x578_lightId != kInvalidUniqueId) {
 | |
|       mgr.FreeScriptObject(x578_lightId);
 | |
|       x578_lightId = kInvalidUniqueId;
 | |
|     }
 | |
|     if (x57a_visorFlareId != kInvalidUniqueId) {
 | |
|       mgr.FreeScriptObject(x57a_visorFlareId);
 | |
|       x57a_visorFlareId = kInvalidUniqueId;
 | |
|     }
 | |
| 
 | |
|     if (x7d0_) {
 | |
|       CSfxManager::RemoveEmitter(x7d0_);
 | |
|       x7d0_.reset();
 | |
|     }
 | |
|     break;
 | |
|   }
 | |
|   case EScriptObjectMessage::Alert:
 | |
|     x834_29_codeTrigger = true;
 | |
|     break;
 | |
|   case EScriptObjectMessage::OnFloor:
 | |
|     if (!x835_26_ && x834_24_waveHit && !IsAlive()) {
 | |
|       x835_26_ = true;
 | |
|       MassiveFrozenDeath(mgr);
 | |
|     }
 | |
|     break;
 | |
|   case EScriptObjectMessage::Registered:
 | |
|     x450_bodyController->Activate(mgr);
 | |
|     x450_bodyController->SetLocomotionType(pas::ELocomotionType::Lurk);
 | |
|     x450_bodyController->BodyStateInfo().SetMaximumPitch(0.f);
 | |
|     x5cc_ = 0.f;
 | |
|     x460_knockBackController.SetEnableFreeze(false);
 | |
|     AddMaterial(EMaterialTypes::AIJoint, mgr);
 | |
|     x578_lightId = mgr.AllocateUniqueId();
 | |
|     mgr.AddObject(new CGameLight(x578_lightId, GetAreaIdAlways(), GetActive(), "LaserLight"sv, {}, GetUniqueId(),
 | |
|                                  CLight::BuildPoint(zeus::skZero3f, zeus::skRed), 0, 0, 0.f));
 | |
|     break;
 | |
|   case EScriptObjectMessage::InitializedInArea: {
 | |
|     x6b0_pathFind.SetArea(mgr.GetWorld()->GetAreaAlways(GetAreaIdAlways())->GetPostConstructed()->x10bc_pathArea);
 | |
|     if (x688_teamMgr == kInvalidUniqueId) {
 | |
|       x688_teamMgr = CTeamAiMgr::GetTeamAiMgr(*this, mgr);
 | |
|       if (GetActive()) {
 | |
|         AddToTeam(mgr);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     x604_ = HealthInfo(mgr)->GetHP();
 | |
|     x55c_moveScale = 1.f / GetModelData()->GetScale();
 | |
|     if (x835_25_)
 | |
|       SetSoundEventPitchBend(0);
 | |
|     break;
 | |
|   }
 | |
|   default:
 | |
|     break;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CDrone::AddToRenderer(const zeus::CFrustum& frustum, CStateManager& mgr) {
 | |
|   CPatterned::AddToRenderer(frustum, mgr);
 | |
| }
 | |
| 
 | |
| void CDrone::PreRender(CStateManager& mgr, const zeus::CFrustum& frustum) {
 | |
|   CPatterned::PreRender(mgr, frustum);
 | |
|   if (x3fc_flavor == EFlavorType::One) {
 | |
|     if (HasModelData() && GetModelData()->HasAnimData() && GetModelData()->HasNormalModel()) {
 | |
|       if (GetModelAlphau8(mgr) == 0)
 | |
|         GetModelData()->GetAnimationData()->BuildPose();
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CDrone::Render(CStateManager& mgr) {
 | |
|   bool isOne = x3fc_flavor == EFlavorType::One;
 | |
|   if (!isOne || GetModelAlphau8(mgr) != 0) {
 | |
|     if (isOne && mgr.GetPlayerState()->GetActiveVisor(mgr) == CPlayerState::EPlayerVisor::XRay) {
 | |
|       CElementGen::SetSubtractBlend(true);
 | |
|       CElementGen::SetMoveRedToAlphaBuffer(true);
 | |
|       CGraphics::SetFog(ERglFogMode::PerspLin, 0.f, 75.f, zeus::skBlack);
 | |
|       GetModelData()->GetAnimationData()->GetParticleDB().RenderSystemsToBeDrawnFirst();
 | |
|       mgr.SetupFogForArea3XRange(GetAreaIdAlways());
 | |
|     }
 | |
|     CPatterned::Render(mgr);
 | |
|     if (isOne && mgr.GetPlayerState()->GetActiveVisor(mgr) == CPlayerState::EPlayerVisor::XRay) {
 | |
|       CGraphics::SetFog(ERglFogMode::PerspLin, 0.f, 75.f, zeus::skBlack);
 | |
|       GetModelData()->GetAnimationData()->GetParticleDB().RenderSystemsToBeDrawnLast();
 | |
|       mgr.SetupFogForArea(GetAreaIdAlways());
 | |
|       CElementGen::SetSubtractBlend(false);
 | |
|       CElementGen::SetMoveRedToAlphaBuffer(false);
 | |
|     }
 | |
| 
 | |
|     if (isOne && !zeus::close_enough(x5dc_, 0)) {
 | |
|       x82c_shieldModel->Render(
 | |
|           mgr, GetLctrTransform("Shield_LCTR"sv), GetActorLights(),
 | |
|           CModelFlags{8, 0, 3, zeus::CColor::lerp({1.f, 1.f, 1.f, x5dc_}, {1.f, 0.f, 0.f, 1.f}, x5e8_shieldTime)});
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| bool CDrone::CanRenderUnsorted(const CStateManager& mgr) const {
 | |
|   if (!zeus::close_enough(x5dc_, 0.f))
 | |
|     return false;
 | |
|   return CPatterned::CanRenderUnsorted(mgr);
 | |
| }
 | |
| 
 | |
| const CDamageVulnerability* CDrone::GetDamageVulnerability(const zeus::CVector3f&, const zeus::CVector3f& dir,
 | |
|                                                            const CDamageInfo&) const {
 | |
|   if (x3fc_flavor == EFlavorType::One && HitShield(-dir)) {
 | |
|     x5e8_shieldTime = 1.f;
 | |
|     return &CDamageVulnerability::ReflectVulnerabilty();
 | |
|   }
 | |
|   return CAi::GetDamageVulnerability();
 | |
| }
 | |
| 
 | |
| void CDrone::Touch(CActor& act, CStateManager& mgr) {
 | |
|   CPatterned::Touch(act, mgr);
 | |
|   if (TCastToPtr<CWeapon> weapon = act) {
 | |
|     if (IsAlive()) {
 | |
|       x834_24_waveHit = weapon->GetType() == EWeaponType::Wave;
 | |
|       if (x3fc_flavor == CPatterned::EFlavorType::One && HitShield(weapon->GetTranslation() - GetTranslation())) {
 | |
|         x5e8_shieldTime = 1.f;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| EWeaponCollisionResponseTypes CDrone::GetCollisionResponseType(const zeus::CVector3f&, const zeus::CVector3f& dir,
 | |
|                                                                const CWeaponMode&, EProjectileAttrib) const {
 | |
|   if (x3fc_flavor == EFlavorType::One && HitShield(-dir)) {
 | |
|     x5e8_shieldTime = 1.f;
 | |
|     return EWeaponCollisionResponseTypes::Unknown86;
 | |
|   }
 | |
|   return EWeaponCollisionResponseTypes::Unknown36;
 | |
| }
 | |
| 
 | |
| void CDrone::DoUserAnimEvent(CStateManager& mgr, const CInt32POINode& node, EUserEventType type, float dt) {
 | |
|   switch (type) {
 | |
|   case EUserEventType::Projectile:
 | |
|     sub_80165984(mgr, GetLctrTransform(node.GetLocatorName()));
 | |
|     return;
 | |
|   case EUserEventType::Delete:
 | |
|     if (x7d0_) {
 | |
|       CSfxManager::RemoveEmitter(x7d0_);
 | |
|       x7d0_.reset();
 | |
|     }
 | |
|     MassiveDeath(mgr);
 | |
|     break;
 | |
|   case EUserEventType::DamageOn: {
 | |
|     if (IsAlive() && x835_24_) {
 | |
|       if (!x824_activeLasers[0]) {
 | |
|         UpdateLaser(mgr, 0, true);
 | |
|         x824_activeLasers[0] = true;
 | |
|         SetVisorFlareEnabled(mgr, true);
 | |
|       } else if (x3fc_flavor == EFlavorType::One) {
 | |
|         UpdateLaser(mgr, 1, true);
 | |
|         x824_activeLasers[1] = true;
 | |
|       }
 | |
|     }
 | |
|     return;
 | |
|   }
 | |
|   case EUserEventType::DamageOff: {
 | |
|     if (x824_activeLasers[0]) {
 | |
|       UpdateLaser(mgr, 0, false);
 | |
|       x824_activeLasers[0] = false;
 | |
|       SetVisorFlareEnabled(mgr, false);
 | |
|     } else if (x3fc_flavor == EFlavorType::One) {
 | |
|       UpdateLaser(mgr, 1, false);
 | |
|       x824_activeLasers[1] = false;
 | |
|     }
 | |
|     return;
 | |
|   }
 | |
|   case EUserEventType::FadeIn: {
 | |
|     if (x3fc_flavor == EFlavorType::One)
 | |
|       x834_30_visible = true;
 | |
|     return;
 | |
|   }
 | |
|   case EUserEventType::FadeOut: {
 | |
|     if (x3fc_flavor == EFlavorType::One)
 | |
|       x834_30_visible = false;
 | |
|     return;
 | |
|   }
 | |
|   default:
 | |
|     break;
 | |
|   }
 | |
|   CPatterned::DoUserAnimEvent(mgr, node, type, dt);
 | |
| }
 | |
| 
 | |
| const CCollisionPrimitive* CDrone::GetCollisionPrimitive() const {
 | |
|   if (!x834_28_)
 | |
|     return &x690_colSphere;
 | |
|   return CPatterned::GetCollisionPrimitive();
 | |
| }
 | |
| 
 | |
| void CDrone::Death(CStateManager& mgr, const zeus::CVector3f& direction, EScriptObjectState state) {
 | |
|   if (!IsAlive())
 | |
|     return;
 | |
| 
 | |
|   x824_activeLasers[0] = false;
 | |
|   x824_activeLasers[1] = false;
 | |
|   UpdateLaser(mgr, 0, false);
 | |
|   UpdateLaser(mgr, 1, false);
 | |
|   SetVisorFlareEnabled(mgr, false);
 | |
| 
 | |
|   if (x3e4_lastHP - HealthInfo(mgr)->GetHP() < x3d8_xDamageThreshold || x834_24_waveHit) {
 | |
|     x330_stateMachineState.SetState(mgr, *this, GetStateMachine(), "Dead"sv);
 | |
|   } else {
 | |
|     x834_28_ = true;
 | |
|     if (x3e0_xDamageDelay <= 0.f) {
 | |
|       SetTransform(zeus::lookAt(GetTranslation(), GetTranslation() - direction) *
 | |
|                    zeus::CTransform::RotateX(zeus::degToRad(45.f)));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (x450_bodyController->GetPercentageFrozen() > 0.f) {
 | |
|     x450_bodyController->UnFreeze();
 | |
|   }
 | |
| 
 | |
|   x400_25_alive = false;
 | |
|   SendScriptMsgs(state, mgr, EScriptObjectMessage::None);
 | |
| }
 | |
| 
 | |
| void CDrone::KnockBack(const zeus::CVector3f& backVec, CStateManager& mgr, const CDamageInfo& info, EKnockBackType type,
 | |
|                        bool inDeferred, float magnitude) {
 | |
|   if (!IsAlive())
 | |
|     return;
 | |
|   CPatterned::KnockBack(backVec, mgr, info, type, inDeferred, magnitude);
 | |
|   if (GetKnockBackController().GetActiveParms().x0_animState == EKnockBackAnimationState::Invalid)
 | |
|     return;
 | |
|   x630_ = 0.5f;
 | |
|   x634_ = 1.f;
 | |
| }
 | |
| 
 | |
| void CDrone::Patrol(CStateManager& mgr, EStateMsg msg, float dt) {
 | |
|   if (msg == EStateMsg::Activate) {
 | |
|     x450_bodyController->SetLocomotionType(pas::ELocomotionType::Lurk);
 | |
|     SetLightEnabled(mgr, true);
 | |
|     x834_25_ = true;
 | |
|   } else if (msg == EStateMsg::Update) {
 | |
|     rstl::reserved_vector<TUniqueId, 1024> nearList;
 | |
|     BuildNearList(EMaterialTypes::Character, EMaterialTypes::Player, nearList, 5.f, mgr);
 | |
|     if (!nearList.empty()) {
 | |
|       zeus::CVector3f sep = x45c_steeringBehaviors.Separation(
 | |
|           *this, static_cast<const CActor*>(mgr.GetObjectById(nearList[0]))->GetTranslation(), 5.f);
 | |
|       if (!sep.isZero()) {
 | |
|         x450_bodyController->GetCommandMgr().DeliverCmd(CBCLocomotionCmd(sep, zeus::skZero3f, 0.5f));
 | |
|       }
 | |
|     }
 | |
|   } else if (msg == EStateMsg::Deactivate) {
 | |
|     SetLightEnabled(mgr, false);
 | |
|     x834_25_ = false;
 | |
|   }
 | |
|   CPatterned::Patrol(mgr, msg, dt);
 | |
| }
 | |
| 
 | |
| void CDrone::PathFind(CStateManager& mgr, EStateMsg msg, float dt) {
 | |
|   if (msg == EStateMsg::Activate) {
 | |
|     auto searchOff = GetDestPos();
 | |
|     CPathFindSearch::EResult res = GetSearchPath()->Search(GetTranslation(), searchOff);
 | |
|     if (res != CPathFindSearch::EResult::Success &&
 | |
|         (res == CPathFindSearch::EResult::NoDestPoint || res == CPathFindSearch::EResult::NoPath)) {
 | |
|       if (GetSearchPath()->FindClosestReachablePoint(GetTranslation(), searchOff) ==
 | |
|           CPathFindSearch::EResult::Success) {
 | |
|         GetSearchPath()->Search(GetTranslation(), searchOff);
 | |
|         SetDestPos(searchOff);
 | |
|       }
 | |
|     }
 | |
|     if (x3fc_flavor == CPatterned::EFlavorType::One) {
 | |
|       x834_30_visible = true;
 | |
|     }
 | |
|   } else if (msg == EStateMsg::Update) {
 | |
|     CPatterned::PathFind(mgr, msg, dt);
 | |
|     x450_bodyController->GetCommandMgr().BlendSteeringCmds();
 | |
|     zeus::CVector3f moveVec = x450_bodyController->GetCommandMgr().GetMoveVector();
 | |
|     if (moveVec.canBeNormalized()) {
 | |
|       moveVec.normalize();
 | |
|       x450_bodyController->GetCommandMgr().ClearLocomotionCmds();
 | |
|       ApplyImpulseWR(GetMass() * (x5e4_ * moveVec), {});
 | |
|       const auto target = (mgr.GetPlayer().GetAimPosition(mgr, 0.f) - GetTranslation()).normalized();
 | |
|       x450_bodyController->GetCommandMgr().DeliverCmd(
 | |
|           CBCLocomotionCmd(FLT_EPSILON * GetTransform().frontVector(), target, 1.f));
 | |
|       x450_bodyController->GetCommandMgr().DeliverTargetVector(target);
 | |
|       StrafeFromCompanions(mgr);
 | |
|       if (x630_ <= 0.f) {
 | |
|         x634_ = 0.333333f;
 | |
|       }
 | |
|     } else if (x630_ <= 0.f) {
 | |
|       x634_ = 0.f;
 | |
|     }
 | |
|   } else if (msg == EStateMsg::Deactivate) {
 | |
|     CPatterned::PathFind(mgr, msg, dt);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CDrone::TargetPlayer(CStateManager& mgr, EStateMsg msg, float dt) {
 | |
|   if (msg == EStateMsg::Activate) {
 | |
|     x3b8_turnSpeed = x5ec_turnSpeed;
 | |
|     x450_bodyController->SetTurnSpeed(x5ec_turnSpeed);
 | |
|     if (x450_bodyController->GetLocomotionType() != pas::ELocomotionType::Combat)
 | |
|       x450_bodyController->SetLocomotionType(pas::ELocomotionType::Combat);
 | |
|     x450_bodyController->BodyStateInfo().SetMaximumPitch(zeus::degToRad(60.f));
 | |
|     SetDestPos(mgr.GetPlayer().GetAimPosition(mgr, 0.f));
 | |
|     x400_24_hitByPlayerProjectile = false;
 | |
|     if (x3fc_flavor == EFlavorType::One)
 | |
|       x834_30_visible = true;
 | |
|     x330_stateMachineState.SetDelay(std::max(0.3f, x624_));
 | |
|   } else if (msg == EStateMsg::Update) {
 | |
|     zeus::CVector3f target = (mgr.GetPlayer().GetAimPosition(mgr, 0.f) - GetTranslation()).normalized();
 | |
|     x450_bodyController->GetCommandMgr().DeliverCmd(
 | |
|         CBCLocomotionCmd(FLT_EPSILON * GetTransform().frontVector(), target, 1.f));
 | |
|     x450_bodyController->GetCommandMgr().DeliverTargetVector(target);
 | |
|     StrafeFromCompanions(mgr);
 | |
|     if (x630_ <= 0.f)
 | |
|       x634_ = 0.f;
 | |
|   } else if (msg == EStateMsg::Deactivate) {
 | |
|     SetDestPos(mgr.GetPlayer().GetTranslation() + zeus::CVector3f{0.f, 0.f, x664_});
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CDrone::TargetCover(CStateManager& mgr, EStateMsg msg, float dt) {
 | |
|   if (msg != EStateMsg::Update) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // Don't ask I have no idea....
 | |
|   const zeus::CVector3f vec{1.f * x5e4_ * 0.f, 1.f * x5e4_ * 0.f, 1.f * x5e4_ * 1.f};
 | |
|   ApplyImpulseWR(GetMoveToORImpulseWR(GetTransform().transposeRotate(vec), 1.f), {});
 | |
| }
 | |
| 
 | |
| void CDrone::Deactivate(CStateManager& mgr, EStateMsg msg, float dt) {
 | |
|   if (msg != EStateMsg::Activate)
 | |
|     return;
 | |
|   DeathDelete(mgr);
 | |
| }
 | |
| 
 | |
| void CDrone::Attack(CStateManager& mgr, EStateMsg msg, float dt) {
 | |
|   if (msg == EStateMsg::Activate) {
 | |
|     x7c8_ = 0;
 | |
|     x834_31_attackOver = false;
 | |
|     const auto playerAimPos = mgr.GetPlayer().GetAimPosition(mgr, 0.f);
 | |
|     const auto& xf = GetTransform();
 | |
|     const auto frontVec = xf.frontVector();
 | |
|     zeus::CVector3f out;
 | |
|     if ((playerAimPos - GetTranslation()).normalized().dot(frontVec) >= 0.8f) {
 | |
|       out = playerAimPos;
 | |
|     } else {
 | |
|       out = GetTranslation() + 10.f * frontVec;
 | |
|     }
 | |
|     s32 state = mgr.GetActiveRandom()->Next() % 4;
 | |
|     if (state == 0) {
 | |
|       x7e0_lasersStart[0] = out + (3.f * xf.rightVector()) - (4.f * xf.upVector());
 | |
|       x7fc_lasersEnd[0] = out - (3.f * xf.rightVector()) + (4.f * xf.upVector());
 | |
|       x7e0_lasersStart[1] = out - (3.f * xf.rightVector()) - (4.f * xf.upVector());
 | |
|       x7fc_lasersEnd[1] = out + (3.f * xf.rightVector()) + (4.f * xf.upVector());
 | |
|     } else if (state == 1) {
 | |
|       x7e0_lasersStart[0] = out + (3.f * xf.rightVector()) + (4.f * xf.upVector());
 | |
|       x7fc_lasersEnd[0] = out - (3.f * xf.rightVector()) - (4.f * xf.upVector());
 | |
|       x7e0_lasersStart[1] = out - (3.f * xf.rightVector()) + (4.f * xf.upVector());
 | |
|       x7fc_lasersEnd[1] = out + (3.f * xf.rightVector()) - (4.f * xf.upVector());
 | |
|     } else if (state == 2) {
 | |
|       x7e0_lasersStart[0] = out - (4.f * xf.rightVector()) - (3.f * xf.upVector());
 | |
|       x7fc_lasersEnd[0] = out + (4.f * xf.rightVector()) + (3.f * xf.upVector());
 | |
|       x7e0_lasersStart[1] = out + (4.f * xf.rightVector()) - (3.f * xf.upVector());
 | |
|       x7fc_lasersEnd[1] = out - (4.f * xf.rightVector()) + (3.f * xf.upVector());
 | |
|     } else if (state == 3) {
 | |
|       x7e0_lasersStart[0] = out - (4.f * xf.rightVector()) + (3.f * xf.upVector());
 | |
|       x7fc_lasersEnd[0] = out + (4.f * xf.rightVector()) - (3.f * xf.upVector());
 | |
|       x7e0_lasersStart[1] = out + (4.f * xf.rightVector()) + (3.f * xf.upVector());
 | |
|       x7fc_lasersEnd[1] = out - (4.f * xf.rightVector()) - (3.f * xf.upVector());
 | |
|     }
 | |
|     x818_lasersTime[0] = 0.f;
 | |
|     x818_lasersTime[1] = 0.f;
 | |
|     x835_24_ = true;
 | |
|   } else if (msg == EStateMsg::Update) {
 | |
|     if (x7c8_ == 0) {
 | |
|       if (GetBodyController()->GetCurrentStateId() == pas::EAnimationState::ProjectileAttack) {
 | |
|         x7c8_ = 1;
 | |
|       } else {
 | |
|         GetBodyController()->GetCommandMgr().DeliverCmd(
 | |
|             CBCProjectileAttackCmd(pas::ESeverity::Two, mgr.GetPlayer().GetTranslation(), false));
 | |
|       }
 | |
|     } else if (x7c8_ == 1) {
 | |
|       if (GetBodyController()->GetCurrentStateId() != pas::EAnimationState::ProjectileAttack) {
 | |
|         x7c8_ = 2;
 | |
|       }
 | |
|     }
 | |
|     if (x630_ <= 0.f) {
 | |
|       x634_ = 0.f;
 | |
|     }
 | |
|   } else if (msg == EStateMsg::Deactivate) {
 | |
|     x824_activeLasers[0] = false;
 | |
|     x824_activeLasers[1] = false;
 | |
|     UpdateLaser(mgr, 0, false);
 | |
|     UpdateLaser(mgr, 1, false);
 | |
|     SetVisorFlareEnabled(mgr, false);
 | |
|     x5d0_ = x5f4_;
 | |
|     x835_24_ = false;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CDrone::Active(CStateManager& mgr, EStateMsg msg, float dt) {
 | |
|   if (msg == EStateMsg::Activate) {
 | |
|     x330_stateMachineState.SetDelay(x5f0_);
 | |
|     GetBodyController()->SetLocomotionType(pas::ELocomotionType::Relaxed);
 | |
|   } else if (msg == EStateMsg::Deactivate) {
 | |
|     x5d0_ = x5f8_;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CDrone::Flee(CStateManager& mgr, EStateMsg msg, float dt) {
 | |
|   if (msg == EStateMsg::Activate) {
 | |
|     x7c8_ = 0;
 | |
|     x832_b = 0;
 | |
|     if (mgr.RayStaticIntersection(GetTranslation(), -GetTransform().frontVector(), 4.f,
 | |
|                                   CMaterialFilter::MakeInclude({EMaterialTypes::Solid}))
 | |
|             .IsValid()) {
 | |
|       x832_b = mgr.GetActiveRandom()->Float() >= 0.5f ? 1 : 2;
 | |
|     }
 | |
|   } else if (msg == EStateMsg::Update) {
 | |
|     if (x7c8_ == 0) {
 | |
|       if (GetBodyController()->GetBodyStateInfo().GetCurrentStateId() == pas::EAnimationState::Step) {
 | |
|         x7c8_ = 1;
 | |
|       } else if (x832_b == 0) {
 | |
|         GetBodyController()->GetCommandMgr().DeliverCmd(
 | |
|             CBCStepCmd(pas::EStepDirection::Backward, pas::EStepType::BreakDodge));
 | |
|       } else if (x832_b == 1) {
 | |
|         GetBodyController()->GetCommandMgr().DeliverCmd(CBCStepCmd(pas::EStepDirection::Left, pas::EStepType::Normal));
 | |
|       } else if (x832_b == 2) {
 | |
|         GetBodyController()->GetCommandMgr().DeliverCmd(CBCStepCmd(pas::EStepDirection::Right, pas::EStepType::Normal));
 | |
|       }
 | |
|     } else if (x7c8_ == 1 &&
 | |
|                GetBodyController()->GetBodyStateInfo().GetCurrentStateId() != pas::EAnimationState::Step) {
 | |
|       x7c8_ = 2;
 | |
|     }
 | |
|     GetBodyController()->GetCommandMgr().DeliverTargetVector(
 | |
|         (mgr.GetPlayer().GetTranslation() - GetTranslation()).normalized());
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CDrone::ProjectileAttack(CStateManager& mgr, EStateMsg msg, float dt) {
 | |
|   // TODO: Finish
 | |
| }
 | |
| 
 | |
| void CDrone::TelegraphAttack(CStateManager& mgr, EStateMsg msg, float dt) {
 | |
|   if (msg == EStateMsg::Activate) {
 | |
|     x7c8_ = 0;
 | |
|   } else if (msg == EStateMsg::Update) {
 | |
|     if (x7c8_ == 1 && x450_bodyController->GetBodyStateInfo().GetCurrentStateId() != pas::EAnimationState::Taunt) {
 | |
|       x7c8_ = 2;
 | |
|     } else if (x7c8_ == 0) {
 | |
|       if (x450_bodyController->GetBodyStateInfo().GetCurrentStateId() == pas::EAnimationState::Taunt) {
 | |
|         x7c8_ = 1;
 | |
|       } else {
 | |
|         x450_bodyController->GetCommandMgr().DeliverCmd(CBCTauntCmd(pas::ETauntType::One));
 | |
|       }
 | |
|     }
 | |
|   } else if (msg == EStateMsg::Deactivate) {
 | |
|     SendScriptMsgs(EScriptObjectState::Zero, mgr, EScriptObjectMessage::None);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CDrone::Dodge(CStateManager& mgr, EStateMsg msg, float dt) {
 | |
|   if (msg == EStateMsg::Activate) {
 | |
|     x7c8_ = 0;
 | |
|     x630_ = 0.5f;
 | |
|     x634_ = 1.f;
 | |
|     if (x3fc_flavor == EFlavorType::One)
 | |
|       x834_30_visible = true;
 | |
|   } else if (msg == EStateMsg::Update) {
 | |
|     if (x7c8_ == 0) {
 | |
|       GetBodyController()->GetCommandMgr().DeliverCmd(CBodyStateCmd(EBodyStateCmd::NextState));
 | |
|       if (x58c_prevDodgeDir == pas::EStepDirection::Down) {
 | |
|         GetBodyController()->GetCommandMgr().DeliverCmd(CBCStepCmd(pas::EStepDirection::Left, pas::EStepType::Dodge));
 | |
|         x58c_prevDodgeDir = pas::EStepDirection::Left;
 | |
|       } else if (x58c_prevDodgeDir == pas::EStepDirection::Up) {
 | |
|         GetBodyController()->GetCommandMgr().DeliverCmd(CBCStepCmd(pas::EStepDirection::Down, pas::EStepType::Dodge));
 | |
|         x58c_prevDodgeDir = pas::EStepDirection::Down;
 | |
|       } else if (x58c_prevDodgeDir == pas::EStepDirection::Right) {
 | |
|         GetBodyController()->GetCommandMgr().DeliverCmd(CBCStepCmd(pas::EStepDirection::Up, pas::EStepType::Dodge));
 | |
|         x58c_prevDodgeDir = pas::EStepDirection::Up;
 | |
|       } else if (x58c_prevDodgeDir == pas::EStepDirection::Left) {
 | |
|         GetBodyController()->GetCommandMgr().DeliverCmd(CBCStepCmd(pas::EStepDirection::Right, pas::EStepType::Dodge));
 | |
|         x58c_prevDodgeDir = pas::EStepDirection::Right;
 | |
|       }
 | |
|       x7c8_ = 1;
 | |
|     } else if (x7c8_ == 1 &&
 | |
|                GetBodyController()->GetBodyStateInfo().GetCurrentStateId() != pas::EAnimationState::Step) {
 | |
|       x7c8_ = 2;
 | |
|     }
 | |
|     GetBodyController()->GetCommandMgr().DeliverTargetVector(
 | |
|         (mgr.GetPlayer().GetTranslation() + zeus::CVector3f{0.f, 0.f, 1.f}) - GetTranslation());
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CDrone::Retreat(CStateManager& mgr, EStateMsg msg, float dt) {
 | |
|   if (msg == EStateMsg::Activate) {
 | |
|     x7c8_ = 0;
 | |
|     if (x3fc_flavor == EFlavorType::One) {
 | |
|       x834_30_visible = true;
 | |
|     }
 | |
|     x330_stateMachineState.SetDelay(x65c_);
 | |
|   } else if (msg == EStateMsg::Update) {
 | |
|     if (x7c8_ == 0) {
 | |
|       if (GetBodyController()->GetBodyStateInfo().GetCurrentStateId() == pas::EAnimationState::Step) {
 | |
|         x7c8_ = 1;
 | |
|       } else {
 | |
|         GetBodyController()->GetCommandMgr().DeliverCmd(
 | |
|             CBCStepCmd(pas::EStepDirection::Backward, pas::EStepType::Normal));
 | |
|       }
 | |
|     } else if (x7c8_ == 1 &&
 | |
|                GetBodyController()->GetBodyStateInfo().GetCurrentStateId() != pas::EAnimationState::Step) {
 | |
|       x7c8_ = 2;
 | |
|     } else if (x7c8_ == 2) {
 | |
|       x7c8_ = 0;
 | |
|     }
 | |
| 
 | |
|     GetBodyController()->GetCommandMgr().DeliverTargetVector(
 | |
|         (mgr.GetPlayer().GetTranslation() - GetTranslation()).normalized());
 | |
|     if (x630_ <= 0.f) {
 | |
|       x634_ = 0.333333; // 1/3
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CDrone::Cover(CStateManager& mgr, EStateMsg msg, float dt) {
 | |
|   if (msg == EStateMsg::Activate) {
 | |
|     x67c_ = zeus::skZero3f;
 | |
|     x670_ = GetTranslation();
 | |
|     for (int i = 0; i < 4; ++i) {
 | |
|       float dVar11 = (x64c_ - x648_) * mgr.GetActiveRandom()->Float() + x648_;
 | |
|       int v = mgr.GetActiveRandom()->Next();
 | |
|       float angle = 0.f;
 | |
|       if (((v >> 3) & 1) == 0) {
 | |
|         const float angleMin = 270.f - x654_;
 | |
|         const float angleMax = 270.f + x650_;
 | |
|         angle = zeus::degToRad((angleMax - angleMin) * x648_ + angleMin);
 | |
|       } else {
 | |
|         const float angleMin = 90.f - x654_;
 | |
|         const float angleMax = 90.f + x650_;
 | |
|         angle = zeus::degToRad((angleMax - angleMin) * x648_ + angleMin);
 | |
|       }
 | |
|       zeus::CQuaternion quat;
 | |
|       quat.rotateZ(angle);
 | |
|       const zeus::CVector3f end =
 | |
|           GetTranslation() +
 | |
|           quat.transform((dVar11 * (mgr.GetPlayer().GetAimPosition(mgr, 0.f) - GetTranslation()).normalized()));
 | |
|       if (mgr.RayCollideWorld(GetTranslation(), end, CMaterialFilter::MakeInclude({EMaterialTypes::Solid}), this)) {
 | |
|         x670_ = end;
 | |
|         x67c_ = end - GetTranslation();
 | |
|         if (x67c_.canBeNormalized())
 | |
|           x67c_.normalize();
 | |
|       }
 | |
|     }
 | |
|   } else if (msg == EStateMsg::Update) {
 | |
|     ApplyImpulseWR(GetMoveToORImpulseWR(GetTransform().transposeRotate(dt * (x658_ * x67c_)), dt), zeus::CAxisAngle());
 | |
|     x450_bodyController->GetCommandMgr().DeliverCmd(
 | |
|         CBCLocomotionCmd(FLT_EPSILON * GetTransform().basis[1],
 | |
|                          (mgr.GetPlayer().GetAimPosition(mgr, 0.f) - GetTranslation()).normalized(), 1.f));
 | |
|   } else if (msg == EStateMsg::Deactivate) {
 | |
|     x644_ = (x640_ - x63c_) * mgr.GetActiveRandom()->Float() + x63c_;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CDrone::SpecialAttack(CStateManager& mgr, EStateMsg msg, float dt) {
 | |
|   if (msg == EStateMsg::Activate) {
 | |
|     if (x3fc_flavor == EFlavorType::One) {
 | |
|       x834_30_visible = true;
 | |
|     }
 | |
|     x330_stateMachineState.SetDelay(x660_);
 | |
|     GetBodyController()->SetLocomotionType(pas::ELocomotionType::Internal10);
 | |
|   } else if (msg == EStateMsg::Update) {
 | |
|     GetBodyController()->GetCommandMgr().DeliverCmd(
 | |
|         CBCLocomotionCmd(GetTransform().frontVector(), zeus::skZero3f, 1.f));
 | |
|     zeus::CVector3f local_74 =
 | |
|         0.5f * (mgr.GetPlayer().GetAimPosition(mgr, 0.f) + mgr.GetPlayer().GetTranslation()) - GetTranslation();
 | |
|     if (((x668_elevation < x664_ && local_74.z() > 0.f) || (x668_elevation > x664_)) && local_74.canBeNormalized()) {
 | |
|       ApplyImpulseWR(GetMoveToORImpulseWR(GetTransform().transposeRotate(dt * (x5e4_ * local_74.normalized())), dt),
 | |
|                      zeus::CAxisAngle());
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CDrone::PathFindEx(CStateManager& mgr, EStateMsg msg, float dt) {
 | |
|   CPatterned::PathFind(mgr, msg, dt);
 | |
|   if (msg == EStateMsg::Activate) {
 | |
|     auto searchOff = mgr.GetPlayer().GetTranslation() + zeus::CVector3f{0.f, 0.f, x664_};
 | |
|     CPathFindSearch::EResult res = GetSearchPath()->Search(GetTranslation(), searchOff);
 | |
|     if (res != CPathFindSearch::EResult::Success &&
 | |
|         (res == CPathFindSearch::EResult::NoDestPoint || res == CPathFindSearch::EResult::NoPath)) {
 | |
|       if (GetSearchPath()->FindClosestReachablePoint(GetTranslation(), searchOff) ==
 | |
|           CPathFindSearch::EResult::Success) {
 | |
|         GetSearchPath()->Search(GetTranslation(), searchOff);
 | |
|         SetDestPos(searchOff);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| bool CDrone::Leash(CStateManager& mgr, float arg) {
 | |
|   return (mgr.GetPlayer().GetTranslation() - GetTranslation()).magSquared() < x3c8_leashRadius * x3c8_leashRadius;
 | |
| }
 | |
| 
 | |
| bool CDrone::InRange(CStateManager& mgr, float arg) {
 | |
|   float mag = (mgr.GetPlayer().GetTranslation() - GetTranslation()).magSquared();
 | |
|   return mag > x2fc_minAttackRange * x2fc_minAttackRange && mag < x300_maxAttackRange * x300_maxAttackRange;
 | |
| }
 | |
| 
 | |
| bool CDrone::SpotPlayer(CStateManager& mgr, float arg) {
 | |
|   if ((mgr.GetPlayer().GetTranslation() - GetTranslation()).magSquared() > x3bc_detectionRange * x3bc_detectionRange)
 | |
|     return false;
 | |
| 
 | |
|   if (!LineOfSight(mgr, arg))
 | |
|     return false;
 | |
| 
 | |
|   return (GetTransform().frontVector() + x5cc_ * GetTransform().rightVector())
 | |
|              .normalized()
 | |
|              .dot((mgr.GetPlayer().GetAimPosition(mgr, 0.f) - GetTranslation()).normalized()) > 0.5f;
 | |
| }
 | |
| 
 | |
| bool CDrone::AnimOver(CStateManager& mgr, float arg) { return x7c8_ == 2; }
 | |
| 
 | |
| bool CDrone::AttackOver(CStateManager& mgr, float arg) { return x834_31_attackOver; }
 | |
| 
 | |
| bool CDrone::ShouldAttack(CStateManager& mgr, float arg) {
 | |
|   if (x5d0_ > 0.f)
 | |
|     return false;
 | |
|   if (TCastToPtr<CTeamAiMgr> teamMgr = mgr.ObjectById(x688_teamMgr)) {
 | |
|     if (teamMgr->HasTeamAiRole(GetUniqueId()))
 | |
|       return teamMgr->AddRangedAttacker(GetUniqueId());
 | |
|   }
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| bool CDrone::ShouldFire(CStateManager& mgr, float arg) {
 | |
|   if (mgr.GetPlayer().GetMorphballTransitionState() == CPlayer::EPlayerMorphBallState::Morphed || x624_ > 0.f) {
 | |
|     return false;
 | |
|   }
 | |
|   const zeus::CVector3f playerAimPos = mgr.GetPlayer().GetAimPosition(mgr, 0.f);
 | |
|   constexpr auto matFilter =
 | |
|       CMaterialFilter::MakeIncludeExclude({EMaterialTypes::Solid, EMaterialTypes::Character},
 | |
|                                           {EMaterialTypes::Player, EMaterialTypes::ProjectilePassthrough});
 | |
|   bool result = mgr.RayCollideWorld(GetLctrTransform("R_GUN_TOP_LCTR"sv).origin, playerAimPos, matFilter, this);
 | |
|   if (!result) {
 | |
|     return false;
 | |
|   }
 | |
|   return mgr.RayCollideWorld(GetLctrTransform("L_GUN_TOP_LCTR"sv).origin, playerAimPos, matFilter, this);
 | |
| }
 | |
| 
 | |
| bool CDrone::HearShot(CStateManager& mgr, float arg) {
 | |
|   rstl::reserved_vector<TUniqueId, 1024> nearList;
 | |
|   BuildNearList(EMaterialTypes::Projectile, EMaterialTypes::Player, nearList, 10.f, mgr);
 | |
|   return std::any_of(nearList.begin(), nearList.end(), [&mgr](TUniqueId uid) {
 | |
|     if (TCastToConstPtr<CWeapon> wp = mgr.GetObjectById(uid))
 | |
|       return wp->GetType() != EWeaponType::AI;
 | |
|     return false;
 | |
|   });
 | |
| }
 | |
| 
 | |
| bool CDrone::CoverCheck(CStateManager& mgr, float arg) {
 | |
|   if (!zeus::close_enough(x67c_, zeus::skZero3f)) {
 | |
|     const zeus::CVector3f diff = x670_ - GetTranslation();
 | |
|     return x67c_.dot(diff) < 0.0f || diff.magSquared() < 0.25f;
 | |
|   }
 | |
| 
 | |
|   return true;
 | |
| }
 | |
| bool CDrone::LineOfSight(CStateManager& mgr, float arg) {
 | |
|   return mgr.RayCollideWorld(
 | |
|       GetTranslation(), mgr.GetPlayer().GetAimPosition(mgr, 0.f),
 | |
|       CMaterialFilter::MakeIncludeExclude({EMaterialTypes::Solid, EMaterialTypes::Character},
 | |
|                                           {EMaterialTypes::Player, EMaterialTypes::ProjectilePassthrough}),
 | |
|       this);
 | |
| }
 | |
| 
 | |
| bool CDrone::ShouldMove(CStateManager& mgr, float arg) { return x644_ <= 0.f; }
 | |
| 
 | |
| bool CDrone::CodeTrigger(CStateManager& mgr, float arg) { return x834_29_codeTrigger; }
 | |
| 
 | |
| void CDrone::Burn(float duration, float damage) {
 | |
|   // Intentionally empty
 | |
| }
 | |
| 
 | |
| CPathFindSearch* CDrone::GetSearchPath() { return &x6b0_pathFind; }
 | |
| 
 | |
| void CDrone::BuildNearList(EMaterialTypes includeMat, EMaterialTypes excludeMat,
 | |
|                            rstl::reserved_vector<TUniqueId, 1024>& listOut, float radius, CStateManager& mgr) {
 | |
|   const zeus::CVector3f pos = GetTranslation();
 | |
|   mgr.BuildNearList(listOut, zeus::CAABox(pos - radius, pos + radius),
 | |
|                     CMaterialFilter::MakeIncludeExclude({includeMat}, {excludeMat}), nullptr);
 | |
| }
 | |
| 
 | |
| void CDrone::SetLightEnabled(CStateManager& mgr, bool activate) {
 | |
|   mgr.SendScriptMsgAlways(x578_lightId, GetUniqueId(),
 | |
|                           activate ? EScriptObjectMessage::Activate : EScriptObjectMessage::Deactivate);
 | |
| }
 | |
| 
 | |
| void CDrone::SetVisorFlareEnabled(CStateManager& mgr, bool activate) {
 | |
|   if (!IsAlive()) {
 | |
|     return;
 | |
|   }
 | |
|   CScriptVisorFlare* flare = TCastToPtr<CScriptVisorFlare>{mgr.ObjectById(x57a_visorFlareId)};
 | |
|   if (flare == nullptr && activate) {
 | |
|     x57a_visorFlareId = mgr.AllocateUniqueId();
 | |
|     flare = new CScriptVisorFlare(x57a_visorFlareId, "DroneVisorFlare"sv,
 | |
|                                   CEntityInfo{GetAreaIdAlways(), CEntity::NullConnectionList}, activate,
 | |
|                                   GetLctrTransform("Beacon_LCTR"sv).origin, CVisorFlare::EBlendMode::Zero, true, 0.1f,
 | |
|                                   1.f, 2.f, 0, 0, x57c_flares);
 | |
|     mgr.AddObject(flare);
 | |
|   }
 | |
|   mgr.SendScriptMsg(flare, GetUniqueId(), activate ? EScriptObjectMessage::Activate : EScriptObjectMessage::Deactivate);
 | |
| }
 | |
| 
 | |
| void CDrone::UpdateVisorFlare(CStateManager& mgr) {
 | |
|   TCastToPtr<CScriptVisorFlare> flare = mgr.ObjectById(x57a_visorFlareId);
 | |
|   SetVisorFlareEnabled(
 | |
|       mgr, (mgr.GetPlayer().GetTranslation() - GetTranslation()).normalized().dot(GetTransform().frontVector()) > 0.f);
 | |
|   if (flare) {
 | |
|     const auto beaconXf = GetLctrTransform("Beacon_LCTR"sv);
 | |
|     flare->SetTranslation(beaconXf.origin + (0.1f * beaconXf.frontVector()));
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CDrone::UpdateTouchBounds(float radius) {
 | |
|   const zeus::CTransform xf = GetLctrTransform("Skeleton_Root"sv);
 | |
|   const zeus::CVector3f diff = xf.origin - GetTranslation();
 | |
|   x690_colSphere.SetSphereCenter(diff);
 | |
|   x690_colSphere.SetSphereRadius(radius);
 | |
|   SetBoundingBox(zeus::CAABox{diff - radius, diff + radius});
 | |
|   x6b0_pathFind.SetCharacterRadius(0.25f + radius);
 | |
| }
 | |
| 
 | |
| bool CDrone::HitShield(const zeus::CVector3f& dir) const {
 | |
|   if (x3fc_flavor == EFlavorType::One && !zeus::close_enough(x5dc_, 0.f)) {
 | |
|     return GetLctrTransform("Shield_LCTR"sv).frontVector().dot(dir.normalized()) > 0.85f;
 | |
|   }
 | |
| 
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| void CDrone::AddToTeam(CStateManager& mgr) const {
 | |
|   if (x688_teamMgr == kInvalidUniqueId) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (TCastToPtr<CTeamAiMgr> teamMgr = mgr.ObjectById(x688_teamMgr)) {
 | |
|     teamMgr->AssignTeamAiRole(*this, CTeamAiRole::ETeamAiRole::Ranged, CTeamAiRole::ETeamAiRole::Melee,
 | |
|                               CTeamAiRole::ETeamAiRole::Invalid);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CDrone::RemoveFromTeam(CStateManager& mgr) const {
 | |
|   if (TCastToPtr<CTeamAiMgr> teamMgr = mgr.ObjectById(x688_teamMgr)) {
 | |
|     if (teamMgr->IsPartOfTeam(GetUniqueId())) {
 | |
|       teamMgr->RemoveTeamAiRole(GetUniqueId());
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CDrone::UpdateLaser(CStateManager& mgr, u32 laserIdx, bool active) {
 | |
|   if (active && x7d8_laserIds[laserIdx] == kInvalidUniqueId) {
 | |
|     x7d8_laserIds[laserIdx] = mgr.AllocateUniqueId();
 | |
|     mgr.AddObject(new CDroneLaser(x7d8_laserIds[laserIdx], GetAreaIdAlways(), GetTransform(), x568_laserParticlesId));
 | |
|   }
 | |
|   if (CEntity* ent = mgr.ObjectById(x7d8_laserIds[laserIdx])) {
 | |
|     mgr.SendScriptMsg(ent, GetUniqueId(), active ? EScriptObjectMessage::Activate : EScriptObjectMessage::Deactivate);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CDrone::FireProjectile(CStateManager& mgr, const zeus::CTransform& xf, const TToken<CWeaponDescription>& weapon) {
 | |
|   // TODO implement
 | |
| }
 | |
| 
 | |
| void CDrone::StrafeFromCompanions(CStateManager& mgr) {
 | |
|   if (x450_bodyController->GetBodyStateInfo().GetCurrentStateId() == pas::EAnimationState::Step)
 | |
|     return;
 | |
|   rstl::reserved_vector<TUniqueId, 1024> nearList;
 | |
|   BuildNearList(EMaterialTypes::Character, EMaterialTypes::Player, nearList, x61c_, mgr);
 | |
|   if (nearList.empty())
 | |
|     return;
 | |
| 
 | |
|   float minDist = FLT_MAX;
 | |
|   zeus::CVector3f nearestPos;
 | |
|   for (TUniqueId uid : nearList) {
 | |
|     if (const CActor* act = static_cast<const CActor*>(mgr.GetObjectById(uid))) {
 | |
|       const float dist = (act->GetTranslation() - GetTranslation()).magSquared();
 | |
|       if (uid != GetUniqueId() && dist < minDist) {
 | |
|         minDist = dist;
 | |
|         nearestPos = act->GetTranslation();
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (nearestPos.isZero() || minDist > x61c_ * x61c_)
 | |
|     return;
 | |
| 
 | |
|   const auto off = nearestPos - GetTranslation();
 | |
|   const float rightOff = GetTransform().rightVector().dot(off);
 | |
|   if (rightOff < -0.2f) {
 | |
|     x450_bodyController->GetCommandMgr().DeliverCmd(CBCStepCmd(pas::EStepDirection::Right, pas::EStepType::Normal));
 | |
|   } else if (rightOff > 0.2f) {
 | |
|     x450_bodyController->GetCommandMgr().DeliverCmd(CBCStepCmd(pas::EStepDirection::Left, pas::EStepType::Normal));
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CDrone::UpdateScanner(CStateManager& mgr, float dt) {
 | |
|   constexpr float deg360 = zeus::degToRad(360.f);
 | |
|   x5d4_ = 1.2f * dt + x5d4_;
 | |
|   if (x5d4_ > deg360) {
 | |
|     x5d4_ -= deg360;
 | |
|   }
 | |
|   if (x5d4_ < 0.f) {
 | |
|     x5d4_ = 0.f;
 | |
|   }
 | |
|   if (x5d8_ > deg360) {
 | |
|     x5d8_ -= deg360;
 | |
|   }
 | |
|   if (x5d8_ < 0.f) {
 | |
|     x5d8_ = 0.f;
 | |
|   }
 | |
|   float angle = zeus::clamp(0.f, 0.5f * (1.f + std::sin(x5d4_)), 1.f);
 | |
|   if (std::fpclassify(angle) != FP_SUBNORMAL)
 | |
|     x5d8_ += 0.03f * std::pow(angle, 5.f);
 | |
|   zeus::CVector3f vec =
 | |
|       GetTransform().rotate(zeus::CVector3f(0.5f * std::cos(x5d8_), 1.f, 0.5f * std::sin(2.05f * x5d8_)).normalized());
 | |
|   TUniqueId id;
 | |
|   rstl::reserved_vector<TUniqueId, 1024> nearList;
 | |
|   nearList.push_back(mgr.GetPlayer().GetUniqueId());
 | |
|   auto res = mgr.RayWorldIntersection(
 | |
|       id, GetLctrTransform("Beacon_LCTR"sv).origin + (0.2f * vec), vec, 10000.f,
 | |
|       CMaterialFilter::MakeIncludeExclude({EMaterialTypes::Solid}, {EMaterialTypes::ProjectilePassthrough}), nearList);
 | |
|   if (res.IsValid() && x578_lightId != kInvalidUniqueId) {
 | |
|     if (TCastToPtr<CGameLight> light = mgr.ObjectById(x578_lightId)) {
 | |
|       light->SetTranslation(res.GetPoint());
 | |
|       x7ac_lightPos = res.GetPoint();
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CDrone::UpdateLasers(CStateManager& mgr, float dt) {
 | |
|   constexpr auto matFilter =
 | |
|       CMaterialFilter::MakeIncludeExclude({EMaterialTypes::Solid}, {EMaterialTypes::ProjectilePassthrough});
 | |
|   const auto beaconXf = GetLctrTransform("Beacon_LCTR"sv);
 | |
|   for (size_t i = 0; i < x818_lasersTime.size(); ++i) {
 | |
|     if (x818_lasersTime[i] >= 1.f || !x824_activeLasers[i]) {
 | |
|       continue;
 | |
|     }
 | |
|     x818_lasersTime[i] += dt;
 | |
|     const auto vec =
 | |
|         (x7e0_lasersStart[i] * (1.f - x818_lasersTime[i]) + (x7fc_lasersEnd[i] * x818_lasersTime[i]) - beaconXf.origin)
 | |
|             .normalized();
 | |
|     auto box = zeus::skInvertedBox;
 | |
|     box.accumulateBounds(GetTranslation() + 1000.f * vec);
 | |
|     box.accumulateBounds(GetTranslation());
 | |
|     rstl::reserved_vector<TUniqueId, 1024> nearList;
 | |
|     mgr.BuildNearList(nearList, box, matFilter, nullptr);
 | |
|     TUniqueId id;
 | |
|     const auto result = mgr.RayWorldIntersection(id, beaconXf.origin + 2.f * vec, vec, 10000.f, matFilter, nearList);
 | |
|     if (result.IsInvalid()) {
 | |
|       continue;
 | |
|     }
 | |
|     if (x7d8_laserIds[i] != kInvalidUniqueId) {
 | |
|       if (auto* laser = static_cast<CDroneLaser*>(mgr.ObjectById(x7d8_laserIds[i]))) {
 | |
|         laser->SetTransform(beaconXf);
 | |
|         laser->sub_80167754(mgr, result.GetPoint(), result.GetPlane().normal());
 | |
|       }
 | |
|     }
 | |
|     if (TCastToPtr<CPlayer> player = mgr.ObjectById(id)) {
 | |
|       if (x420_curDamageRemTime <= 0.f) {
 | |
|         mgr.ApplyDamage(GetUniqueId(), player->GetUniqueId(), GetUniqueId(), GetContactDamage(),
 | |
|                         CMaterialFilter::MakeIncludeExclude({EMaterialTypes::Solid}, {}), zeus::skZero3f);
 | |
|         x420_curDamageRemTime = x424_damageWaitTime;
 | |
|         mgr.GetPlayerState()->GetStaticInterference().AddSource(GetUniqueId(), 0.3f, 1.f);
 | |
|         CSfxManager::AddEmitter(x7cc_laserSfx, result.GetPoint(), zeus::skZero3f, true, false, 127, GetAreaIdAlways());
 | |
|       }
 | |
|     }
 | |
|     if (id != GetUniqueId() && TCastToPtr<CAi>{mgr.ObjectById(id)}) {
 | |
|       x834_31_attackOver = true;
 | |
|       float rem = GetModelData()->GetAnimationData()->GetAnimTimeRemaining("Whole Body"sv);
 | |
|       UpdateAnimation(rem, mgr, true);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CDrone::sub_801633a8(CStateManager& mgr) {
 | |
|   // TODO implement
 | |
| }
 | |
| 
 | |
| void CDrone::sub_8015f25c(float dt, CStateManager& mgr) {
 | |
|   // TODO implement
 | |
| }
 | |
| 
 | |
| void CDrone::sub_8015f158(float dt) {
 | |
|   // TODO implement
 | |
| }
 | |
| 
 | |
| void CDrone::sub_80165984(CStateManager& mgr, const zeus::CTransform& xf) {
 | |
|   /*constexpr*/ float sin60 = std::sqrt(3.f) / 2.f;
 | |
|   const auto playerAimPos = mgr.GetPlayer().GetAimPosition(mgr, 0.f);
 | |
|   const auto distNorm = (playerAimPos - xf.origin).normalized();
 | |
|   if (distNorm.dot(xf.frontVector()) <= sin60) {
 | |
|     sub_801656d4(xf, mgr);
 | |
|   } else {
 | |
|     zeus::CVector3f vec;
 | |
|     if (mgr.GetActiveRandom()->Float() > 0.2f) {
 | |
|       const auto lookAt = zeus::lookAt(xf.origin, playerAimPos);
 | |
|       vec = zeus::CQuaternion::fromAxisAngle(lookAt.frontVector(), mgr.GetActiveRandom()->Range(0.f, M_PIF))
 | |
|                 .transform(4.f * lookAt.rightVector());
 | |
|     }
 | |
|     sub_801656d4(zeus::lookAt(xf.origin, playerAimPos + vec), mgr);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CDrone::sub_801656d4(const zeus::CTransform& xf, CStateManager& mgr) {
 | |
|   constexpr auto matFilter =
 | |
|       CMaterialFilter::MakeIncludeExclude({EMaterialTypes::Solid}, {EMaterialTypes::ProjectilePassthrough});
 | |
|   rstl::reserved_vector<TUniqueId, 1024> nearList;
 | |
|   mgr.BuildNearList(nearList, xf.origin, xf.frontVector(), 100000.f, matFilter, this);
 | |
|   TUniqueId id;
 | |
|   const auto result = mgr.RayWorldIntersection(id, xf.origin, xf.frontVector(), 100000.f, matFilter, nearList);
 | |
|   if (result.IsInvalid()) {
 | |
|     return;
 | |
|   }
 | |
|   if (id == mgr.GetPlayer().GetUniqueId()) {
 | |
|     mgr.ApplyDamage(GetUniqueId(), id, GetUniqueId(), x5ac_,
 | |
|                     CMaterialFilter::MakeIncludeExclude({EMaterialTypes::Solid}, {}), zeus::skZero3f);
 | |
|   }
 | |
|   mgr.sub_80044098(*x56c_.GetObj(), result, id, x5ac_.GetWeaponMode(), 1, xe6_27_thermalVisorFlags);
 | |
| }
 | |
| 
 | |
| void CDrone::Dead(CStateManager& mgr, EStateMsg msg, float arg) {
 | |
|   if (msg == EStateMsg::Activate) {
 | |
|     x460_knockBackController.SetAutoResetImpulse(false);
 | |
|     if (x834_24_waveHit) {
 | |
|       SetMomentumWR({0.f, 0.f, -GetWeight()});
 | |
|     } else {
 | |
|       Stop();
 | |
|       SetVelocityWR(zeus::skZero3f);
 | |
|       SetMomentumWR(zeus::skZero3f);
 | |
|     }
 | |
|     x401_26_disableMove = true;
 | |
|     x5c8_ = 0.f;
 | |
|     SetVisorFlareEnabled(mgr, false);
 | |
|     x7c8_ = 0;
 | |
|   } else if (msg == EStateMsg::Update && x7c0_ == 0) {
 | |
|     if (x834_24_waveHit) {
 | |
|       GetBodyController()->GetCommandMgr().DeliverCmd(CBCHurledCmd());
 | |
|       x7c8_ = 1;
 | |
|     } else {
 | |
|       GetBodyController()->GetCommandMgr().DeliverCmd(CBCKnockDownCmd(zeus::skZero3f, pas::ESeverity::Zero));
 | |
|       x7c8_ = 1;
 | |
|       Stop();
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| } // namespace urde::MP1
 |