mirror of https://github.com/AxioDL/metaforce.git
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 "Audio/SFX/Drones.h"
|
|
|
|
#include "TCastTo.hpp" // Generated file, do not modify include path
|
|
|
|
#include <algorithm>
|
|
|
|
namespace metaforce::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);
|
|
EntityList 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) {
|
|
EntityList 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) {
|
|
EntityList 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, EntityList& 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::Additive, 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;
|
|
EntityList 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;
|
|
EntityList 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());
|
|
EntityList 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});
|
|
EntityList 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 metaforce::MP1
|