mirror of
https://github.com/AxioDL/metaforce.git
synced 2025-12-15 12:46:08 +00:00
CGrenadeLauncher implementation
This commit is contained in:
@@ -5,6 +5,9 @@
|
||||
#include "Runtime/CSimplePool.hpp"
|
||||
#include "Runtime/CStateManager.hpp"
|
||||
#include "Runtime/GameGlobalObjects.hpp"
|
||||
#include "Runtime/Weapon/CGameProjectile.hpp"
|
||||
#include "Runtime/World/CExplosion.hpp"
|
||||
#include "Runtime/World/CPatterned.hpp"
|
||||
#include "Runtime/World/CPlayer.hpp"
|
||||
|
||||
namespace urde {
|
||||
@@ -12,19 +15,21 @@ namespace MP1 {
|
||||
CGrenadeLauncher::CGrenadeLauncher(TUniqueId uid, std::string_view name, const CEntityInfo& info,
|
||||
const zeus::CTransform& xf, CModelData&& mData, const zeus::CAABox& bounds,
|
||||
const CHealthInfo& healthInfo, const CDamageVulnerability& vulnerability,
|
||||
const CActorParameters& actParams, TUniqueId otherId,
|
||||
const CActorParameters& actParams, TUniqueId parentId,
|
||||
const CGrenadeLauncherData& data, float f1)
|
||||
: CPhysicsActor(uid, true, name, info, xf, std::move(mData), {EMaterialTypes::Character, EMaterialTypes::Solid}, bounds,
|
||||
{1000.f}, actParams, 0.3f, 0.1f)
|
||||
, x25c_(healthInfo)
|
||||
, x25c_healthInfo(healthInfo)
|
||||
, x264_vulnerability(vulnerability)
|
||||
, x2cc_otherId(otherId)
|
||||
, x2cc_parentId(parentId)
|
||||
, x2d0_data(data)
|
||||
, x328_cSphere({{}, mData.GetScale().z()}, {EMaterialTypes::Character, EMaterialTypes::Solid})
|
||||
, x350_actParms(actParams)
|
||||
, x3c0_particleGenDesc(g_SimplePool->GetObj({SBIG('PART'), data.x40_}))
|
||||
, x3d8_(actParams.GetThermalMag())
|
||||
, x3e8_thermalMag(actParams.GetThermalMag())
|
||||
, x3f8_(f1) {
|
||||
if (data.x40_.IsValid()) {
|
||||
x3b8_particleGenDesc = g_SimplePool->GetObj({SBIG('PART'), data.x40_});
|
||||
}
|
||||
GetModelData()->EnableLooping(true);
|
||||
const CPASDatabase& pasDatabase = GetModelData()->GetAnimationData()->GetCharacterInfo().GetPASDatabase();
|
||||
for (int i = 0; i < 4; i++) {
|
||||
@@ -81,5 +86,215 @@ void CGrenadeLauncher::CalculateGrenadeTrajectory(const zeus::CVector3f& target,
|
||||
angleOut = angle;
|
||||
velocityOut = velocity;
|
||||
}
|
||||
|
||||
void CGrenadeLauncher::AcceptScriptMsg(EScriptObjectMessage msg, TUniqueId uid, CStateManager& mgr) {
|
||||
CActor::AcceptScriptMsg(msg, uid, mgr);
|
||||
switch (msg) {
|
||||
case EScriptObjectMessage::Start:
|
||||
if (x2cc_parentId == uid && x258_started != 1) {
|
||||
x258_started = 1;
|
||||
sub_80230438();
|
||||
}
|
||||
break;
|
||||
case EScriptObjectMessage::Stop:
|
||||
if (x2cc_parentId == uid && x258_started != 0) {
|
||||
x258_started = 0;
|
||||
sub_80230438();
|
||||
}
|
||||
break;
|
||||
case EScriptObjectMessage::Action:
|
||||
if (x2cc_parentId == uid && x258_started == 1) {
|
||||
x3fc_launchGrenade = true;
|
||||
}
|
||||
break;
|
||||
case EScriptObjectMessage::Registered:
|
||||
sub_80230438();
|
||||
break;
|
||||
case EScriptObjectMessage::Damage:
|
||||
x3ec_damageTimer = 0.33f;
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void CGrenadeLauncher::AddToRenderer(const zeus::CFrustum& frustum, const CStateManager& mgr) const {
|
||||
CActor::AddToRenderer(frustum, mgr);
|
||||
}
|
||||
|
||||
std::optional<zeus::CAABox> CGrenadeLauncher::GetTouchBounds() const {
|
||||
return x328_cSphere.CalculateAABox(GetTransform());
|
||||
}
|
||||
|
||||
void CGrenadeLauncher::PreRender(CStateManager& mgr, const zeus::CFrustum& frustum) {
|
||||
if (x3f4_color3.a() == 1.f) {
|
||||
xb4_drawFlags = {2, 0, 3, {x3f4_color3, 1.f}};
|
||||
} else {
|
||||
xb4_drawFlags = {5, 0, 3, x3f4_color3};
|
||||
}
|
||||
CActor::PreRender(mgr, frustum);
|
||||
}
|
||||
|
||||
void CGrenadeLauncher::Render(const CStateManager& mgr) const {
|
||||
if (x3fd_visible) {
|
||||
CPhysicsActor::Render(mgr);
|
||||
}
|
||||
}
|
||||
|
||||
void CGrenadeLauncher::Think(float dt, CStateManager& mgr) {
|
||||
if (GetActive()) {
|
||||
if (x3fc_launchGrenade) {
|
||||
LaunchGrenade(mgr);
|
||||
x3fc_launchGrenade = false;
|
||||
}
|
||||
|
||||
UpdateCollision();
|
||||
UpdateColor(dt);
|
||||
sub_8022f9e0(mgr, dt);
|
||||
sub_8022f69c(dt);
|
||||
|
||||
const SAdvancementDeltas& deltas = CActor::UpdateAnimation(dt, mgr, true);
|
||||
MoveToOR(deltas.x0_posDelta, dt);
|
||||
RotateToOR(deltas.xc_rotDelta, dt);
|
||||
|
||||
TCastToPtr<CPatterned> parent = mgr.ObjectById(x2cc_parentId);
|
||||
if (parent == nullptr || !parent->IsAlive() || parent->HealthInfo(mgr)->GetHP() <= 0.f) {
|
||||
mgr.SendScriptMsg(parent, GetUniqueId(), EScriptObjectMessage::Damage);
|
||||
CreateExplosion(mgr);
|
||||
mgr.FreeScriptObject(GetUniqueId());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void CGrenadeLauncher::Touch(CActor& act, CStateManager& mgr) {
|
||||
if (TCastToPtr<CGameProjectile> projectile = act) {
|
||||
if (projectile->GetOwnerId() == mgr.GetPlayer().GetUniqueId() &&
|
||||
GetDamageVulnerability()->WeaponHurts(CWeaponMode{projectile->GetType()}, false)) {
|
||||
x348_shotTimer = 0.5f;
|
||||
CEntity* parent = mgr.ObjectById(x2cc_parentId);
|
||||
if (parent != nullptr) {
|
||||
mgr.SendScriptMsg(parent, GetUniqueId(), EScriptObjectMessage::Touched);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void CGrenadeLauncher::UpdateCollision() {
|
||||
x328_cSphere.SetSphereCenter(GetLocatorTransform("lockon_target_LCTR"sv).origin);
|
||||
}
|
||||
|
||||
void CGrenadeLauncher::UpdateColor(float arg) {
|
||||
if (x348_shotTimer > 0.f) {
|
||||
x348_shotTimer = std::max(0.f, x348_shotTimer - arg);
|
||||
x34c_color1 = zeus::CColor::lerp(zeus::skWhite, zeus::skRed, x348_shotTimer);
|
||||
}
|
||||
}
|
||||
|
||||
void CGrenadeLauncher::sub_8022f69c(float arg) {
|
||||
if (x3ec_damageTimer <= 0.f) {
|
||||
xd0_damageMag = x3e8_thermalMag;
|
||||
} else {
|
||||
x3ec_damageTimer = std::max(0.f, x3ec_damageTimer - arg);
|
||||
x3f4_color3 = zeus::CColor::lerp(zeus::skBlack, x3f0_color2, std::clamp(x3ec_damageTimer / 0.33f, 0.f, 1.f));
|
||||
xd0_damageMag = 5.f * x3ec_damageTimer + x3e8_thermalMag;
|
||||
}
|
||||
}
|
||||
|
||||
void CGrenadeLauncher::CreateExplosion(CStateManager& mgr) {
|
||||
if (!x3b8_particleGenDesc)
|
||||
return;
|
||||
mgr.AddObject(new CExplosion(*x3b8_particleGenDesc, mgr.AllocateUniqueId(), true,
|
||||
{GetAreaIdAlways(), CEntity::NullConnectionList}, "Grenade Launcher Explode Fx"sv,
|
||||
GetTransform(), 0, GetModelData()->GetScale(), zeus::skWhite));
|
||||
CSfxManager::SfxStart(x2d0_data.x44_launcherExplodeSfx, 1.f, 1.f, false, 0x7f, false, kInvalidAreaId);
|
||||
}
|
||||
|
||||
void CGrenadeLauncher::sub_8022f9e0(CStateManager& mgr, float dt) {
|
||||
CModelData* modelData = GetModelData();
|
||||
CAnimData* animData;
|
||||
|
||||
if (modelData != nullptr && (animData = modelData->GetAnimationData()) != nullptr && x258_started == 1 && x3fe_) {
|
||||
const zeus::CVector3f& target = mgr.GetPlayer().GetAimPosition(mgr, 0.f) - GetTranslation();
|
||||
const zeus::CVector3f& rot = GetTransform().rotate({target.x(), target.y(), 0.f}); // TODO double check
|
||||
|
||||
if (rot.canBeNormalized()) {
|
||||
constexpr float p36d = zeus::degToRad(36.476f);
|
||||
constexpr float n45d = zeus::degToRad(-45.f);
|
||||
constexpr float p45d = zeus::degToRad(45.f);
|
||||
|
||||
float l84 = p36d * std::clamp(std::atan2(rot.x(), rot.y()), n45d, p45d);
|
||||
float l88 = std::clamp((0.25f * (l84 - x3d8_)) / dt, -3.f, 3.f);
|
||||
float l8c = std::clamp((l88 - x3dc_) / dt, -10.f, 10.f);
|
||||
x3dc_ += dt * l8c;
|
||||
|
||||
float l90 = p36d * std::clamp(std::atan2(rot.z(), rot.toVec2f().magnitude()), n45d, p45d);
|
||||
l88 = std::clamp((0.25f * (l90 - x3e0_)) / dt, -3.f, 3.f);
|
||||
l8c = std::clamp((l88 - x3e4_) / dt, -10.f, 10.f);
|
||||
x3e4_ += dt * l8c;
|
||||
|
||||
float dVar7 = std::clamp(dt * x3dc_ + x3d8_, -0.5f, 0.5f);
|
||||
float dVar8 = std::clamp(dt * x3e4_ + x3e0_, -0.5f, 0.5f);
|
||||
|
||||
if (dVar7 != x3d8_) {
|
||||
if (std::abs(x3d8_) > 0.f && x3d8_ * dVar7 <= 0.f) {
|
||||
animData->DelAdditiveAnimation(x3c8_animIds[x3d8_ >= 0.f ? 1 : 0]);
|
||||
}
|
||||
float weight = std::abs(dVar7);
|
||||
if (weight > 0.f) {
|
||||
animData->AddAdditiveAnimation(x3c8_animIds[dVar7 >= 0.f ? 1 : 0], weight, false, false);
|
||||
}
|
||||
}
|
||||
if (dVar8 != x3e0_) {
|
||||
if (std::abs(x3e0_) > 0.f && x3e0_ * dVar8 <= 0.f) {
|
||||
animData->DelAdditiveAnimation(x3c8_animIds[x3e0_ <= 0.f ? 3 : 2]);
|
||||
}
|
||||
float weight = std::abs(dVar8);
|
||||
if (weight > 0.f) {
|
||||
animData->AddAdditiveAnimation(x3c8_animIds[dVar8 <= 0.f ? 3 : 2], weight, false, false);
|
||||
}
|
||||
}
|
||||
x3d8_ = dVar7;
|
||||
x3e0_ = dVar8;
|
||||
}
|
||||
} else {
|
||||
if (x3d8_ != 0.f) {
|
||||
animData->DelAdditiveAnimation(x3c8_animIds[x3d8_ >= 0.f ? 1 : 0]);
|
||||
x3d8_ = 0.f;
|
||||
}
|
||||
if (x3e0_ != 0.f) {
|
||||
animData->DelAdditiveAnimation(x3c8_animIds[x3e0_ <= 0.f ? 3 : 2]);
|
||||
x3e0_ = 0.f;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void CGrenadeLauncher::sub_80230438() {
|
||||
CModelData* modelData = GetModelData();
|
||||
CAnimData* animData;
|
||||
if (modelData == nullptr || (animData = modelData->GetAnimationData()) == nullptr || x258_started <= -1 ||
|
||||
x258_started >= 2)
|
||||
return;
|
||||
|
||||
constexpr std::array arr = {0, 3};
|
||||
const auto& anim = animData->GetCharacterInfo().GetPASDatabase().FindBestAnimation(
|
||||
{5, CPASAnimParm::FromEnum(0), CPASAnimParm::FromEnum(arr[x258_started])}, -1);
|
||||
if (anim.first > 0.f) {
|
||||
animData->SetAnimation({anim.second, -1, 1.f, true}, false);
|
||||
modelData->EnableLooping(true);
|
||||
}
|
||||
}
|
||||
|
||||
void CGrenadeLauncher::LaunchGrenade(CStateManager& mgr) {
|
||||
CModelData* modelData = GetModelData();
|
||||
CAnimData* animData;
|
||||
if (modelData == nullptr || (animData = modelData->GetAnimationData()) == nullptr)
|
||||
return;
|
||||
|
||||
const auto& anim = animData->GetCharacterInfo().GetPASDatabase().FindBestAnimation({24}, -1);
|
||||
if (anim.first > 0.f) {
|
||||
animData->AddAdditiveAnimation(anim.second, 1.f, false, true);
|
||||
// TODO
|
||||
}
|
||||
}
|
||||
} // namespace MP1
|
||||
} // namespace urde
|
||||
|
||||
Reference in New Issue
Block a user