mirror of
				https://github.com/AxioDL/metaforce.git
				synced 2025-10-25 11:30:24 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			179 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			179 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "Runtime/MP1/World/CShockWave.hpp"
 | |
| 
 | |
| #include "Runtime/Collision/CCollisionActor.hpp"
 | |
| #include "Runtime/CSimplePool.hpp"
 | |
| #include "Runtime/CStateManager.hpp"
 | |
| #include "Runtime/GameGlobalObjects.hpp"
 | |
| #include "Runtime/Graphics/CCubeRenderer.hpp"
 | |
| #include "Runtime/World/CActorParameters.hpp"
 | |
| #include "Runtime/World/CGameLight.hpp"
 | |
| #include "Runtime/World/CHUDBillboardEffect.hpp"
 | |
| #include "Runtime/World/CPlayer.hpp"
 | |
| 
 | |
| #include "TCastTo.hpp" // Generated file, do not modify include path
 | |
| 
 | |
| namespace metaforce::MP1 {
 | |
| CShockWave::CShockWave(TUniqueId uid, std::string_view name, const CEntityInfo& info, const zeus::CTransform& xf,
 | |
|                        TUniqueId parent, const SShockWaveData& data, float minActiveTime, float knockback)
 | |
| : CActor(uid, true, name, info, xf, CModelData::CModelDataNull(), {EMaterialTypes::Projectile},
 | |
|          CActorParameters::None(), kInvalidUniqueId)
 | |
| , xe8_parentId(parent)
 | |
| , xec_damageInfo(data.GetDamageInfo())
 | |
| , x108_elementGenDesc(g_SimplePool->GetObj({SBIG('PART'), data.GetParticleDescId()}))
 | |
| , x110_elementGen(std::make_unique<CElementGen>(x108_elementGenDesc))
 | |
| , x114_data(data)
 | |
| , x150_radius(data.GetInitialRadius())
 | |
| , x154_expansionSpeed(data.GetInitialExpansionSpeed())
 | |
| , x15c_minActiveTime(minActiveTime)
 | |
| , x160_knockback(knockback) {
 | |
|   if (data.GetWeaponDescId().IsValid()) {
 | |
|     x974_electricDesc = g_SimplePool->GetObj({SBIG('ELSC'), data.GetWeaponDescId()});
 | |
|   }
 | |
|   x110_elementGen->SetParticleEmission(true);
 | |
|   x110_elementGen->SetOrientation(GetTransform().getRotation());
 | |
|   x110_elementGen->SetGlobalTranslation(GetTranslation());
 | |
|   xe6_27_thermalVisorFlags = 2;
 | |
| }
 | |
| 
 | |
| void CShockWave::Accept(IVisitor& visitor) { visitor.Visit(this); }
 | |
| 
 | |
| void CShockWave::AcceptScriptMsg(EScriptObjectMessage msg, TUniqueId uid, CStateManager& mgr) {
 | |
|   if (msg == EScriptObjectMessage::Registered) {
 | |
|     if (x110_elementGen->SystemHasLight()) {
 | |
|       x980_id2 = mgr.AllocateUniqueId();
 | |
|       mgr.AddObject(new CGameLight(x980_id2, GetAreaIdAlways(), GetActive(), "ShockWaveLight_" + x10_name,
 | |
|                                    GetTransform(), GetUniqueId(), x110_elementGen->GetLight(),
 | |
|                                    x114_data.GetParticleDescId().Value(), 1, 0.f));
 | |
|     }
 | |
|   } else if (msg == EScriptObjectMessage::Deleted) {
 | |
|     mgr.FreeScriptObject(x980_id2);
 | |
|     x980_id2 = kInvalidUniqueId;
 | |
|   }
 | |
|   CActor::AcceptScriptMsg(msg, uid, mgr);
 | |
|   mgr.SendScriptMsgAlways(x980_id2, uid, msg);
 | |
| }
 | |
| 
 | |
| void CShockWave::AddToRenderer(const zeus::CFrustum& frustum, CStateManager& mgr) {
 | |
|   CActor::AddToRenderer(frustum, mgr);
 | |
|   g_Renderer->AddParticleGen(*x110_elementGen);
 | |
| }
 | |
| 
 | |
| std::optional<zeus::CAABox> CShockWave::GetTouchBounds() const {
 | |
|   if (x150_radius <= 0.f) {
 | |
|     return std::nullopt;
 | |
|   }
 | |
|   return zeus::CAABox({-x150_radius, -x150_radius, 0.f}, {x150_radius, x150_radius, 1.f})
 | |
|       .getTransformedAABox(GetTransform());
 | |
| }
 | |
| 
 | |
| void CShockWave::Render(CStateManager& mgr) {
 | |
|   CActor::Render(mgr);
 | |
|   x110_elementGen->Render();
 | |
| }
 | |
| 
 | |
| void CShockWave::Think(float dt, CStateManager& mgr) {
 | |
|   if (GetActive()) {
 | |
|     x110_elementGen->Update(dt);
 | |
|     x158_activeTime += dt;
 | |
|     x150_radius += x154_expansionSpeed * dt;
 | |
|     x154_expansionSpeed += dt * x114_data.GetSpeedIncrease();
 | |
|     x110_elementGen->SetExternalVar(0, x150_radius);
 | |
|     for (size_t i = 0; i < x110_elementGen->GetNumActiveChildParticles(); ++i) {
 | |
|       auto& particle = static_cast<CElementGen&>(x110_elementGen->GetActiveChildParticle(i));
 | |
|       if (particle.Get4CharId() == SBIG('PART')) {
 | |
|         particle.SetExternalVar(0, x150_radius);
 | |
|       }
 | |
|     }
 | |
|     if (x16c_hitPlayerInAir) {
 | |
|       x164_timeSinceHitPlayerInAir += dt;
 | |
|       x16c_hitPlayerInAir = false;
 | |
|     }
 | |
|     if (x16d_hitPlayer) {
 | |
|       x168_timeSinceHitPlayer += dt;
 | |
|       x16d_hitPlayer = false;
 | |
|     }
 | |
|   }
 | |
|   if (x110_elementGen->IsSystemDeletable() && x15c_minActiveTime > 0.f && x158_activeTime >= x15c_minActiveTime) {
 | |
|     mgr.FreeScriptObject(GetUniqueId());
 | |
|   } else if (x980_id2 != kInvalidUniqueId) {
 | |
|     if (TCastToPtr<CGameLight> light = mgr.ObjectById(x980_id2)) {
 | |
|       if (light->GetActive()) {
 | |
|         light->SetLight(x110_elementGen->GetLight());
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CShockWave::Touch(CActor& actor, CStateManager& mgr) {
 | |
|   if (x158_activeTime >= x15c_minActiveTime) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   bool isParent = xe8_parentId == actor.GetUniqueId();
 | |
|   if (TCastToConstPtr<CCollisionActor> cactor = mgr.GetObjectById(actor.GetUniqueId())) {
 | |
|     isParent = xe8_parentId == cactor->GetOwnerId();
 | |
|   }
 | |
|   if (isParent) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   float mmax = x150_radius * x150_radius;
 | |
|   float mmin = mmax * x114_data.GetWidthPercent() * x114_data.GetWidthPercent();
 | |
|   zeus::CVector3f dist = actor.GetTranslation() - GetTranslation();
 | |
|   CDamageInfo damageInfo = xec_damageInfo;
 | |
|   float knockBackScale = std::max(0.f, 1.f - x160_knockback * x158_activeTime);
 | |
|   bool isPlayer = mgr.GetPlayer().GetUniqueId() == actor.GetUniqueId();
 | |
|   bool isPlayerInAir = isPlayer && mgr.GetPlayer().GetPlayerMovementState() != CPlayer::EPlayerMovementState::OnGround;
 | |
|   float distXYMag = dist.toVec2f().magSquared();
 | |
|   if (distXYMag < mmin || distXYMag > mmax) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (isPlayer) {
 | |
|     if (mgr.GetPlayer().GetPlayerMovementState() == CPlayer::EPlayerMovementState::OnGround) {
 | |
|       const zeus::CTransform& playerTransform = mgr.GetPlayer().GetTransform();
 | |
|       zeus::CVector3f playerDir = GetTranslation() - playerTransform.origin;
 | |
|       if (playerDir.canBeNormalized()) {
 | |
|         playerDir.normalize();
 | |
|         float dot = std::abs(playerDir.dot(playerTransform.frontVector()));
 | |
|         knockBackScale = std::max(0.12f, 0.88f * dot * dot);
 | |
|       }
 | |
|     }
 | |
|     if (mgr.GetPlayer().GetVelocity().magnitude() > 40.f) {
 | |
|       x168_timeSinceHitPlayer = 0.2666f;
 | |
|     }
 | |
|   }
 | |
|   damageInfo.SetKnockBackPower(knockBackScale * damageInfo.GetKnockBackPower());
 | |
| 
 | |
|   if (isPlayer && (x164_timeSinceHitPlayerInAir >= 0.1333f || x168_timeSinceHitPlayer >= 0.2666f)) {
 | |
|     return;
 | |
|   }
 | |
|   if (!IsHit(actor.GetUniqueId())) {
 | |
|     mgr.ApplyDamage(GetUniqueId(), actor.GetUniqueId(), GetUniqueId(), damageInfo,
 | |
|                     CMaterialFilter::MakeInclude({EMaterialTypes::Solid}), zeus::skZero3f);
 | |
|     if (isPlayer && x974_electricDesc) {
 | |
|       mgr.AddObject(new CHUDBillboardEffect(std::nullopt, x974_electricDesc, mgr.AllocateUniqueId(), true,
 | |
|                                             "VisorElectricFx", CHUDBillboardEffect::GetNearClipDistance(mgr),
 | |
|                                             CHUDBillboardEffect::GetScaleForPOV(mgr), zeus::skWhite, zeus::skOne3f,
 | |
|                                             zeus::skZero3f));
 | |
|       CSfxManager::SfxStart(x114_data.GetElectrocuteSfx(), 1.f, 1.f, false, 0x7f, false, kInvalidAreaId);
 | |
|     }
 | |
|     x170_hitIds.push_back(actor.GetUniqueId());
 | |
|   } else {
 | |
|     damageInfo.SetDamage(0.f);
 | |
|     mgr.ApplyDamage(GetUniqueId(), actor.GetUniqueId(), GetUniqueId(), damageInfo,
 | |
|                     CMaterialFilter::MakeInclude({EMaterialTypes::Solid}), zeus::skZero3f);
 | |
|   }
 | |
|   if (isPlayerInAir) {
 | |
|     x16c_hitPlayerInAir = true;
 | |
|   }
 | |
|   if (isPlayer) {
 | |
|     x16d_hitPlayer = true;
 | |
|   }
 | |
| }
 | |
| 
 | |
| bool CShockWave::IsHit(TUniqueId id) const {
 | |
|   return std::find(x170_hitIds.begin(), x170_hitIds.end(), id) != x170_hitIds.end();
 | |
| }
 | |
| } // namespace metaforce::MP1
 |