metaforce/Runtime/World/CScriptTrigger.cpp

358 lines
14 KiB
C++
Raw Normal View History

2016-04-19 21:25:26 +00:00
#include "CScriptTrigger.hpp"
#include "CActorParameters.hpp"
2016-04-29 10:08:46 +00:00
#include "CStateManager.hpp"
2017-01-15 03:07:01 +00:00
#include "TCastTo.hpp"
2017-01-15 03:59:37 +00:00
#include "World/CPlayer.hpp"
#include "Weapon/CGameProjectile.hpp"
#include "Camera/CGameCamera.hpp"
2016-04-19 21:25:26 +00:00
namespace urde
{
2017-11-13 06:19:18 +00:00
CScriptTrigger::CScriptTrigger(TUniqueId uid, std::string_view name, const CEntityInfo& info,
2017-01-15 03:59:37 +00:00
const zeus::CVector3f& pos, const zeus::CAABox& bounds, const CDamageInfo& dInfo,
2017-11-26 03:04:25 +00:00
const zeus::CVector3f& forceField, ETriggerFlags triggerFlags, bool active,
bool deactivateOnEntered, bool deactivateOnExited)
2016-04-19 21:25:26 +00:00
: CActor(uid, active, name, info, zeus::CTransform::Translate(pos), CModelData::CModelDataNull(),
2017-01-15 03:59:37 +00:00
CMaterialList(EMaterialTypes::Trigger), CActorParameters::None(), kInvalidUniqueId)
, x100_damageInfo(dInfo)
, x11c_forceField(forceField)
, x128_forceMagnitude(forceField.magnitude())
, x12c_flags(triggerFlags)
, x130_bounds(bounds)
2016-04-19 21:25:26 +00:00
{
x148_24_detectCamera = false;
x148_25_camSubmerged = false;
x148_26_deactivateOnEntered = deactivateOnEntered;
x148_27_deactivateOnExited = deactivateOnExited;
x148_28_playerTriggerProc = false;
x148_29_didPhazonDamage = false;
2017-11-24 08:23:28 +00:00
SetCallTouch(false);
2016-04-19 21:25:26 +00:00
}
2017-02-13 21:29:00 +00:00
void CScriptTrigger::Accept(IVisitor& visitor) { visitor.Visit(this); }
2017-01-15 03:07:01 +00:00
2017-01-15 03:59:37 +00:00
void CScriptTrigger::Think(float dt, CStateManager& mgr)
2016-04-29 10:08:46 +00:00
{
2017-01-15 03:59:37 +00:00
if (GetActive())
UpdateInhabitants(dt, mgr);
}
2016-11-20 21:53:15 +00:00
2017-01-15 03:59:37 +00:00
void CScriptTrigger::AcceptScriptMsg(EScriptObjectMessage msg, TUniqueId uid, CStateManager& mgr)
{
2017-02-14 04:27:20 +00:00
if (GetActive() && (msg == EScriptObjectMessage::Deactivate || msg == EScriptObjectMessage::Deleted))
2017-01-15 03:59:37 +00:00
{
if (msg == EScriptObjectMessage::Deactivate)
{
xe8_inhabitants.clear();
2017-08-13 07:56:35 +00:00
x148_25_camSubmerged = false;
2017-01-15 03:59:37 +00:00
}
2017-11-26 03:04:25 +00:00
if (x148_28_playerTriggerProc)
2017-01-15 03:59:37 +00:00
{
2017-11-26 03:04:25 +00:00
x148_28_playerTriggerProc = false;
2017-01-15 03:59:37 +00:00
if (x148_29_didPhazonDamage)
{
mgr.Player()->DecrementPhazon();
x148_29_didPhazonDamage = false;
}
2017-01-18 22:30:02 +00:00
if (x8_uid == mgr.GetLastTriggerId())
mgr.SetLastTriggerId(kInvalidUniqueId);
2017-01-15 03:59:37 +00:00
}
}
CEntity::AcceptScriptMsg(msg, uid, mgr);
2016-04-29 10:08:46 +00:00
}
2017-01-15 03:59:37 +00:00
CScriptTrigger::CObjectTracker* CScriptTrigger::FindObject(TUniqueId id)
2016-04-29 10:08:46 +00:00
{
2017-01-15 03:59:37 +00:00
auto& inhabitants = GetInhabitants();
const auto& iter = std::find(inhabitants.begin(), inhabitants.end(), id);
if (iter != inhabitants.end())
return &(*iter);
return nullptr;
2016-04-29 10:08:46 +00:00
}
2017-01-15 03:59:37 +00:00
void CScriptTrigger::UpdateInhabitants(float dt, CStateManager& mgr)
2016-04-29 10:08:46 +00:00
{
bool sendInside = false;
bool sendExited = false;
std::list<CObjectTracker>::iterator nextIt;
for (auto it = xe8_inhabitants.begin(); it != xe8_inhabitants.end(); it = nextIt)
2017-01-15 03:59:37 +00:00
{
nextIt = it;
++nextIt;
if (TCastToPtr<CActor> act = mgr.ObjectById(it->GetObjectId()))
2017-01-15 03:59:37 +00:00
{
bool playerValid = true;
if (it->GetObjectId() == mgr.GetPlayer().GetUniqueId())
2017-01-15 03:59:37 +00:00
{
if ((x12c_flags & ETriggerFlags::DetectPlayer) == ETriggerFlags::None &&
((mgr.GetPlayer().GetMorphballTransitionState() == CPlayer::EPlayerMorphBallState::Morphed &&
(x12c_flags & ETriggerFlags::DetectUnmorphedPlayer) != ETriggerFlags::None) ||
(mgr.GetPlayer().GetMorphballTransitionState() == CPlayer::EPlayerMorphBallState::Unmorphed &&
(x12c_flags & ETriggerFlags::DetectMorphedPlayer) != ETriggerFlags::None)))
2017-01-15 03:59:37 +00:00
{
playerValid = false;
}
if (!playerValid)
{
xe8_inhabitants.erase(it);
sendExited = true;
2017-11-26 03:04:25 +00:00
if (x148_28_playerTriggerProc)
2017-01-15 03:59:37 +00:00
{
2017-11-26 03:04:25 +00:00
x148_28_playerTriggerProc = false;
2017-01-15 03:59:37 +00:00
if (x148_29_didPhazonDamage)
{
mgr.GetPlayer().DecrementPhazon();
2017-01-15 03:59:37 +00:00
x148_29_didPhazonDamage = false;
}
2017-02-13 21:29:00 +00:00
if (mgr.GetLastTriggerId() == GetUniqueId())
mgr.SetLastTriggerId(kInvalidUniqueId);
2017-01-15 03:59:37 +00:00
}
InhabitantExited(*act, mgr);
continue;
}
}
auto touchBounds = GetTouchBounds();
auto actTouchBounds = act->GetTouchBounds();
if (touchBounds && actTouchBounds && touchBounds->intersects(*actTouchBounds))
2017-01-15 03:59:37 +00:00
{
sendInside = true;
InhabitantIdle(*act, mgr);
if (act->HealthInfo(mgr) && x100_damageInfo.GetDamage() > 0.f)
mgr.ApplyDamage(GetUniqueId(), act->GetUniqueId(), GetUniqueId(), x100_damageInfo,
CMaterialFilter::MakeIncludeExclude({EMaterialTypes::Solid}, {}),
zeus::CVector3f::skZero);
2017-01-15 03:59:37 +00:00
if (x128_forceMagnitude > 0.f)
{
if (TCastToPtr<CPhysicsActor> pact = act.GetPtr())
2017-01-15 03:59:37 +00:00
{
float forceMult = 1.f;
if ((x12c_flags & ETriggerFlags::UseBooleanIntersection) != ETriggerFlags::None)
2017-02-13 21:29:00 +00:00
forceMult =
touchBounds->booleanIntersection(*actTouchBounds).volume() / actTouchBounds->volume();
2017-01-15 03:59:37 +00:00
zeus::CVector3f force = forceMult * x11c_forceField;
if ((x12c_flags & ETriggerFlags::UseCollisionImpulses) != ETriggerFlags::None)
2017-01-15 03:59:37 +00:00
{
pact->ApplyImpulseWR(force, zeus::CAxisAngle::sIdentity);
pact->UseCollisionImpulses();
2017-01-15 03:59:37 +00:00
}
else
pact->ApplyForceWR(force, zeus::CAxisAngle::sIdentity);
2017-01-15 03:59:37 +00:00
}
}
}
else
{
2017-12-19 03:05:50 +00:00
TUniqueId tmpId = it->GetObjectId();
xe8_inhabitants.erase(it);
sendExited = true;
2017-12-19 03:05:50 +00:00
if (mgr.GetPlayer().GetUniqueId() == tmpId && x148_28_playerTriggerProc)
2017-01-15 03:59:37 +00:00
{
2017-11-26 03:04:25 +00:00
x148_28_playerTriggerProc = false;
if (x148_29_didPhazonDamage)
2017-01-15 03:59:37 +00:00
{
mgr.Player()->DecrementPhazon();
x148_29_didPhazonDamage = false;
2017-01-15 03:59:37 +00:00
}
if (mgr.GetLastTriggerId() == GetUniqueId())
mgr.SetLastTriggerId(kInvalidUniqueId);
}
InhabitantExited(*act, mgr);
}
}
else
{
2017-12-19 03:05:50 +00:00
TUniqueId tmpId = it->GetObjectId();
xe8_inhabitants.erase(it);
2017-12-19 03:05:50 +00:00
if (mgr.GetPlayer().GetUniqueId() == tmpId && x148_28_playerTriggerProc)
{
2017-11-26 03:04:25 +00:00
x148_28_playerTriggerProc = false;
if (x148_29_didPhazonDamage)
{
mgr.Player()->DecrementPhazon();
x148_29_didPhazonDamage = false;
2017-01-15 03:59:37 +00:00
}
2017-02-13 21:29:00 +00:00
if (mgr.GetLastTriggerId() == GetUniqueId())
mgr.SetLastTriggerId(kInvalidUniqueId);
}
}
}
2017-01-15 03:59:37 +00:00
2017-11-28 10:06:40 +00:00
if ((x12c_flags & ETriggerFlags::DetectCamera) != ETriggerFlags::None || x148_24_detectCamera)
{
CGameCamera* cam = mgr.GetCameraManager()->GetCurrentCamera(mgr);
bool camInTrigger = GetTriggerBoundsWR().pointInside(cam->GetTranslation());
if (x148_25_camSubmerged)
{
if (!camInTrigger)
{
x148_25_camSubmerged = false;
2017-11-28 10:06:40 +00:00
if ((x12c_flags & ETriggerFlags::DetectCamera) != ETriggerFlags::None)
{
sendExited = true;
InhabitantExited(*cam, mgr);
}
}
else
{
2017-11-28 10:06:40 +00:00
if ((x12c_flags & ETriggerFlags::DetectCamera) != ETriggerFlags::None)
{
InhabitantIdle(*cam, mgr);
sendInside = true;
}
2017-01-15 03:59:37 +00:00
}
}
else
{
if (camInTrigger)
2017-01-15 03:59:37 +00:00
{
x148_25_camSubmerged = true;
2017-11-28 10:06:40 +00:00
if ((x12c_flags & ETriggerFlags::DetectCamera) != ETriggerFlags::None)
2017-01-15 03:59:37 +00:00
{
InhabitantAdded(*cam, mgr);
SendScriptMsgs(EScriptObjectState::Entered, mgr, EScriptObjectMessage::Activate);
2017-01-15 03:59:37 +00:00
}
}
}
}
if (sendInside)
2017-01-15 03:59:37 +00:00
{
SendScriptMsgs(EScriptObjectState::Inside, mgr, EScriptObjectMessage::Activate);
2017-01-15 03:59:37 +00:00
}
if (sendExited)
2017-01-15 03:59:37 +00:00
{
SendScriptMsgs(EScriptObjectState::Exited, mgr, EScriptObjectMessage::Activate);
2017-01-15 03:59:37 +00:00
if (x148_27_deactivateOnExited)
{
2017-02-13 21:29:00 +00:00
mgr.SendScriptMsg(GetUniqueId(), mgr.GetEditorIdForUniqueId(GetUniqueId()),
EScriptObjectMessage::Deactivate, EScriptObjectState::Exited);
2017-01-15 03:59:37 +00:00
}
}
2016-04-29 10:08:46 +00:00
}
2017-01-15 03:59:37 +00:00
std::list<CScriptTrigger::CObjectTracker>& CScriptTrigger::GetInhabitants() { return xe8_inhabitants; }
std::experimental::optional<zeus::CAABox> CScriptTrigger::GetTouchBounds() const
2016-04-29 10:08:46 +00:00
{
if (x30_24_active)
return {GetTriggerBoundsWR()};
return {};
}
2017-01-15 03:59:37 +00:00
static const CWeaponMode sktonOHurtWeaponMode = CWeaponMode(EWeaponType::Power, false, false, true);
void CScriptTrigger::Touch(CActor& act, CStateManager& mgr)
{
if (!act.GetActive() || act.GetMaterialList().HasMaterial(EMaterialTypes::Trigger))
return;
if (FindObject(act.GetUniqueId()) == nullptr)
{
ETriggerFlags testFlags = ETriggerFlags::None;
TCastToPtr<CPlayer> pl(act);
if (pl)
{
if (x128_forceMagnitude > 0.f && (x12c_flags & ETriggerFlags::DetectPlayer) != ETriggerFlags::None &&
2017-01-18 22:30:02 +00:00
mgr.GetLastTriggerId() == kInvalidUniqueId)
mgr.SetLastTriggerId(x8_uid);
2017-01-15 03:59:37 +00:00
testFlags |= ETriggerFlags::DetectPlayer;
if (pl->GetMorphballTransitionState() == CPlayer::EPlayerMorphBallState::Unmorphed)
testFlags |= ETriggerFlags::DetectUnmorphedPlayer;
else if (pl->GetMorphballTransitionState() == CPlayer::EPlayerMorphBallState::Morphed)
testFlags |= ETriggerFlags::DetectMorphedPlayer;
}
else if (TCastToPtr<CAi>(act))
{
testFlags |= ETriggerFlags::DetectAI;
}
else if (TCastToPtr<CGameProjectile>(act))
{
testFlags |= ETriggerFlags::DetectProjectiles1 | ETriggerFlags::DetectProjectiles2 |
ETriggerFlags::DetectProjectiles3 | ETriggerFlags::DetectProjectiles4 |
ETriggerFlags::DetectProjectiles5 | ETriggerFlags::DetectProjectiles6 |
ETriggerFlags::DetectProjectiles7;
}
else if (CWeapon* weap = TCastToPtr<CWeapon>(act))
{
2018-02-12 05:30:21 +00:00
if ((weap->GetAttribField() & EProjectileAttrib::Bombs) == EProjectileAttrib::Bombs)
2017-01-15 03:59:37 +00:00
testFlags |= ETriggerFlags::DetectBombs;
2018-02-12 05:30:21 +00:00
else if ((weap->GetAttribField() & EProjectileAttrib::PowerBombs) == EProjectileAttrib::PowerBombs)
2017-01-15 03:59:37 +00:00
testFlags |= ETriggerFlags::DetectPowerBombs;
}
if ((testFlags & x12c_flags) != ETriggerFlags::None)
{
xe8_inhabitants.push_back(act.GetUniqueId());
InhabitantAdded(act, mgr);
if (pl)
{
2017-11-26 03:04:25 +00:00
if (!x148_28_playerTriggerProc)
2017-01-15 03:59:37 +00:00
{
2017-11-26 03:04:25 +00:00
x148_28_playerTriggerProc = true;
2017-01-15 03:59:37 +00:00
if (x148_29_didPhazonDamage)
{
mgr.Player()->DecrementPhazon();
x148_29_didPhazonDamage = false;
}
else if (x100_damageInfo.GetDamage() > 0.f)
{
const CDamageVulnerability* dVuln = mgr.Player()->GetDamageVulnerability();
if (dVuln->WeaponHurts(x100_damageInfo.GetWeaponMode(), 0) &&
x100_damageInfo.GetWeaponMode().GetType() == EWeaponType::Phazon &&
!mgr.GetPlayerState()->HasPowerUp(CPlayerState::EItemType::PhazonSuit))
{
pl->IncrementPhazon();
x148_29_didPhazonDamage = true;
}
}
}
}
SendScriptMsgs(EScriptObjectState::Entered, mgr, EScriptObjectMessage::None);
2017-01-15 03:59:37 +00:00
if (x148_26_deactivateOnEntered)
{
mgr.SendScriptMsg(x8_uid, mgr.GetEditorIdForUniqueId(x8_uid), EScriptObjectMessage::Deactivate,
EScriptObjectState::Entered);
2017-08-20 05:23:22 +00:00
if (act.HealthInfo(mgr) && x100_damageInfo.GetDamage() > 0.f)
2017-01-15 03:59:37 +00:00
{
mgr.ApplyDamage(x8_uid, act.GetUniqueId(), x8_uid, x100_damageInfo,
CMaterialFilter::MakeIncludeExclude({EMaterialTypes::Solid}, {0ull}),
zeus::CVector3f::skZero);
}
}
2017-08-20 05:23:22 +00:00
if ((x12c_flags & ETriggerFlags::KillOnEnter) != ETriggerFlags::None && act.HealthInfo(mgr))
2017-01-15 03:59:37 +00:00
{
2017-08-20 05:23:22 +00:00
CHealthInfo* hInfo = act.HealthInfo(mgr);
2017-01-15 03:59:37 +00:00
mgr.ApplyDamage(
x8_uid, act.GetUniqueId(), x8_uid, {sktonOHurtWeaponMode, 10.f * hInfo->GetHP(), 0.f, 0.f},
CMaterialFilter::MakeIncludeExclude({EMaterialTypes::Solid}, {0ull}), zeus::CVector3f::skZero);
}
}
else
InhabitantRejected(act, mgr);
}
}
2016-04-29 10:08:46 +00:00
zeus::CAABox CScriptTrigger::GetTriggerBoundsWR() const
{
return {x130_bounds.min + x34_transform.origin, x130_bounds.max + x34_transform.origin};
}
2016-04-19 21:25:26 +00:00
}