source-engine/game/server/hl2/npc_attackchopper.cpp
FluorescentCIAAfricanAmerican 3bf9df6b27 1
2020-04-22 12:56:21 -04:00

6118 lines
188 KiB
C++

//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
//===========================================================================//
#include "cbase.h"
#include "ai_network.h"
#include "ai_default.h"
#include "ai_schedule.h"
#include "ai_hull.h"
#include "ai_node.h"
#include "ai_task.h"
#include "entitylist.h"
#include "basecombatweapon.h"
#include "soundenvelope.h"
#include "gib.h"
#include "gamerules.h"
#include "ammodef.h"
#include "grenade_homer.h"
#include "cbasehelicopter.h"
#include "engine/IEngineSound.h"
#include "IEffects.h"
#include "globals.h"
#include "explode.h"
#include "movevars_shared.h"
#include "smoke_trail.h"
#include "ar2_explosion.h"
#include "collisionutils.h"
#include "props.h"
#include "EntityFlame.h"
#include "decals.h"
#include "effect_dispatch_data.h"
#include "te_effect_dispatch.h"
#include "ai_spotlight.h"
#include "vphysics/constraints.h"
#include "physics_saverestore.h"
#include "ai_memory.h"
#include "npc_attackchopper.h"
#ifdef HL2_EPISODIC
#include "physics_bone_follower.h"
#endif // HL2_EPISODIC
// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"
// -------------------------------------
// Bone controllers
// -------------------------------------
#define CHOPPER_DRONE_NAME "models/combine_helicopter/helicopter_bomb01.mdl"
#define CHOPPER_MODEL_NAME "models/combine_helicopter.mdl"
#define CHOPPER_MODEL_CORPSE_NAME "models/combine_helicopter_broken.mdl"
#define CHOPPER_RED_LIGHT_SPRITE "sprites/redglow1.vmt"
#define CHOPPER_MAX_SMALL_CHUNKS 1
#define CHOPPER_MAX_CHUNKS 3
static const char *s_pChunkModelName[CHOPPER_MAX_CHUNKS] =
{
"models/gibs/helicopter_brokenpiece_01.mdl",
"models/gibs/helicopter_brokenpiece_02.mdl",
"models/gibs/helicopter_brokenpiece_03.mdl",
};
#define BOMB_SKIN_LIGHT_ON 1
#define BOMB_SKIN_LIGHT_OFF 0
#define HELICOPTER_CHUNK_COCKPIT "models/gibs/helicopter_brokenpiece_04_cockpit.mdl"
#define HELICOPTER_CHUNK_TAIL "models/gibs/helicopter_brokenpiece_05_tailfan.mdl"
#define HELICOPTER_CHUNK_BODY "models/gibs/helicopter_brokenpiece_06_body.mdl"
#define CHOPPER_MAX_SPEED (60 * 17.6f)
#define CHOPPER_MAX_FIRING_SPEED 250.0f
#define CHOPPER_MAX_GUN_DIST 2000.0f
#define CHOPPER_ACCEL_RATE 500
#define CHOPPER_ACCEL_RATE_BOOST 1500
#define DEFAULT_FREE_KNOWLEDGE_DURATION 5.0f
// -------------------------------------
// Pathing data
#define CHOPPER_LEAD_DISTANCE 800.0f
#define CHOPPER_MIN_CHASE_DIST_DIFF 128.0f // Distance threshold used to determine when a target has moved enough to update our navigation to it
#define CHOPPER_MIN_AGGRESSIVE_CHASE_DIST_DIFF 64.0f
#define CHOPPER_AVOID_DIST 512.0f
#define CHOPPER_ARRIVE_DIST 128.0f
#define CHOPPER_MAX_CIRCLE_OF_DEATH_FOLLOW_SPEED 450.0f
#define CHOPPER_MIN_CIRCLE_OF_DEATH_RADIUS 150.0f
#define CHOPPER_MAX_CIRCLE_OF_DEATH_RADIUS 350.0f
#define CHOPPER_BOMB_DROP_COUNT 6
// Bullrush
#define CHOPPER_BULLRUSH_MODE_DISTANCE g_helicopter_bullrush_distance.GetFloat()
#define CHOPPER_BULLRUSH_ENEMY_BOMB_DISTANCE g_helicopter_bullrush_bomb_enemy_distance.GetFloat()
#define CHOPPER_BULLRUSH_ENEMY_BOMB_TIME g_helicopter_bullrush_bomb_time.GetFloat()
#define CHOPPER_BULLRUSH_ENEMY_BOMB_SPEED g_helicopter_bullrush_bomb_speed.GetFloat()
#define CHOPPER_BULLRUSH_SHOOTING_VERTICAL_OFFSET g_helicopter_bullrush_shoot_height.GetFloat()
#define CHOPPER_GUN_CHARGE_TIME g_helicopter_chargetime.GetFloat()
#define CHOPPER_GUN_IDLE_TIME g_helicopter_idletime.GetFloat()
#define CHOPPER_GUN_MAX_FIRING_DIST g_helicopter_maxfiringdist.GetFloat()
#define BULLRUSH_IDLE_PLAYER_FIRE_TIME 6.0f
#define DRONE_SPEED sk_helicopter_drone_speed.GetFloat()
#define SF_HELICOPTER_LOUD_ROTOR_SOUND 0x00010000
#define SF_HELICOPTER_ELECTRICAL_DRONE 0x00020000
#define SF_HELICOPTER_LIGHTS 0x00040000
#define SF_HELICOPTER_IGNORE_AVOID_FORCES 0x00080000
#define SF_HELICOPTER_AGGRESSIVE 0x00100000
#define SF_HELICOPTER_LONG_SHADOW 0x00200000
#define CHOPPER_SLOW_BOMB_SPEED 250
#define CHOPPER_BULLRUSH_SLOW_SHOOT_SPEED 250
#define CHOPPER_BULLRUSH_SLOW_SHOOT_SPEED_SQ (CHOPPER_BULLRUSH_SLOW_SHOOT_SPEED * CHOPPER_BULLRUSH_SLOW_SHOOT_SPEED)
#define CHOPPER_BULLRUSH_SLOW_SHOOT_SPEED_2 450
#define CHOPPER_BULLRUSH_SLOW_SHOOT_SPEED_2_SQ (CHOPPER_BULLRUSH_SLOW_SHOOT_SPEED_2 * CHOPPER_BULLRUSH_SLOW_SHOOT_SPEED_2)
// CVars
ConVar sk_helicopter_health( "sk_helicopter_health","5600");
ConVar sk_helicopter_firingcone( "sk_helicopter_firingcone","20.0", 0, "The angle in degrees of the cone in which the shots will be fired" );
ConVar sk_helicopter_burstcount( "sk_helicopter_burstcount","12", 0, "How many shot bursts to fire after charging up. The bigger the number, the longer the firing is" );
ConVar sk_helicopter_roundsperburst( "sk_helicopter_roundsperburst","5", 0, "How many shots to fire in a single burst" );
ConVar sk_helicopter_grenadedamage( "sk_helicopter_grenadedamage","25.0", 0, "The amount of damage the helicopter grenade deals." );
ConVar sk_helicopter_grenaderadius( "sk_helicopter_grenaderadius","275.0", 0, "The damage radius of the helicopter grenade." );
ConVar sk_helicopter_grenadeforce( "sk_helicopter_grenadeforce","55000.0", 0, "The physics force that the helicopter grenade exerts." );
ConVar sk_helicopter_grenade_puntscale( "sk_helicopter_grenade_puntscale","1.5", 0, "When physpunting a chopper's grenade, scale its velocity by this much." );
// Number of bomb hits it takes to kill a chopper on each skill level.
ConVar sk_helicopter_num_bombs1("sk_helicopter_num_bombs1", "3");
ConVar sk_helicopter_num_bombs2("sk_helicopter_num_bombs2", "5");
ConVar sk_helicopter_num_bombs3("sk_helicopter_num_bombs3", "5");
ConVar sk_npc_dmg_helicopter_to_plr( "sk_npc_dmg_helicopter_to_plr","3", 0, "Damage helicopter shots deal to the player" );
ConVar sk_npc_dmg_helicopter( "sk_npc_dmg_helicopter","6", 0, "Damage helicopter shots deal to everything but the player" );
ConVar sk_helicopter_drone_speed( "sk_helicopter_drone_speed","450.0", 0, "How fast does the zapper drone move?" );
ConVar g_helicopter_chargetime( "g_helicopter_chargetime","2.0", 0, "How much time we have to wait (on average) between the time we start hearing the charging sound + the chopper fires" );
ConVar g_helicopter_bullrush_distance("g_helicopter_bullrush_distance", "5000");
ConVar g_helicopter_bullrush_bomb_enemy_distance("g_helicopter_bullrush_bomb_enemy_distance", "0");
ConVar g_helicopter_bullrush_bomb_time("g_helicopter_bullrush_bomb_time", "10");
ConVar g_helicopter_idletime( "g_helicopter_idletime","3.0", 0, "How much time we have to wait (on average) after we fire before we can charge up again" );
ConVar g_helicopter_maxfiringdist( "g_helicopter_maxfiringdist","2500.0", 0, "The maximum distance the player can be from the chopper before it stops firing" );
ConVar g_helicopter_bullrush_bomb_speed( "g_helicopter_bullrush_bomb_speed","850.0", 0, "The maximum distance the player can be from the chopper before it stops firing" );
ConVar g_helicopter_bullrush_shoot_height( "g_helicopter_bullrush_shoot_height","650.0", 0, "The maximum distance the player can be from the chopper before it stops firing" );
ConVar g_helicopter_bullrush_mega_bomb_health( "g_helicopter_bullrush_mega_bomb_health","0.25", 0, "Fraction of the health of the chopper before it mega-bombs" );
ConVar g_helicopter_bomb_danger_radius( "g_helicopter_bomb_danger_radius", "120" );
Activity ACT_HELICOPTER_DROP_BOMB;
Activity ACT_HELICOPTER_CRASHING;
static const char *s_pBlinkLightThinkContext = "BlinkLights";
static const char *s_pSpotlightThinkContext = "SpotlightThink";
static const char *s_pRampSoundContext = "RampSound";
static const char *s_pWarningBlinkerContext = "WarningBlinker";
static const char *s_pAnimateThinkContext = "Animate";
#define CHOPPER_LIGHT_BLINK_TIME 1.0f
#define CHOPPER_LIGHT_BLINK_TIME_SHORT 0.1f
#define BOMB_LIFETIME 2.5f // Don't access this directly. Call GetBombLifetime();
#define BOMB_RAMP_SOUND_TIME 1.0f
enum
{
MAX_HELICOPTER_LIGHTS = 3,
};
enum
{
SF_GRENADE_HELICOPTER_MEGABOMB = 0x1,
};
#define GRENADE_HELICOPTER_MODEL "models/combine_helicopter/helicopter_bomb01.mdl"
LINK_ENTITY_TO_CLASS( info_target_helicopter_crash, CPointEntity );
//------------------------------------------------------------------------------
// Purpose :
//------------------------------------------------------------------------------
static inline float ClampSplineRemapVal( float flValue, float flMinValue, float flMaxValue, float flOutMin, float flOutMax )
{
Assert( flMinValue <= flMaxValue );
float flClampedVal = clamp( flValue, flMinValue, flMaxValue );
return SimpleSplineRemapVal( flClampedVal, flMinValue, flMaxValue, flOutMin, flOutMax );
}
//-----------------------------------------------------------------------------
// The bombs the attack helicopter drops
//-----------------------------------------------------------------------------
enum
{
SKIN_REGULAR,
SKIN_DUD,
};
class CGrenadeHelicopter : public CBaseGrenade
{
DECLARE_DATADESC();
public:
DECLARE_CLASS( CGrenadeHelicopter, CBaseGrenade );
virtual void Precache( );
virtual void Spawn( );
virtual void UpdateOnRemove();
virtual void OnEntityEvent( EntityEvent_t event, void *pEventData );
virtual void PhysicsSimulate( void );
virtual float GetShakeAmplitude( void ) { return 25.0; }
virtual float GetShakeRadius( void ) { return sk_helicopter_grenaderadius.GetFloat() * 2; }
virtual int OnTakeDamage( const CTakeDamageInfo &info );
virtual void VPhysicsCollision( int index, gamevcollisionevent_t *pEvent );
void SetExplodeOnContact( bool bExplode ) { m_bExplodeOnContact = bExplode; }
virtual QAngle PreferredCarryAngles( void ) { return QAngle( -12, 98, 55 ); }
virtual bool HasPreferredCarryAnglesForPlayer( CBasePlayer *pPlayer ) { return true; }
float GetBombLifetime();
#ifdef HL2_EPISODIC
virtual void OnPhysGunPickup(CBasePlayer *pPhysGunUser, PhysGunPickup_t reason );
virtual void OnPhysGunDrop( CBasePlayer *pPhysGunUser, PhysGunDrop_t reason );
virtual Class_T Classify( void ) { return CLASS_MISSILE; }
void SetCollisionObject( CBaseEntity *pEntity ) { m_hCollisionObject = pEntity; }
void SendMissEvent();
bool IsThrownByPlayer();
virtual bool ShouldPuntUseLaunchForces( PhysGunForce_t reason ) { return ( reason == PHYSGUN_FORCE_LAUNCHED ); }
virtual Vector PhysGunLaunchVelocity( const Vector &forward, float flMass );
void InputExplodeIn( inputdata_t &inputdata );
#endif // HL2_EPISODIC
private:
// Pow!
void DoExplosion( const Vector &vecOrigin, const Vector &vecVelocity );
void ExplodeThink();
void RampSoundThink();
void WarningBlinkerThink();
void StopWarningBlinker();
void AnimateThink();
void ExplodeConcussion( CBaseEntity *pOther );
void BecomeActive();
void ResolveFlyCollisionCustom( trace_t &trace, Vector &vecVelocity );
bool m_bActivated;
bool m_bExplodeOnContact;
CSoundPatch *m_pWarnSound;
EHANDLE m_hWarningSprite;
bool m_bBlinkerAtTop;
#ifdef HL2_EPISODIC
float m_flLifetime;
EHANDLE m_hCollisionObject; // Pointer to object we re-enable collisions with when picked up
bool m_bPickedUp;
float m_flBlinkFastTime;
COutputEvent m_OnPhysGunOnlyPickup;
#endif // HL2_EPISODIC
};
//-----------------------------------------------------------------------------
// The bombs the attack helicopter drops
//-----------------------------------------------------------------------------
class CBombDropSensor : public CBaseEntity
{
DECLARE_DATADESC();
public:
DECLARE_CLASS( CBombDropSensor, CBaseEntity );
void Spawn();
// Drop a bomb at a particular location
void InputDropBomb( inputdata_t &inputdata );
void InputDropBombStraightDown( inputdata_t &inputdata );
void InputDropBombAtTarget( inputdata_t &inputdata );
void InputDropBombAtTargetAlways( inputdata_t &inputdata );
void InputDropBombDelay( inputdata_t &inputdata );
};
//-----------------------------------------------------------------------------
// This entity is used to create boxes that the helicopter can't bomb in
//-----------------------------------------------------------------------------
class CBombSuppressor : public CBaseEntity
{
DECLARE_DATADESC();
public:
DECLARE_CLASS( CBombSuppressor, CBaseEntity );
virtual void Spawn( );
virtual void Activate();
virtual void UpdateOnRemove();
static bool CanBomb( const Vector &vecPosition );
private:
typedef CHandle<CBombSuppressor> BombSuppressorHandle_t;
static CUtlVector< BombSuppressorHandle_t > s_BombSuppressors;
};
enum
{
CHUNK_COCKPIT,
CHUNK_BODY,
CHUNK_TAIL
};
//-----------------------------------------------------------------------------
// This entity is used for helicopter gibs with specific properties
//-----------------------------------------------------------------------------
class CHelicopterChunk : public CBaseAnimating
{
DECLARE_DATADESC();
public:
DECLARE_CLASS( CHelicopterChunk, CBaseAnimating );
virtual void Spawn( void );
virtual void VPhysicsCollision( int index, gamevcollisionevent_t *pEvent );
static CHelicopterChunk *CreateHelicopterChunk( const Vector &vecPos, const QAngle &vecAngles, const Vector &vecVelocity, const char *pszModelName, int chunkID );
int m_nChunkID;
CHandle<CHelicopterChunk> m_hMaster;
IPhysicsConstraint *m_pTailConstraint;
IPhysicsConstraint *m_pCockpitConstraint;
protected:
void CollisionCallback( CHelicopterChunk *pCaller );
void FallThink( void );
bool m_bLanded;
};
//-----------------------------------------------------------------------------
// The attack helicopter
//-----------------------------------------------------------------------------
class CNPC_AttackHelicopter : public CBaseHelicopter
{
public:
DECLARE_CLASS( CNPC_AttackHelicopter, CBaseHelicopter );
DECLARE_DATADESC();
DEFINE_CUSTOM_AI;
CNPC_AttackHelicopter();
~CNPC_AttackHelicopter();
virtual void Precache( void );
virtual void Spawn( void );
virtual void Activate( void );
virtual bool CreateComponents();
virtual int ObjectCaps();
#ifdef HL2_EPISODIC
virtual bool CreateVPhysics( void );
#endif // HL2_EPISODIC
virtual void UpdateOnRemove();
virtual void StopLoopingSounds();
int BloodColor( void ) { return DONT_BLEED; }
Class_T Classify ( void ) { return CLASS_COMBINE_GUNSHIP; }
virtual int OnTakeDamage_Alive( const CTakeDamageInfo &info );
virtual void TraceAttack( const CTakeDamageInfo &info, const Vector &vecDir, trace_t *ptr, CDmgAccumulator *pAccumulator );
virtual int OnTakeDamage( const CTakeDamageInfo &info );
// Shot spread
virtual Vector GetAttackSpread( CBaseCombatWeapon *pWeapon, CBaseEntity *pTarget );
// More Enemy visibility check
virtual bool FVisible( CBaseEntity *pEntity, int traceMask = MASK_BLOCKLOS, CBaseEntity **ppBlocker = NULL );
// Think!
virtual void PrescheduleThink( void );
// Purpose: Set the gunship's paddles flailing!
virtual void Event_Killed( const CTakeDamageInfo &info );
// Drop a bomb at a particular location
void InputDropBomb( inputdata_t &inputdata );
void InputDropBombStraightDown( inputdata_t &inputdata );
void InputDropBombAtTarget( inputdata_t &inputdata );
void InputDropBombAtTargetAlways( inputdata_t &inputdata );
void InputDropBombAtTargetInternal( inputdata_t &inputdata, bool bCheckFairness );
void InputDropBombDelay( inputdata_t &inputdata );
void InputStartCarpetBombing( inputdata_t &inputdata );
void InputStopCarpetBombing( inputdata_t &inputdata );
virtual void SetTransmit( CCheckTransmitInfo *pInfo, bool bAlways );
virtual const char *GetTracerType( void );
virtual void DoImpactEffect( trace_t &tr, int nDamageType );
virtual void DoMuzzleFlash( void );
// FIXME: Work this back into the base class
virtual bool ShouldUseFixedPatrolLogic() { return true; }
protected:
int m_poseWeapon_Pitch, m_poseWeapon_Yaw, m_poseRudder;
virtual void PopulatePoseParameters( void );
private:
enum GunState_t
{
GUN_STATE_IDLE = 0,
GUN_STATE_CHARGING,
GUN_STATE_FIRING,
};
// Gets the max speed of the helicopter
virtual float GetMaxSpeed();
virtual float GetMaxSpeedFiring();
// Startup the chopper
virtual void Startup();
void InitializeRotorSound( void );
// Weaponry
bool FireGun( void );
// Movement:
virtual void Flight( void );
// Changes the main thinking method of helicopters
virtual void Hunt( void );
// For scripted times where it *has* to shoot
void InputResetIdleTime( inputdata_t &inputdata );
void InputSetHealthFraction( inputdata_t &inputdata );
void InputStartBombExplodeOnContact( inputdata_t &inputdata );
void InputStopBombExplodeOnContact( inputdata_t &inputdata );
void InputEnableAlwaysTransition( inputdata_t &inputdata );
void InputDisableAlwaysTransition( inputdata_t &inputdata );
void InputOutsideTransition( inputdata_t &inputdata );
void InputSetOutsideTransitionTarget( inputdata_t &inputdata );
// Turns off the gun
void InputGunOff( inputdata_t &inputdata );
// Vehicle attack modes
void InputStartBombingVehicle( inputdata_t &inputdata );
void InputStartTrailingVehicle( inputdata_t &inputdata );
void InputStartDefaultBehavior( inputdata_t &inputdata );
void InputStartAlwaysLeadingVehicle( inputdata_t &inputdata );
// Deadly shooting, tex!
void InputEnableDeadlyShooting( inputdata_t &inputdata );
void InputDisableDeadlyShooting( inputdata_t &inputdata );
void InputStartNormalShooting( inputdata_t &inputdata );
void InputStartLongCycleShooting( inputdata_t &inputdata );
void InputStartContinuousShooting( inputdata_t &inputdata );
void InputStartFastShooting( inputdata_t &inputdata );
int GetShootingMode( );
bool IsDeadlyShooting();
// Bombing suppression
void InputEnableBombing( inputdata_t &inputdata );
void InputDisableBombing( inputdata_t &inputdata );
// Visibility tests
void InputDisablePathVisibilityTests( inputdata_t &inputdata );
void InputEnablePathVisibilityTests( inputdata_t &inputdata );
// Death, etc.
void InputSelfDestruct( inputdata_t &inputdata );
// Enemy visibility check
CBaseEntity *FindTrackBlocker( const Vector &vecViewPoint, const Vector &vecTargetPos );
// Special path navigation when we explicitly want to follow a path
void UpdateFollowPathNavigation();
// Find interesting nearby things to shoot
int BuildMissTargetList( int nCount, CBaseEntity **ppMissCandidates );
// Shoot when the player's your enemy :
void ShootAtPlayer( const Vector &vBasePos, const Vector &vGunDir );
// Shoot when the player's your enemy + he's driving a vehicle
void ShootAtVehicle( const Vector &vBasePos, const Vector &vGunDir );
// Shoot where we're facing
void ShootAtFacingDirection( const Vector &vBasePos, const Vector &vGunDir, bool bFirstShotAccurate );
// Updates the facing direction
void UpdateFacingDirection( const Vector &vecActualDesiredPosition );
// Various states of the helicopter firing...
bool PoseGunTowardTargetDirection( const Vector &vTargetDir );
// Compute the position to fire at (vehicle + non-vehicle case)
void ComputeFireAtPosition( Vector *pVecActualTargetPosition );
void ComputeVehicleFireAtPosition( Vector *pVecActualTargetPosition );
// Various states of the helicopter firing...
bool DoGunIdle( const Vector &vecGunDir, const Vector &vTargetDir );
bool DoGunCharging( );
bool DoGunFiring( const Vector &vBasePos, const Vector &vGunDir, const Vector &vecFireAtPosition );
void FireElectricityGun( );
// Chooses a point within the circle of death to fire in
void PickDirectionToCircleOfDeath( const Vector &vBasePos, const Vector &vecFireAtPosition, Vector *pResult );
// Gets a vehicle the enemy is in (if any)
CBaseEntity *GetEnemyVehicle();
// Updates the perpendicular path distance for the chopper
float UpdatePerpPathDistance( float flMaxPathOffset );
// Purpose :
void UpdateEnemyLeading( void );
// Drop those bombs!
void DropBombs( );
// Should we drop those bombs?
bool ShouldDropBombs( void );
// Returns the max firing distance
float GetMaxFiringDistance();
// Make sure we don't hit too many times
void FireBullets( const FireBulletsInfo_t &info );
// Is it "fair" to drop this bomb?
bool IsBombDropFair( const Vector &vecBombStartPos, const Vector &vecVelocity );
// Actually drops the bomb
void CreateBomb( bool bCheckForFairness = true, Vector *pVecVelocity = NULL, bool bMegaBomb = false );
CGrenadeHelicopter *SpawnBombEntity( const Vector &vecPos, const Vector &vecVelocity ); // Spawns the bomb entity and sets it up
// Deliberately aims as close as possible w/o hitting
void AimCloseToTargetButMiss( CBaseEntity *pTarget, float flMinDist, float flMaxDist, const Vector &shootOrigin, Vector *pResult );
// Pops a shot inside the circle of death using the burst rules
void ShootInsideCircleOfDeath( const Vector &vBasePos, const Vector &vecFireAtPosition );
// How easy is the target to hit?
void UpdateTargetHittability();
// Add a smoke trail since we've taken more damage
void AddSmokeTrail( const Vector &vecPos );
// Destroy all smoke trails
void DestroySmokeTrails();
// Creates the breakable husk of an attack chopper
void CreateChopperHusk();
// Pow!
void ExplodeAndThrowChunk( const Vector &vecExplosionPos );
// Drop a corpse!
void DropCorpse( int nDamage );
// Should we trigger a damage effect?
bool ShouldTriggerDamageEffect( int nPrevHealth, int nEffectCount ) const;
// Become indestructible
void InputBecomeIndestructible( inputdata_t &inputdata );
// Purpose :
float CreepTowardEnemy( float flSpeed, float flMinSpeed, float flMaxSpeed, float flMinDist, float flMaxDist );
// Start bullrush
void InputStartBullrushBehavior( inputdata_t &inputdata );
void GetMaxSpeedAndAccel( float *pMaxSpeed, float *pAccelRate );
void ComputeAngularVelocity( const Vector &vecGoalUp, const Vector &vecFacingDirection );
void ComputeVelocity( const Vector &deltaPos, float flAdditionalHeight, float flMinDistFromSegment, float flMaxDistFromSegment, Vector *pVecAccel );
void FlightDirectlyOverhead( void );
// Methods related to computing leading distance
float ComputeBombingLeadingDistance( float flSpeed, float flSpeedAlongPath, bool bEnemyInVehicle );
float ComputeBullrushLeadingDistance( float flSpeed, float flSpeedAlongPath, bool bEnemyInVehicle );
bool IsCarpetBombing() { return m_bIsCarpetBombing == true; }
// Update the bullrush state
void UpdateBullrushState( void );
// Whether to shoot at or bomb an idle player
bool ShouldBombIdlePlayer( void );
// Different bomb-dropping behavior
void BullrushBombs( );
// Switch to idle
void SwitchToBullrushIdle( void );
// Secondary mode
void SetSecondaryMode( int nMode, bool bRetainTime = false );
bool IsInSecondaryMode( int nMode );
float SecondaryModeTime( ) const;
// Should the chopper shoot the idle player?
bool ShouldShootIdlePlayerInBullrush();
// Shutdown shooting during bullrush
void ShutdownGunDuringBullrush( );
// Updates the enemy
virtual float EnemySearchDistance( );
// Prototype zapper
bool IsValidZapTarget( CBaseEntity *pTarget );
void CreateZapBeam( const Vector &vecTargetPos );
void CreateEntityZapEffect( CBaseEntity *pEnt );
// Blink lights
void BlinkLightsThink();
// Spotlights
void SpotlightThink();
void SpotlightStartup();
void SpotlightShutdown();
CBaseEntity *GetCrashPoint() { return m_hCrashPoint.Get(); }
private:
enum
{
ATTACK_MODE_DEFAULT = 0,
ATTACK_MODE_BOMB_VEHICLE,
ATTACK_MODE_TRAIL_VEHICLE,
ATTACK_MODE_ALWAYS_LEAD_VEHICLE,
ATTACK_MODE_BULLRUSH_VEHICLE,
};
enum
{
MAX_SMOKE_TRAILS = 5,
MAX_EXPLOSIONS = 13,
MAX_CORPSES = 2,
};
enum
{
BULLRUSH_MODE_WAIT_FOR_ENEMY = 0,
BULLRUSH_MODE_GET_DISTANCE,
BULLRUSH_MODE_DROP_BOMBS_FIXED_SPEED,
BULLRUSH_MODE_DROP_BOMBS_FOLLOW_PLAYER,
BULLRUSH_MODE_SHOOT_GUN,
BULLRUSH_MODE_MEGA_BOMB,
BULLRUSH_MODE_SHOOT_IDLE_PLAYER,
};
enum
{
SHOOT_MODE_DEFAULT = 0,
SHOOT_MODE_LONG_CYCLE,
SHOOT_MODE_CONTINUOUS,
SHOOT_MODE_FAST,
};
#ifdef HL2_EPISODIC
void InitBoneFollowers( void );
CBoneFollowerManager m_BoneFollowerManager;
#endif // HL2_EPISODIC
CAI_Spotlight m_Spotlight;
Vector m_angGun;
QAngle m_vecAngAcceleration;
int m_iAmmoType;
float m_flLastCorpseFall;
GunState_t m_nGunState;
float m_flChargeTime;
float m_flIdleTimeDelay;
int m_nRemainingBursts;
int m_nGrenadeCount;
float m_flPathOffset;
float m_flAcrossTime;
float m_flCurrPathOffset;
int m_nBurstHits;
int m_nMaxBurstHits;
float m_flCircleOfDeathRadius;
int m_nAttackMode;
float m_flInputDropBombTime;
CHandle<CBombDropSensor> m_hSensor;
float m_flAvoidMetric;
AngularImpulse m_vecLastAngVelocity;
CHandle<CBaseEntity> m_hSmokeTrail[MAX_SMOKE_TRAILS];
int m_nSmokeTrailCount;
bool m_bIndestructible;
float m_flGracePeriod;
bool m_bBombsExplodeOnContact;
bool m_bNonCombat;
int m_nNearShots;
int m_nMaxNearShots;
// Bomb dropping attachments
int m_nGunTipAttachment;
int m_nGunBaseAttachment;
int m_nBombAttachment;
int m_nSpotlightAttachment;
float m_flLastFastTime;
// Secondary modes
int m_nSecondaryMode;
float m_flSecondaryModeStartTime;
// Bullrush behavior
bool m_bRushForward;
float m_flBullrushAdditionalHeight;
int m_nBullrushBombMode;
float m_flNextBullrushBombTime;
float m_flNextMegaBombHealth;
// Shooting method
int m_nShootingMode;
bool m_bDeadlyShooting;
// Bombing suppression
bool m_bBombingSuppressed;
// Blinking lights
CHandle<CSprite> m_hLights[MAX_HELICOPTER_LIGHTS];
bool m_bShortBlink;
// Path behavior
bool m_bIgnorePathVisibilityTests;
// Teleport
bool m_bAlwaysTransition;
string_t m_iszTransitionTarget;
// Special attacks
bool m_bIsCarpetBombing;
// Fun damage effects
float m_flGoalRollDmg;
float m_flGoalYawDmg;
// Sounds
CSoundPatch *m_pGunFiringSound;
// Outputs
COutputInt m_OnHealthChanged;
COutputEvent m_OnShotDown;
// Crashing
EHANDLE m_hCrashPoint;
};
#ifdef HL2_EPISODIC
static const char *pFollowerBoneNames[] =
{
"Chopper.Body"
};
#endif // HL2_EPISODIC
LINK_ENTITY_TO_CLASS( npc_helicopter, CNPC_AttackHelicopter );
BEGIN_DATADESC( CNPC_AttackHelicopter )
DEFINE_ENTITYFUNC( FlyTouch ),
DEFINE_EMBEDDED( m_Spotlight ),
#ifdef HL2_EPISODIC
DEFINE_EMBEDDED( m_BoneFollowerManager ),
#endif
DEFINE_FIELD( m_angGun, FIELD_VECTOR ),
DEFINE_FIELD( m_vecAngAcceleration, FIELD_VECTOR ),
DEFINE_FIELD( m_iAmmoType, FIELD_INTEGER ),
DEFINE_FIELD( m_flLastCorpseFall, FIELD_TIME ),
DEFINE_FIELD( m_nGunState, FIELD_INTEGER ),
DEFINE_FIELD( m_flChargeTime, FIELD_TIME ),
DEFINE_FIELD( m_flIdleTimeDelay, FIELD_FLOAT ),
DEFINE_FIELD( m_nRemainingBursts, FIELD_INTEGER ),
DEFINE_FIELD( m_nGrenadeCount, FIELD_INTEGER ),
DEFINE_FIELD( m_flPathOffset, FIELD_FLOAT ),
DEFINE_FIELD( m_flAcrossTime, FIELD_TIME ),
DEFINE_FIELD( m_flCurrPathOffset, FIELD_FLOAT ),
DEFINE_FIELD( m_nBurstHits, FIELD_INTEGER ),
DEFINE_FIELD( m_nMaxBurstHits, FIELD_INTEGER ),
DEFINE_FIELD( m_flCircleOfDeathRadius, FIELD_FLOAT ),
DEFINE_FIELD( m_nAttackMode, FIELD_INTEGER ),
DEFINE_FIELD( m_flInputDropBombTime, FIELD_TIME ),
DEFINE_FIELD( m_hSensor, FIELD_EHANDLE ),
DEFINE_FIELD( m_flAvoidMetric, FIELD_FLOAT ),
DEFINE_FIELD( m_vecLastAngVelocity, FIELD_VECTOR ),
DEFINE_AUTO_ARRAY( m_hSmokeTrail, FIELD_EHANDLE ),
DEFINE_FIELD( m_nSmokeTrailCount, FIELD_INTEGER ),
DEFINE_FIELD( m_nNearShots, FIELD_INTEGER ),
DEFINE_FIELD( m_nMaxNearShots, FIELD_INTEGER ),
// DEFINE_FIELD( m_nGunTipAttachment, FIELD_INTEGER ),
// DEFINE_FIELD( m_nGunBaseAttachment, FIELD_INTEGER ),
// DEFINE_FIELD( m_nBombAttachment, FIELD_INTEGER ),
// DEFINE_FIELD( m_nSpotlightAttachment, FIELD_INTEGER ),
DEFINE_FIELD( m_flLastFastTime, FIELD_TIME ),
DEFINE_FIELD( m_nSecondaryMode, FIELD_INTEGER ),
DEFINE_FIELD( m_flSecondaryModeStartTime, FIELD_TIME ),
DEFINE_FIELD( m_bRushForward, FIELD_BOOLEAN ),
DEFINE_FIELD( m_flBullrushAdditionalHeight, FIELD_FLOAT ),
DEFINE_FIELD( m_nBullrushBombMode, FIELD_INTEGER ),
DEFINE_FIELD( m_flNextBullrushBombTime, FIELD_TIME ),
DEFINE_FIELD( m_flNextMegaBombHealth, FIELD_FLOAT ),
DEFINE_FIELD( m_nShootingMode, FIELD_INTEGER ),
DEFINE_FIELD( m_bDeadlyShooting, FIELD_BOOLEAN ),
DEFINE_FIELD( m_bBombingSuppressed, FIELD_BOOLEAN ),
DEFINE_SOUNDPATCH( m_pGunFiringSound ),
DEFINE_AUTO_ARRAY( m_hLights, FIELD_EHANDLE ),
DEFINE_FIELD( m_bIgnorePathVisibilityTests, FIELD_BOOLEAN ),
DEFINE_FIELD( m_bShortBlink, FIELD_BOOLEAN ),
DEFINE_FIELD( m_bIndestructible, FIELD_BOOLEAN ),
DEFINE_FIELD( m_bBombsExplodeOnContact, FIELD_BOOLEAN ),
DEFINE_KEYFIELD( m_bAlwaysTransition, FIELD_BOOLEAN, "AlwaysTransition" ),
DEFINE_KEYFIELD( m_iszTransitionTarget, FIELD_STRING, "TransitionTarget" ),
DEFINE_FIELD( m_bIsCarpetBombing, FIELD_BOOLEAN ),
DEFINE_INPUTFUNC( FIELD_VOID, "EnableAlwaysTransition", InputEnableAlwaysTransition ),
DEFINE_INPUTFUNC( FIELD_VOID, "DisableAlwaysTransition", InputDisableAlwaysTransition ),
DEFINE_INPUTFUNC( FIELD_VOID, "OutsideTransition", InputOutsideTransition ),
DEFINE_INPUTFUNC( FIELD_STRING, "SetTransitionTarget", InputSetOutsideTransitionTarget ),
DEFINE_KEYFIELD( m_flGracePeriod, FIELD_FLOAT, "GracePeriod" ),
DEFINE_KEYFIELD( m_flMaxSpeed, FIELD_FLOAT, "PatrolSpeed" ),
DEFINE_KEYFIELD( m_bNonCombat, FIELD_BOOLEAN, "NonCombat" ),
DEFINE_FIELD( m_hCrashPoint, FIELD_EHANDLE ),
DEFINE_INPUTFUNC( FIELD_VOID, "ResetIdleTime", InputResetIdleTime ),
DEFINE_INPUTFUNC( FIELD_VOID, "StartAlwaysLeadingVehicle", InputStartAlwaysLeadingVehicle ),
DEFINE_INPUTFUNC( FIELD_VOID, "StartBombingVehicle", InputStartBombingVehicle ),
DEFINE_INPUTFUNC( FIELD_VOID, "StartTrailingVehicle", InputStartTrailingVehicle ),
DEFINE_INPUTFUNC( FIELD_VOID, "StartDefaultBehavior", InputStartDefaultBehavior ),
DEFINE_INPUTFUNC( FIELD_VOID, "StartBullrushBehavior", InputStartBullrushBehavior ),
DEFINE_INPUTFUNC( FIELD_VOID, "DropBomb", InputDropBomb ),
DEFINE_INPUTFUNC( FIELD_VOID, "DropBombStraightDown", InputDropBombStraightDown ),
DEFINE_INPUTFUNC( FIELD_STRING, "DropBombAtTargetAlways", InputDropBombAtTargetAlways ),
DEFINE_INPUTFUNC( FIELD_STRING, "DropBombAtTarget", InputDropBombAtTarget ),
DEFINE_INPUTFUNC( FIELD_FLOAT, "DropBombDelay", InputDropBombDelay ),
DEFINE_INPUTFUNC( FIELD_VOID, "StartCarpetBombing", InputStartCarpetBombing ),
DEFINE_INPUTFUNC( FIELD_VOID, "StopCarpetBombing", InputStopCarpetBombing ),
DEFINE_INPUTFUNC( FIELD_VOID, "BecomeIndestructible", InputBecomeIndestructible ),
DEFINE_INPUTFUNC( FIELD_VOID, "EnableDeadlyShooting", InputEnableDeadlyShooting ),
DEFINE_INPUTFUNC( FIELD_VOID, "DisableDeadlyShooting", InputDisableDeadlyShooting ),
DEFINE_INPUTFUNC( FIELD_VOID, "StartNormalShooting", InputStartNormalShooting ),
DEFINE_INPUTFUNC( FIELD_VOID, "StartLongCycleShooting", InputStartLongCycleShooting ),
DEFINE_INPUTFUNC( FIELD_VOID, "StartContinuousShooting", InputStartContinuousShooting ),
DEFINE_INPUTFUNC( FIELD_VOID, "StartFastShooting", InputStartFastShooting ),
DEFINE_INPUTFUNC( FIELD_VOID, "GunOff", InputGunOff ),
DEFINE_INPUTFUNC( FIELD_FLOAT, "SetHealthFraction", InputSetHealthFraction ),
DEFINE_INPUTFUNC( FIELD_VOID, "StartBombExplodeOnContact", InputStartBombExplodeOnContact ),
DEFINE_INPUTFUNC( FIELD_VOID, "StopBombExplodeOnContact", InputStopBombExplodeOnContact ),
DEFINE_INPUTFUNC( FIELD_VOID, "DisablePathVisibilityTests", InputDisablePathVisibilityTests ),
DEFINE_INPUTFUNC( FIELD_VOID, "EnablePathVisibilityTests", InputEnablePathVisibilityTests ),
DEFINE_INPUTFUNC( FIELD_VOID, "SelfDestruct", InputSelfDestruct ),
DEFINE_THINKFUNC( BlinkLightsThink ),
DEFINE_THINKFUNC( SpotlightThink ),
DEFINE_OUTPUT( m_OnHealthChanged, "OnHealthChanged" ),
DEFINE_OUTPUT( m_OnShotDown, "OnShotDown" ),
END_DATADESC()
//------------------------------------------------------------------------------
// Purpose :
//------------------------------------------------------------------------------
CNPC_AttackHelicopter::CNPC_AttackHelicopter() :
m_bNonCombat( false ),
m_flGracePeriod( 2.0f ),
m_bBombsExplodeOnContact( false )
{
m_flMaxSpeed = 0;
}
CNPC_AttackHelicopter::~CNPC_AttackHelicopter(void)
{
}
//-----------------------------------------------------------------------------
// Purpose: Shuts down looping sounds when we are killed in combat or deleted.
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::StopLoopingSounds()
{
BaseClass::StopLoopingSounds();
if ( m_pGunFiringSound )
{
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
controller.SoundDestroy( m_pGunFiringSound );
m_pGunFiringSound = NULL;
}
}
//------------------------------------------------------------------------------
// Purpose :
//------------------------------------------------------------------------------
void Chopper_PrecacheChunks( CBaseEntity *pChopper )
{
for ( int i = 0; i < CHOPPER_MAX_CHUNKS; ++i )
{
pChopper->PrecacheModel( s_pChunkModelName[i] );
}
pChopper->PrecacheModel( HELICOPTER_CHUNK_COCKPIT );
pChopper->PrecacheModel( HELICOPTER_CHUNK_TAIL );
pChopper->PrecacheModel( HELICOPTER_CHUNK_BODY );
}
//------------------------------------------------------------------------------
// Purpose :
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::Precache( void )
{
BaseClass::Precache();
if ( !HasSpawnFlags(SF_HELICOPTER_ELECTRICAL_DRONE) )
{
PrecacheModel( CHOPPER_MODEL_NAME );
}
else
{
PrecacheModel( CHOPPER_DRONE_NAME );
}
PrecacheModel( CHOPPER_RED_LIGHT_SPRITE );
//PrecacheModel( CHOPPER_MODEL_CORPSE_NAME );
// If we're never going to engage in combat, we don't need to load these assets!
if ( m_bNonCombat == false )
{
UTIL_PrecacheOther( "grenade_helicopter" );
UTIL_PrecacheOther( "env_fire_trail" );
Chopper_PrecacheChunks( this );
PrecacheModel("models/combine_soldier.mdl");
}
PrecacheScriptSound("NPC_AttackHelicopter.ChargeGun");
if ( HasSpawnFlags( SF_HELICOPTER_LOUD_ROTOR_SOUND ) )
{
PrecacheScriptSound("NPC_AttackHelicopter.RotorsLoud");
}
else
{
PrecacheScriptSound("NPC_AttackHelicopter.Rotors");
}
PrecacheScriptSound( "NPC_AttackHelicopter.DropMine" );
PrecacheScriptSound( "NPC_AttackHelicopter.BadlyDamagedAlert" );
PrecacheScriptSound( "NPC_AttackHelicopter.CrashingAlarm1" );
PrecacheScriptSound( "NPC_AttackHelicopter.MegabombAlert" );
PrecacheScriptSound( "NPC_AttackHelicopter.RotorBlast" );
PrecacheScriptSound( "NPC_AttackHelicopter.EngineFailure" );
PrecacheScriptSound( "NPC_AttackHelicopter.FireGun" );
PrecacheScriptSound( "NPC_AttackHelicopter.Crash" );
PrecacheScriptSound( "HelicopterBomb.HardImpact" );
PrecacheScriptSound( "ReallyLoudSpark" );
PrecacheScriptSound( "NPC_AttackHelicopterGrenade.Ping" );
}
int CNPC_AttackHelicopter::ObjectCaps()
{
int caps = BaseClass::ObjectCaps();
if ( m_bAlwaysTransition )
caps |= FCAP_NOTIFY_ON_TRANSITION;
return caps;
}
void CNPC_AttackHelicopter::InputOutsideTransition( inputdata_t &inputdata )
{
CBaseEntity *pEnt = gEntList.FindEntityByName( NULL, m_iszTransitionTarget );
if ( pEnt )
{
Vector teleportLocation = pEnt->GetAbsOrigin();
QAngle teleportAngles = pEnt->GetAbsAngles();
Teleport( &teleportLocation, &teleportAngles, &vec3_origin );
Teleported();
}
else
{
DevMsg( 2, "NPC \"%s\" failed to find a suitable transition a point\n", STRING(GetEntityName()) );
}
}
void CNPC_AttackHelicopter::InputSetOutsideTransitionTarget( inputdata_t &inputdata )
{
m_iszTransitionTarget = MAKE_STRING( inputdata.value.String() );
}
//-----------------------------------------------------------------------------
// Create components
//-----------------------------------------------------------------------------
bool CNPC_AttackHelicopter::CreateComponents()
{
if ( !BaseClass::CreateComponents() )
return false;
m_Spotlight.Init( this, AI_SPOTLIGHT_NO_DLIGHTS, 45.0f, 500.0f );
return true;
}
//-----------------------------------------------------------------------------
// Purpose :
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::Spawn( void )
{
Precache( );
m_bIndestructible = false;
m_bDeadlyShooting = false;
m_bBombingSuppressed = false;
m_bIgnorePathVisibilityTests = false;
if ( !HasSpawnFlags(SF_HELICOPTER_ELECTRICAL_DRONE) )
{
SetModel( CHOPPER_MODEL_NAME );
}
else
{
SetModel( CHOPPER_DRONE_NAME );
}
ExtractBbox( SelectHeaviestSequence( ACT_IDLE ), m_cullBoxMins, m_cullBoxMaxs );
GetEnemies()->SetFreeKnowledgeDuration( DEFAULT_FREE_KNOWLEDGE_DURATION );
float flLoadedSpeed = m_flMaxSpeed;
BaseClass::Spawn();
float flChaseDist = HasSpawnFlags( SF_HELICOPTER_AGGRESSIVE ) ?
CHOPPER_MIN_AGGRESSIVE_CHASE_DIST_DIFF : CHOPPER_MIN_CHASE_DIST_DIFF;
InitPathingData( CHOPPER_ARRIVE_DIST, flChaseDist, CHOPPER_AVOID_DIST );
SetFarthestPathDist( GetMaxFiringDistance() );
m_takedamage = DAMAGE_YES;
m_nGunState = GUN_STATE_IDLE;
SetHullType( HULL_LARGE_CENTERED );
SetHullSizeNormal();
#ifdef HL2_EPISODIC
CreateVPhysics();
#endif // HL2_EPISODIC
SetPauseState( PAUSE_NO_PAUSE );
m_iMaxHealth = m_iHealth = sk_helicopter_health.GetInt();
m_flMaxSpeed = flLoadedSpeed;
if ( m_flMaxSpeed <= 0 )
{
m_flMaxSpeed = CHOPPER_MAX_SPEED;
}
m_flNextMegaBombHealth = m_iMaxHealth - m_iMaxHealth * g_helicopter_bullrush_mega_bomb_health.GetFloat();
m_nGrenadeCount = CHOPPER_BOMB_DROP_COUNT;
m_flFieldOfView = -1.0; // 360 degrees
m_flIdleTimeDelay = 0.0f;
m_iAmmoType = GetAmmoDef()->Index("HelicopterGun");
InitBoneControllers();
m_fHelicopterFlags = BITS_HELICOPTER_GUN_ON;
m_bSuppressSound = false;
m_flAcrossTime = -1.0f;
m_flPathOffset = 0.0f;
m_flCurrPathOffset = 0.0f;
m_nAttackMode = ATTACK_MODE_DEFAULT;
m_flInputDropBombTime = gpGlobals->curtime;
SetActivity( ACT_IDLE );
int nBombAttachment = LookupAttachment("bomb");
m_hSensor = static_cast<CBombDropSensor*>(CreateEntityByName( "npc_helicoptersensor" ));
m_hSensor->Spawn();
m_hSensor->SetParent( this, nBombAttachment );
m_hSensor->SetLocalOrigin( vec3_origin );
m_hSensor->SetLocalAngles( vec3_angle );
m_hSensor->SetOwnerEntity( this );
AddFlag( FL_AIMTARGET );
m_hCrashPoint.Set( NULL );
}
#ifdef HL2_EPISODIC
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
bool CNPC_AttackHelicopter::CreateVPhysics( void )
{
InitBoneFollowers();
return BaseClass::CreateVPhysics();
}
#endif // HL2_EPISODIC
//------------------------------------------------------------------------------
// Startup the chopper
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::Startup()
{
BaseClass::Startup();
if ( HasSpawnFlags( SF_HELICOPTER_LIGHTS ) )
{
for ( int i = 0; i < MAX_HELICOPTER_LIGHTS; ++i )
{
// See if there's an attachment for this smoke trail
char buf[32];
Q_snprintf( buf, 32, "Light_Red%d", i );
int nAttachment = LookupAttachment( buf );
if ( nAttachment == 0 )
{
m_hLights[i] = NULL;
continue;
}
m_hLights[i] = CSprite::SpriteCreate( CHOPPER_RED_LIGHT_SPRITE, vec3_origin, false );
if ( !m_hLights[i] )
continue;
m_hLights[i]->SetParent( this, nAttachment );
m_hLights[i]->SetLocalOrigin( vec3_origin );
m_hLights[i]->SetLocalVelocity( vec3_origin );
m_hLights[i]->SetMoveType( MOVETYPE_NONE );
m_hLights[i]->SetTransparency( kRenderTransAdd, 255, 255, 255, 200, kRenderFxNone );
m_hLights[i]->SetScale( 1.0f );
m_hLights[i]->TurnOn();
}
SetContextThink( &CNPC_AttackHelicopter::BlinkLightsThink, gpGlobals->curtime + CHOPPER_LIGHT_BLINK_TIME_SHORT, s_pBlinkLightThinkContext );
}
}
//------------------------------------------------------------------------------
// Startup the chopper
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::BlinkLightsThink()
{
bool bIsOn = false;
for ( int i = 0; i < MAX_HELICOPTER_LIGHTS; ++i )
{
if ( !m_hLights[i] )
continue;
if ( m_hLights[i]->GetScale() > 0.1f )
{
m_hLights[i]->SetScale( 0.1f, CHOPPER_LIGHT_BLINK_TIME_SHORT );
}
else
{
m_hLights[i]->SetScale( 0.5f, 0.0f );
bIsOn = true;
}
}
float flTime;
if ( bIsOn )
{
flTime = CHOPPER_LIGHT_BLINK_TIME_SHORT;
}
else
{
flTime = m_bShortBlink ? CHOPPER_LIGHT_BLINK_TIME_SHORT : CHOPPER_LIGHT_BLINK_TIME;
m_bShortBlink = !m_bShortBlink;
}
SetContextThink( &CNPC_AttackHelicopter::BlinkLightsThink, gpGlobals->curtime + flTime, s_pBlinkLightThinkContext );
}
//------------------------------------------------------------------------------
// Start up spotlights
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::SpotlightStartup()
{
if ( !HasSpawnFlags( SF_HELICOPTER_LIGHTS ) )
return;
Vector vecForward;
Vector vecOrigin;
GetAttachment( m_nSpotlightAttachment, vecOrigin, &vecForward );
m_Spotlight.SpotlightCreate( m_nSpotlightAttachment, vecForward );
SpotlightThink();
}
//------------------------------------------------------------------------------
// Shutdown spotlights
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::SpotlightShutdown()
{
m_Spotlight.SpotlightDestroy();
SetContextThink( NULL, gpGlobals->curtime, s_pSpotlightThinkContext );
}
//------------------------------------------------------------------------------
// Spotlights
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::SpotlightThink()
{
// NOTE: This function should deal with all deactivation cases
if ( m_lifeState != LIFE_ALIVE )
{
SpotlightShutdown();
return;
}
switch( m_nAttackMode )
{
case ATTACK_MODE_BULLRUSH_VEHICLE:
{
switch ( m_nSecondaryMode )
{
case BULLRUSH_MODE_SHOOT_GUN:
{
Vector vecForward;
Vector vecOrigin;
GetAttachment( m_nSpotlightAttachment, vecOrigin, &vecForward );
m_Spotlight.SetSpotlightTargetDirection( vecForward );
}
break;
case BULLRUSH_MODE_SHOOT_IDLE_PLAYER:
if ( GetEnemy() )
{
m_Spotlight.SetSpotlightTargetPos( GetEnemy()->WorldSpaceCenter() );
}
break;
default:
SpotlightShutdown();
return;
}
}
break;
default:
SpotlightShutdown();
return;
}
m_Spotlight.Update();
SetContextThink( &CNPC_AttackHelicopter::SpotlightThink, gpGlobals->curtime + TICK_INTERVAL, s_pSpotlightThinkContext );
}
//-----------------------------------------------------------------------------
// Purpose: Always transition along with the player
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::InputEnableAlwaysTransition( inputdata_t &inputdata )
{
m_bAlwaysTransition = true;
}
//-----------------------------------------------------------------------------
// Purpose: Stop always transitioning along with the player
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::InputDisableAlwaysTransition( inputdata_t &inputdata )
{
m_bAlwaysTransition = false;
}
//------------------------------------------------------------------------------
// On Remove
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::UpdateOnRemove()
{
BaseClass::UpdateOnRemove();
StopLoopingSounds();
UTIL_Remove(m_hSensor);
DestroySmokeTrails();
for ( int i = 0; i < MAX_HELICOPTER_LIGHTS; ++i )
{
if ( m_hLights[i] )
{
UTIL_Remove( m_hLights[i] );
m_hLights[i] = NULL;
}
}
#ifdef HL2_EPISODIC
m_BoneFollowerManager.DestroyBoneFollowers();
#endif // HL2_EPISODIC
}
//------------------------------------------------------------------------------
// Purpose :
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::Activate( void )
{
BaseClass::Activate();
m_nGunBaseAttachment = LookupAttachment("gun");
m_nGunTipAttachment = LookupAttachment("muzzle");
m_nBombAttachment = LookupAttachment("bomb");
m_nSpotlightAttachment = LookupAttachment("spotlight");
if ( HasSpawnFlags( SF_HELICOPTER_LONG_SHADOW ) )
{
SetShadowCastDistance( 2048 );
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
const char *CNPC_AttackHelicopter::GetTracerType( void )
{
return "HelicopterTracer";
}
//-----------------------------------------------------------------------------
// Allows the shooter to change the impact effect of his bullets
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::DoImpactEffect( trace_t &tr, int nDamageType )
{
UTIL_ImpactTrace( &tr, nDamageType, "HelicopterImpact" );
}
//------------------------------------------------------------------------------
// Purpose : Create our rotor sound
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::InitializeRotorSound( void )
{
if ( !m_pRotorSound )
{
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
CPASAttenuationFilter filter( this );
if ( HasSpawnFlags( SF_HELICOPTER_LOUD_ROTOR_SOUND ) )
{
m_pRotorSound = controller.SoundCreate( filter, entindex(), "NPC_AttackHelicopter.RotorsLoud" );
}
else
{
m_pRotorSound = controller.SoundCreate( filter, entindex(), "NPC_AttackHelicopter.Rotors" );
}
m_pRotorBlast = controller.SoundCreate( filter, entindex(), "NPC_AttackHelicopter.RotorBlast" );
m_pGunFiringSound = controller.SoundCreate( filter, entindex(), "NPC_AttackHelicopter.FireGun" );
controller.Play( m_pGunFiringSound, 0.0, 100 );
}
else
{
Assert(m_pRotorSound);
Assert(m_pRotorBlast);
Assert(m_pGunFiringSound);
}
BaseClass::InitializeRotorSound();
}
//------------------------------------------------------------------------------
// Gets the max speed of the helicopter
//------------------------------------------------------------------------------
float CNPC_AttackHelicopter::GetMaxSpeed()
{
if ( HasSpawnFlags(SF_HELICOPTER_ELECTRICAL_DRONE) )
return DRONE_SPEED;
if ( ( m_nAttackMode == ATTACK_MODE_BULLRUSH_VEHICLE ) && IsInSecondaryMode( BULLRUSH_MODE_DROP_BOMBS_FIXED_SPEED ) )
return CHOPPER_BULLRUSH_ENEMY_BOMB_SPEED;
if ( !GetEnemyVehicle() )
return BaseClass::GetMaxSpeed();
return 3000.0f;
}
float CNPC_AttackHelicopter::GetMaxSpeedFiring()
{
if ( HasSpawnFlags(SF_HELICOPTER_ELECTRICAL_DRONE) )
return DRONE_SPEED;
if ( ( m_nAttackMode == ATTACK_MODE_BULLRUSH_VEHICLE ) && IsInSecondaryMode( BULLRUSH_MODE_DROP_BOMBS_FIXED_SPEED ) )
return CHOPPER_BULLRUSH_ENEMY_BOMB_SPEED;
if ( !GetEnemyVehicle() )
return BaseClass::GetMaxSpeedFiring();
return 3000.0f;
}
//------------------------------------------------------------------------------
// Returns the max firing distance
//------------------------------------------------------------------------------
float CNPC_AttackHelicopter::GetMaxFiringDistance()
{
if ( !GetEnemyVehicle() )
return CHOPPER_GUN_MAX_FIRING_DIST;
return 8000.0f;
}
//------------------------------------------------------------------------------
// Updates the enemy
//------------------------------------------------------------------------------
float CNPC_AttackHelicopter::EnemySearchDistance( )
{
return 6000.0f;
}
//------------------------------------------------------------------------------
// Leading behaviors
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::InputStartBombingVehicle( inputdata_t &inputdata )
{
m_nAttackMode = ATTACK_MODE_BOMB_VEHICLE;
SetLeadingDistance( 1500.0f );
}
void CNPC_AttackHelicopter::InputStartTrailingVehicle( inputdata_t &inputdata )
{
m_nAttackMode = ATTACK_MODE_TRAIL_VEHICLE;
SetLeadingDistance( -1500.0f );
}
void CNPC_AttackHelicopter::InputStartDefaultBehavior( inputdata_t &inputdata )
{
m_nAttackMode = ATTACK_MODE_DEFAULT;
}
void CNPC_AttackHelicopter::InputStartAlwaysLeadingVehicle( inputdata_t &inputdata )
{
m_nAttackMode = ATTACK_MODE_ALWAYS_LEAD_VEHICLE;
SetLeadingDistance( 0.0f );
}
void CNPC_AttackHelicopter::InputStartBullrushBehavior( inputdata_t &inputdata )
{
if ( m_nAttackMode != ATTACK_MODE_BULLRUSH_VEHICLE )
{
m_nAttackMode = ATTACK_MODE_BULLRUSH_VEHICLE;
SetSecondaryMode( BULLRUSH_MODE_WAIT_FOR_ENEMY );
SetLeadingDistance( 0.0f );
}
}
//------------------------------------------------------------------------------
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::InputStartCarpetBombing( inputdata_t &inputdata )
{
m_bIsCarpetBombing = true;
}
//------------------------------------------------------------------------------
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::InputStopCarpetBombing( inputdata_t &inputdata )
{
m_bIsCarpetBombing = false;
}
//------------------------------------------------------------------------------
// Become indestructible
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::InputBecomeIndestructible( inputdata_t &inputdata )
{
m_bIndestructible = true;
}
//------------------------------------------------------------------------------
// Deadly shooting, tex!
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::InputEnableDeadlyShooting( inputdata_t &inputdata )
{
m_bDeadlyShooting = true;
}
void CNPC_AttackHelicopter::InputDisableDeadlyShooting( inputdata_t &inputdata )
{
m_bDeadlyShooting = false;
}
void CNPC_AttackHelicopter::InputStartNormalShooting( inputdata_t &inputdata )
{
m_nShootingMode = SHOOT_MODE_DEFAULT;
}
void CNPC_AttackHelicopter::InputStartLongCycleShooting( inputdata_t &inputdata )
{
m_nShootingMode = SHOOT_MODE_LONG_CYCLE;
}
void CNPC_AttackHelicopter::InputStartContinuousShooting( inputdata_t &inputdata )
{
m_nShootingMode = SHOOT_MODE_CONTINUOUS;
}
void CNPC_AttackHelicopter::InputStartFastShooting( inputdata_t &inputdata )
{
m_nShootingMode = SHOOT_MODE_FAST;
}
//------------------------------------------------------------------------------
// Deadly shooting, tex!
//------------------------------------------------------------------------------
bool CNPC_AttackHelicopter::IsDeadlyShooting()
{
if ( m_bDeadlyShooting )
return true;
if (( m_nAttackMode == ATTACK_MODE_BULLRUSH_VEHICLE ) && IsInSecondaryMode( BULLRUSH_MODE_SHOOT_IDLE_PLAYER ) )
{
return (!GetEnemyVehicle()) && GetEnemy() && GetEnemy()->IsPlayer();
}
return false;
}
int CNPC_AttackHelicopter::GetShootingMode( )
{
if ( IsDeadlyShooting() )
return SHOOT_MODE_LONG_CYCLE;
if ( m_nAttackMode == ATTACK_MODE_BULLRUSH_VEHICLE )
return SHOOT_MODE_CONTINUOUS;
return m_nShootingMode;
}
//-----------------------------------------------------------------------------
// Bombing suppression
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::InputEnableBombing( inputdata_t &inputdata )
{
m_bBombingSuppressed = false;
}
void CNPC_AttackHelicopter::InputDisableBombing( inputdata_t &inputdata )
{
m_bBombingSuppressed = true;
}
//-----------------------------------------------------------------------------
// Visibility tests
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::InputDisablePathVisibilityTests( inputdata_t &inputdata )
{
m_bIgnorePathVisibilityTests = true;
GetEnemies()->SetUnforgettable( GetEnemy(), true );
}
void CNPC_AttackHelicopter::InputEnablePathVisibilityTests( inputdata_t &inputdata )
{
m_bIgnorePathVisibilityTests = false;
GetEnemies()->SetUnforgettable( GetEnemy(), false );
}
//-----------------------------------------------------------------------------
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::InputSelfDestruct( inputdata_t &inputdata )
{
m_lifeState = LIFE_ALIVE; // Force to die properly.
CTakeDamageInfo info( this, this, Vector(0, 0, 1), WorldSpaceCenter(), GetMaxHealth(), CLASS_MISSILE );
TakeDamage( info );
}
//-----------------------------------------------------------------------------
// For scripted times where it *has* to shoot
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::InputSetHealthFraction( inputdata_t &inputdata )
{
// Sets the health fraction, no damage effects
if ( inputdata.value.Float() > 0 )
{
SetHealth( GetMaxHealth() * inputdata.value.Float() * 0.01f );
}
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : &inputdata -
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::InputStartBombExplodeOnContact( inputdata_t &inputdata )
{
m_bBombsExplodeOnContact = true;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : &inputdata -
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::InputStopBombExplodeOnContact( inputdata_t &inputdata )
{
m_bBombsExplodeOnContact = false;
}
//------------------------------------------------------------------------------
// For scripted times where it *has* to shoot
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::InputResetIdleTime( inputdata_t &inputdata )
{
if ( m_nGunState == GUN_STATE_IDLE )
{
m_flNextAttack = gpGlobals->curtime;
}
}
//-----------------------------------------------------------------------------
// This trace filter ignores all breakables + physics props
//-----------------------------------------------------------------------------
class CTraceFilterChopper : public CTraceFilterSimple
{
DECLARE_CLASS( CTraceFilterChopper, CTraceFilterSimple );
public:
CTraceFilterChopper( const IHandleEntity *passentity, int collisionGroup );
virtual bool ShouldHitEntity( IHandleEntity *pServerEntity, int contentsMask );
private:
const IHandleEntity *m_pPassEnt;
int m_collisionGroup;
};
CTraceFilterChopper::CTraceFilterChopper( const IHandleEntity *passentity, int collisionGroup ) :
CTraceFilterSimple( passentity, collisionGroup )
{
}
bool CTraceFilterChopper::ShouldHitEntity( IHandleEntity *pServerEntity, int contentsMask )
{
CBaseEntity *pEnt = static_cast<IServerUnknown*>(pServerEntity)->GetBaseEntity();
if ( pEnt )
{
if ( FClassnameIs( pEnt, "func_breakable" ) ||
FClassnameIs( pEnt, "func_physbox" ) ||
FClassnameIs( pEnt, "prop_physics" ) ||
FClassnameIs( pEnt, "physics_prop" ) )
{
return false;
}
}
return BaseClass::ShouldHitEntity( pServerEntity, contentsMask );
}
//-----------------------------------------------------------------------------
// Enemy visibility check
//-----------------------------------------------------------------------------
CBaseEntity *CNPC_AttackHelicopter::FindTrackBlocker( const Vector &vecViewPoint, const Vector &vecTargetPos )
{
if ( m_bIgnorePathVisibilityTests )
return NULL;
CTraceFilterChopper chopperFilter( this, COLLISION_GROUP_NONE );
trace_t tr;
AI_TraceHull( vecViewPoint, vecTargetPos, -Vector(4,4,4), Vector(4,4,4), MASK_SHOT, &chopperFilter, &tr );
if ( tr.fraction != 1.0f )
{
Assert( tr.m_pEnt );
}
return (tr.fraction != 1.0f) ? tr.m_pEnt : NULL;
}
//-----------------------------------------------------------------------------
// More Enemy visibility check
//-----------------------------------------------------------------------------
bool CNPC_AttackHelicopter::FVisible( CBaseEntity *pEntity, int traceMask, CBaseEntity **ppBlocker )
{
if ( pEntity->GetFlags() & FL_NOTARGET )
return false;
#if 0
// FIXME: only block LOS through opaque water
// don't look through water
if ((m_nWaterLevel != 3 && pEntity->m_nWaterLevel == 3)
|| (m_nWaterLevel == 3 && pEntity->m_nWaterLevel == 0))
return false;
#endif
Vector vecLookerOrigin = EyePosition();//look through the caller's 'eyes'
Vector vecTargetOrigin = pEntity->EyePosition();
CTraceFilterChopper chopperFilter( this, COLLISION_GROUP_NONE );
trace_t tr;
UTIL_TraceLine(vecLookerOrigin, vecTargetOrigin, traceMask, &chopperFilter, &tr);
if (tr.fraction != 1.0)
{
// Got line of sight!
if ( tr.m_pEnt == pEntity )
return true;
// Got line of sight on the vehicle the player is driving!
if ( pEntity && pEntity->IsPlayer() )
{
CBasePlayer *pPlayer = assert_cast<CBasePlayer*>( pEntity );
if ( tr.m_pEnt == pPlayer->GetVehicleEntity() )
return true;
}
if (ppBlocker)
{
*ppBlocker = tr.m_pEnt;
}
return false;// Line of sight is not established
}
return true;// line of sight is valid.
}
//------------------------------------------------------------------------------
// Shot spread
//------------------------------------------------------------------------------
#define PLAYER_TIGHTEN_FACTOR 0.75f
Vector CNPC_AttackHelicopter::GetAttackSpread( CBaseCombatWeapon *pWeapon, CBaseEntity *pTarget )
{
float flSinConeDegrees = sin( sk_helicopter_firingcone.GetFloat() * PLAYER_TIGHTEN_FACTOR * 0.5f * (3.14f / 180.0f) );
Vector vecSpread( flSinConeDegrees, flSinConeDegrees, flSinConeDegrees );
return vecSpread;
}
//------------------------------------------------------------------------------
// Find interesting nearby things to shoot
//------------------------------------------------------------------------------
int CNPC_AttackHelicopter::BuildMissTargetList( int nCount, CBaseEntity **ppMissCandidates )
{
int numMissCandidates = 0;
CBaseEntity *pEnts[256];
Vector radius( 150, 150, 150 );
const Vector &vecSource = GetEnemy()->WorldSpaceCenter();
int numEnts = UTIL_EntitiesInBox( pEnts, 256, vecSource - radius, vecSource+radius, 0 );
for ( int i = 0; i < numEnts; i++ )
{
if ( pEnts[i] == NULL )
continue;
if ( numMissCandidates >= nCount )
break;
// Miss candidates cannot include the player or his vehicle
if ( pEnts[i] == GetEnemyVehicle() || pEnts[i] == GetEnemy() )
continue;
// See if it's a good target candidate
if ( FClassnameIs( pEnts[i], "prop_dynamic" ) ||
FClassnameIs( pEnts[i], "prop_physics" ) ||
FClassnameIs( pEnts[i], "physics_prop" ) )
{
ppMissCandidates[numMissCandidates++] = pEnts[i];
}
}
return numMissCandidates;
}
//------------------------------------------------------------------------------
// Gets a vehicle the enemy is in (if any)
//------------------------------------------------------------------------------
CBaseEntity *CNPC_AttackHelicopter::GetEnemyVehicle()
{
if ( !GetEnemy() )
return NULL;
if ( !GetEnemy()->IsPlayer() )
return NULL;
return static_cast<CBasePlayer*>(GetEnemy())->GetVehicleEntity();
}
//------------------------------------------------------------------------------
// Purpose :
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::ShootAtPlayer( const Vector &vBasePos, const Vector &vGunDir )
{
// Fire one shots per round right at the player, using usual rules
FireBulletsInfo_t info;
info.m_vecSrc = vBasePos;
info.m_vecSpread = VECTOR_CONE_PRECALCULATED;
info.m_flDistance = MAX_COORD_RANGE;
info.m_iAmmoType = m_iAmmoType;
info.m_iTracerFreq = 1;
info.m_vecDirShooting = GetActualShootTrajectory( vBasePos );
info.m_nFlags = FIRE_BULLETS_TEMPORARY_DANGER_SOUND;
DoMuzzleFlash();
QAngle vGunAng;
VectorAngles( vGunDir, vGunAng );
FireBullets( info );
// Fire the rest of the bullets at objects around the player
CBaseEntity *ppNearbyTargets[16];
int nActualTargets = BuildMissTargetList( 16, ppNearbyTargets );
// Randomly sort it...
int i;
for ( i = 0; i < nActualTargets; ++i )
{
int nSwap = random->RandomInt( 0, nActualTargets - 1 );
V_swap( ppNearbyTargets[i], ppNearbyTargets[nSwap] );
}
// Just shoot where we're facing
float flSinConeDegrees = sin( sk_helicopter_firingcone.GetFloat() * 0.5f * (3.14f / 180.0f) );
Vector vecSpread( flSinConeDegrees, flSinConeDegrees, flSinConeDegrees );
// How many times should we hit the player this time?
int nDesiredHitCount = (int)(((float)( m_nMaxBurstHits - m_nBurstHits ) / (float)m_nRemainingBursts) + 0.5f);
int nNearbyTargetCount = 0;
int nPlayerShotCount = 0;
for ( i = sk_helicopter_roundsperburst.GetInt() - 1; --i >= 0; )
{
// Find something interesting around the enemy to shoot instead of just missing.
if ( nActualTargets > nNearbyTargetCount )
{
// FIXME: Constrain to the firing cone?
ppNearbyTargets[nNearbyTargetCount]->CollisionProp()->RandomPointInBounds( Vector(.25, .25, .25), Vector(.75, .75, .75), &info.m_vecDirShooting );
info.m_vecDirShooting -= vBasePos;
VectorNormalize( info.m_vecDirShooting );
info.m_vecSpread = VECTOR_CONE_PRECALCULATED;
info.m_flDistance = MAX_COORD_RANGE;
info.m_nFlags = FIRE_BULLETS_TEMPORARY_DANGER_SOUND;
FireBullets( info );
++nNearbyTargetCount;
continue;
}
if ( GetEnemy() && ( nPlayerShotCount < nDesiredHitCount ))
{
GetEnemy()->CollisionProp()->RandomPointInBounds( Vector(0, 0, 0), Vector(1, 1, 1), &info.m_vecDirShooting );
info.m_vecDirShooting -= vBasePos;
VectorNormalize( info.m_vecDirShooting );
info.m_vecSpread = VECTOR_CONE_PRECALCULATED;
info.m_flDistance = MAX_COORD_RANGE;
info.m_nFlags = FIRE_BULLETS_TEMPORARY_DANGER_SOUND;
FireBullets( info );
++nPlayerShotCount;
continue;
}
// Nothing nearby; just fire randomly...
info.m_vecDirShooting = vGunDir;
info.m_vecSpread = vecSpread;
info.m_flDistance = 8192;
info.m_nFlags = FIRE_BULLETS_TEMPORARY_DANGER_SOUND;
FireBullets( info );
}
}
//-----------------------------------------------------------------------------
// Chooses a point within the circle of death to fire in
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::PickDirectionToCircleOfDeath( const Vector &vBasePos, const Vector &vecFireAtPosition, Vector *pResult )
{
*pResult = vecFireAtPosition;
float x, y;
do
{
x = random->RandomFloat( -1.0f, 1.0f );
y = random->RandomFloat( -1.0f, 1.0f );
} while ( (x * x + y * y) > 1.0f );
pResult->x += x * m_flCircleOfDeathRadius;
pResult->y += y * m_flCircleOfDeathRadius;
*pResult -= vBasePos;
VectorNormalize( *pResult );
}
//-----------------------------------------------------------------------------
// Deliberately aims as close as possible w/o hitting
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::AimCloseToTargetButMiss( CBaseEntity *pTarget, float flMinDist, float flMaxDist, const Vector &shootOrigin, Vector *pResult )
{
Vector vecDirection;
VectorSubtract( pTarget->WorldSpaceCenter(), shootOrigin, vecDirection );
float flDist = VectorNormalize( vecDirection );
float flRadius = pTarget->BoundingRadius() + random->RandomFloat( flMinDist, flMaxDist );
float flMinRadius = flRadius;
if ( flDist > flRadius )
{
flMinRadius = flDist * flRadius / sqrt( flDist * flDist - flRadius * flRadius );
}
// Choose random points in a plane perpendicular to the shoot origin.
Vector vecRandomDir;
vecRandomDir.Random( -1.0f, 1.0f );
VectorMA( vecRandomDir, -DotProduct( vecDirection, vecRandomDir ), vecDirection, vecRandomDir );
VectorNormalize( vecRandomDir );
vecRandomDir *= flMinRadius;
vecRandomDir += pTarget->WorldSpaceCenter();
VectorSubtract( vecRandomDir, shootOrigin, *pResult );
VectorNormalize( *pResult );
}
//-----------------------------------------------------------------------------
// Make sure we don't hit too many times
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::FireBullets( const FireBulletsInfo_t &info )
{
// Use this to count the number of hits in a burst
bool bIsPlayer = GetEnemy() && GetEnemy()->IsPlayer();
if ( !bIsPlayer )
{
BaseClass::FireBullets( info );
return;
}
if ( !GetEnemyVehicle() && !IsDeadlyShooting() )
{
if ( m_nBurstHits >= m_nMaxBurstHits )
{
FireBulletsInfo_t actualInfo = info;
actualInfo.m_pAdditionalIgnoreEnt = GetEnemy();
BaseClass::FireBullets( actualInfo );
return;
}
}
CBasePlayer *pPlayer = assert_cast<CBasePlayer*>(GetEnemy());
int nPrevHealth = pPlayer->GetHealth();
int nPrevArmor = pPlayer->ArmorValue();
BaseClass::FireBullets( info );
if (( pPlayer->GetHealth() < nPrevHealth ) || ( pPlayer->ArmorValue() < nPrevArmor ))
{
++m_nBurstHits;
}
}
//------------------------------------------------------------------------------
// Purpose :
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::ShootInsideCircleOfDeath( const Vector &vBasePos, const Vector &vecFireAtPosition )
{
Vector vecFireDirection;
if ( m_nAttackMode == ATTACK_MODE_BULLRUSH_VEHICLE )
{
PickDirectionToCircleOfDeath( vBasePos, vecFireAtPosition, &vecFireDirection );
}
else if ( ( m_nNearShots < m_nMaxNearShots ) || !GetEnemyVehicle() )
{
if ( ( m_nBurstHits < m_nMaxBurstHits ) || !GetEnemy() )
{
++m_nNearShots;
PickDirectionToCircleOfDeath( vBasePos, vecFireAtPosition, &vecFireDirection );
}
else
{
m_nNearShots += 6;
AimCloseToTargetButMiss( GetEnemy(), 20.0f, 50.0f, vBasePos, &vecFireDirection );
}
}
else
{
AimCloseToTargetButMiss( GetEnemyVehicle(), 10.0f, 80.0f, vBasePos, &vecFireDirection );
}
FireBulletsInfo_t info( 1, vBasePos, vecFireDirection, VECTOR_CONE_PRECALCULATED, MAX_COORD_RANGE, m_iAmmoType );
info.m_iTracerFreq = 1;
info.m_nFlags = FIRE_BULLETS_TEMPORARY_DANGER_SOUND;
FireBullets( info );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::DoMuzzleFlash( void )
{
BaseClass::DoMuzzleFlash();
CEffectData data;
data.m_nAttachmentIndex = LookupAttachment( "muzzle" );
data.m_nEntIndex = entindex();
DispatchEffect( "ChopperMuzzleFlash", data );
}
//------------------------------------------------------------------------------
// Purpose :
//------------------------------------------------------------------------------
#define HIT_VEHICLE_SPEED_MIN 200.0f
#define HIT_VEHICLE_SPEED_MAX 500.0f
void CNPC_AttackHelicopter::ShootAtVehicle( const Vector &vBasePos, const Vector &vecFireAtPosition )
{
int nShotsRemaining = sk_helicopter_roundsperburst.GetInt();
DoMuzzleFlash();
// Do special code against episodic drivers
if ( hl2_episodic.GetBool() )
{
Vector vecVelocity;
GetEnemyVehicle()->GetVelocity( &vecVelocity, NULL );
float flSpeed = clamp( vecVelocity.Length(), 0.0f, 400.0f );
float flRange = RemapVal( flSpeed, 0.0f, 400.0f, 0.05f, 1.0f );
// Alter each shot's trajectory based on our speed
for ( int i = 0; i < nShotsRemaining; i++ )
{
Vector vecShotDir;
// If they're at a dead stand-still, just hit them
if ( flRange <= 0.1f )
{
VectorSubtract( GetEnemy()->EyePosition(), vBasePos, vecShotDir );
Vector vecOffset;
vecOffset.Random( -40.0f, 40.0f );
vecShotDir += vecOffset;
VectorNormalize( vecShotDir );
}
else
{
// Aim in a cone around them
AimCloseToTargetButMiss( GetEnemy(), (3*12) * flRange, (10*12) * flRange, vBasePos, &vecShotDir );
}
FireBulletsInfo_t info( 1, vBasePos, vecShotDir, VECTOR_CONE_PRECALCULATED, MAX_COORD_RANGE, m_iAmmoType );
info.m_iTracerFreq = 1;
FireBullets( info );
}
// We opt out of the rest of the function
// FIXME: Should we emulate the below functionality and have half the bullets attempt to miss admirably? -- jdw
return;
}
// Pop one at the player based on how fast he's going
if ( m_nBurstHits < m_nMaxBurstHits )
{
Vector vecDir;
VectorSubtract( GetEnemy()->EyePosition(), vBasePos, vecDir );
Vector vecOffset;
vecOffset.Random( -5.0f, 5.0f );
vecDir += vecOffset;
VectorNormalize( vecDir );
FireBulletsInfo_t info( 1, vBasePos, vecDir, VECTOR_CONE_PRECALCULATED, MAX_COORD_RANGE, m_iAmmoType );
info.m_iTracerFreq = 1;
FireBullets( info );
--nShotsRemaining;
}
// Fire half of the bullets within the circle of death, the other half at interesting things
int i;
int nFireInCircle = nShotsRemaining >> 1;
nShotsRemaining -= nFireInCircle;
for ( i = 0; i < nFireInCircle; ++i )
{
ShootInsideCircleOfDeath( vBasePos, vecFireAtPosition );
}
// Fire the rest of the bullets at objects around the enemy
CBaseEntity *ppNearbyTargets[16];
int nActualTargets = BuildMissTargetList( 16, ppNearbyTargets );
// Randomly sort it...
for ( i = 0; i < nActualTargets; ++i )
{
int nSwap = random->RandomInt( 0, nActualTargets - 1 );
V_swap( ppNearbyTargets[i], ppNearbyTargets[nSwap] );
}
// Just shoot where we're facing
float flSinConeDegrees = sin( sk_helicopter_firingcone.GetFloat() * 0.5f * (3.14f / 180.0f) );
Vector vecSpread( flSinConeDegrees, flSinConeDegrees, flSinConeDegrees );
for ( i = nShotsRemaining; --i >= 0; )
{
// Find something interesting around the enemy to shoot instead of just missing.
if ( nActualTargets > i )
{
Vector vecFireDirection;
ppNearbyTargets[i]->CollisionProp()->RandomPointInBounds( Vector(.25, .25, .25), Vector(.75, .75, .75), &vecFireDirection );
vecFireDirection -= vBasePos;
VectorNormalize( vecFireDirection );
// FIXME: Constrain to the firing cone?
// I put in all the default arguments simply so I could guarantee the first shot of one of the bursts always hits
FireBulletsInfo_t info( 1, vBasePos, vecFireDirection, VECTOR_CONE_PRECALCULATED, MAX_COORD_RANGE, m_iAmmoType );
info.m_iTracerFreq = 1;
FireBullets( info );
}
else
{
ShootInsideCircleOfDeath( vBasePos, vecFireAtPosition );
}
}
}
//------------------------------------------------------------------------------
// Various states of the helicopter firing...
//------------------------------------------------------------------------------
bool CNPC_AttackHelicopter::PoseGunTowardTargetDirection( const Vector &vTargetDir )
{
Vector vecOut;
VectorIRotate( vTargetDir, EntityToWorldTransform(), vecOut );
QAngle angles;
VectorAngles(vecOut, angles);
if (angles.y > 180)
{
angles.y = angles.y - 360;
}
else if (angles.y < -180)
{
angles.y = angles.y + 360;
}
if (angles.x > 180)
{
angles.x = angles.x - 360;
}
else if (angles.x < -180)
{
angles.x = angles.x + 360;
}
if ( ( m_nAttackMode == ATTACK_MODE_BULLRUSH_VEHICLE ) && !IsInSecondaryMode(BULLRUSH_MODE_SHOOT_IDLE_PLAYER) && GetEnemy())
{
if ( GetEnemyVehicle() )
{
angles.x = clamp( angles.x, -12.0f, 0.0f );
angles.y = clamp( angles.y, -10.0f, 10.0f );
}
else
{
angles.x = clamp( angles.x, -10.0f, 10.0f );
angles.y = clamp( angles.y, -10.0f, 10.0f );
}
}
if (angles.x > m_angGun.x)
{
m_angGun.x = MIN( angles.x, m_angGun.x + 12 );
}
if (angles.x < m_angGun.x)
{
m_angGun.x = MAX( angles.x, m_angGun.x - 12 );
}
if (angles.y > m_angGun.y)
{
m_angGun.y = MIN( angles.y, m_angGun.y + 12 );
}
if (angles.y < m_angGun.y)
{
m_angGun.y = MAX( angles.y, m_angGun.y - 12 );
}
SetPoseParameter( m_poseWeapon_Pitch, -m_angGun.x );
SetPoseParameter( m_poseWeapon_Yaw, m_angGun.y );
return true;
}
//------------------------------------------------------------------------------
// Compute the enemy position (non-vehicle case)
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::ComputeFireAtPosition( Vector *pVecActualTargetPosition )
{
// Deal with various leading behaviors...
*pVecActualTargetPosition = m_vecTargetPosition;
}
//------------------------------------------------------------------------------
// Compute the enemy position (non-vehicle case)
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::ComputeVehicleFireAtPosition( Vector *pVecActualTargetPosition )
{
CBaseEntity *pVehicle = GetEnemyVehicle();
// Make sure the circle of death doesn't move more than N units
// This will cause the target to have to maintain a large enough speed
*pVecActualTargetPosition = pVehicle->BodyTarget( GetAbsOrigin(), false );
// NDebugOverlay::Box( *pVecActualTargetPosition,
// Vector(-m_flCircleOfDeathRadius, -m_flCircleOfDeathRadius, 0),
// Vector(m_flCircleOfDeathRadius, m_flCircleOfDeathRadius, 0),
// 0, 0, 255, false, 0.1f );
}
//------------------------------------------------------------------------------
// Here's what we do when we're looking for a target
//------------------------------------------------------------------------------
bool CNPC_AttackHelicopter::DoGunIdle( const Vector &vGunDir, const Vector &vTargetDir )
{
// When bullrushing, skip the idle
if ( ( m_nAttackMode == ATTACK_MODE_BULLRUSH_VEHICLE ) &&
( IsInSecondaryMode( BULLRUSH_MODE_SHOOT_GUN ) || IsInSecondaryMode(BULLRUSH_MODE_SHOOT_IDLE_PLAYER) ) )
{
EmitSound( "NPC_AttackHelicopter.ChargeGun" );
m_flChargeTime = gpGlobals->curtime + CHOPPER_GUN_CHARGE_TIME;
m_nGunState = GUN_STATE_CHARGING;
m_flCircleOfDeathRadius = CHOPPER_MAX_CIRCLE_OF_DEATH_RADIUS;
return true;
}
// Can't continually fire....
if (m_flNextAttack > gpGlobals->curtime)
return false;
// Don't fire if we're too far away, or if the enemy isn't in front of us
if (!GetEnemy())
return false;
float flMaxDistSqr = GetMaxFiringDistance();
flMaxDistSqr *= flMaxDistSqr;
float flDistSqr = WorldSpaceCenter().DistToSqr( GetEnemy()->WorldSpaceCenter() );
if (flDistSqr > flMaxDistSqr)
return false;
// If he's mostly within the cone, shoot away!
float flChargeCone = sk_helicopter_firingcone.GetFloat() * 0.5f;
if ( flChargeCone < 15.0f )
{
flChargeCone = 15.0f;
}
float flCosConeDegrees = cos( flChargeCone * (3.14f / 180.0f) );
float fDotPr = DotProduct( vGunDir, vTargetDir );
if (fDotPr < flCosConeDegrees)
return false;
// Fast shooting doesn't charge up
if( m_nShootingMode == SHOOT_MODE_FAST )
{
m_flChargeTime = gpGlobals->curtime;
m_nGunState = GUN_STATE_CHARGING;
m_flAvoidMetric = 0.0f;
m_vecLastAngVelocity.Init( 0, 0, 0 );
}
else
{
EmitSound( "NPC_AttackHelicopter.ChargeGun" );
float flChargeTime = CHOPPER_GUN_CHARGE_TIME;
float flVariance = flChargeTime * 0.1f;
m_flChargeTime = gpGlobals->curtime + random->RandomFloat(flChargeTime - flVariance, flChargeTime + flVariance);
m_nGunState = GUN_STATE_CHARGING;
m_flAvoidMetric = 0.0f;
m_vecLastAngVelocity.Init( 0, 0, 0 );
}
return true;
}
//------------------------------------------------------------------------------
// How easy is the target to hit?
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::UpdateTargetHittability()
{
// This simply is a measure of how much juking is going on.
// Along with how much steering is happening.
if ( GetEnemyVehicle() )
{
Vector vecVelocity;
AngularImpulse vecAngVelocity;
GetEnemyVehicle()->GetVelocity( &vecVelocity, &vecAngVelocity );
float flDist = fabs( vecAngVelocity.z - m_vecLastAngVelocity.z );
m_flAvoidMetric += flDist;
m_vecLastAngVelocity = vecAngVelocity;
}
}
//------------------------------------------------------------------------------
// Here's what we do when we're getting ready to fire
//------------------------------------------------------------------------------
bool CNPC_AttackHelicopter::DoGunCharging( )
{
// Update the target hittability, which will indicate how many hits we'll accept.
UpdateTargetHittability();
if ( m_flChargeTime > gpGlobals->curtime )
return false;
m_nGunState = GUN_STATE_FIRING;
if ( HasSpawnFlags( SF_HELICOPTER_AGGRESSIVE ) )
{
SetPauseState( PAUSE_AT_NEXT_LOS_POSITION );
}
int nHitFactor = 1;
switch( GetShootingMode() )
{
case SHOOT_MODE_DEFAULT:
case SHOOT_MODE_FAST:
{
int nBurstCount = sk_helicopter_burstcount.GetInt();
m_nRemainingBursts = random->RandomInt( nBurstCount, 2.0 * nBurstCount );
m_flIdleTimeDelay = 0.1f * ( m_nRemainingBursts - nBurstCount );
}
break;
case SHOOT_MODE_LONG_CYCLE:
{
m_nRemainingBursts = 60;
m_flIdleTimeDelay = 0.0f;
nHitFactor = 2;
}
break;
case SHOOT_MODE_CONTINUOUS:
if ( m_nAttackMode == ATTACK_MODE_BULLRUSH_VEHICLE )
{
// We're relying on the special aiming behavior for bullrushing to just randomly deal damage
m_nRemainingBursts = 1;
m_flIdleTimeDelay = 0.0f;
}
else
{
m_nRemainingBursts = 0;
m_flIdleTimeDelay = 0.0f;
nHitFactor = 1000;
}
break;
}
if ( !GetEnemyVehicle() )
{
m_nMaxBurstHits = !IsDeadlyShooting() ? random->RandomInt( 6, 9 ) : 200;
m_nMaxNearShots = 10000;
}
else
{
Vector vecVelocity;
GetEnemyVehicle()->GetVelocity( &vecVelocity, NULL );
float flSpeed = vecVelocity.Length();
flSpeed = clamp( flSpeed, 150.0f, 600.0f );
flSpeed = RemapVal( flSpeed, 150.0f, 600.0f, 0.0f, 1.0f );
float flAvoid = clamp( m_flAvoidMetric, 100.0f, 400.0f );
flAvoid = RemapVal( flAvoid, 100.0f, 400.0f, 0.0f, 1.0f );
float flTotal = 0.5f * ( flSpeed + flAvoid );
int nHitCount = (int)(RemapVal( flTotal, 0.0f, 1.0f, 7, -0.5 ) + 0.5f);
int nMin = nHitCount >= 1 ? nHitCount - 1 : 0;
m_nMaxBurstHits = random->RandomInt( nMin, nHitCount + 1 );
int nNearShots = (int)(RemapVal( flTotal, 0.0f, 1.0f, 70, 5 ) + 0.5f);
int nMinNearShots = nNearShots >= 5 ? nNearShots - 5 : 0;
m_nMaxNearShots = random->RandomInt( nMinNearShots, nNearShots + 5 );
// Set up the circle of death parameters at this point
m_flCircleOfDeathRadius = SimpleSplineRemapVal( flTotal, 0.0f, 1.0f,
CHOPPER_MIN_CIRCLE_OF_DEATH_RADIUS, CHOPPER_MAX_CIRCLE_OF_DEATH_RADIUS );
}
m_nMaxBurstHits *= nHitFactor;
m_nMaxNearShots *= nHitFactor;
m_nBurstHits = 0;
m_nNearShots = 0;
return true;
}
//------------------------------------------------------------------------------
// Shoot where we're facing
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::ShootAtFacingDirection( const Vector &vBasePos, const Vector &vGunDir, bool bFirstShotAccurate )
{
// Just shoot where we're facing
float flSinConeDegrees = sin( sk_helicopter_firingcone.GetFloat() * 0.5f * (3.14f / 180.0f) );
Vector vecSpread( flSinConeDegrees, flSinConeDegrees, flSinConeDegrees );
int nShotCount = sk_helicopter_roundsperburst.GetInt();
if ( bFirstShotAccurate && GetEnemy() )
{
// Check to see if the enemy is within his firing cone
if ( GetEnemy() )
{
// Find the closest point to the gunDir
const Vector &vecCenter = GetEnemy()->WorldSpaceCenter();
float t;
Vector vNearPoint;
Vector vEndPoint;
VectorMA( vBasePos, 1024.0f, vGunDir, vEndPoint );
CalcClosestPointOnLine( vecCenter, vBasePos, vEndPoint, vNearPoint, &t );
if ( t > 0.0f )
{
Vector vecDelta;
VectorSubtract( vecCenter, vBasePos, vecDelta );
float flDist = VectorNormalize( vecDelta );
float flPerpDist = vecCenter.DistTo( vNearPoint );
float flSinAngle = flPerpDist / flDist;
if ( flSinAngle <= flSinConeDegrees )
{
FireBulletsInfo_t info( 1, vBasePos, vecDelta, VECTOR_CONE_PRECALCULATED, 8192, m_iAmmoType );
info.m_iTracerFreq = 1;
FireBullets( info );
--nShotCount;
}
}
}
}
#ifdef HL2_EPISODIC
if( GetEnemy() != NULL )
{
CSoundEnt::InsertSound( SOUND_DANGER, GetEnemy()->WorldSpaceCenter(), 180.0f, 0.5f, this, SOUNDENT_CHANNEL_REPEATED_DANGER );
}
#endif//HL2_EPISODIC
DoMuzzleFlash();
FireBulletsInfo_t info( nShotCount, vBasePos, vGunDir, vecSpread, 8192, m_iAmmoType );
info.m_iTracerFreq = 1;
FireBullets( info );
}
//-----------------------------------------------------------------------------
// Can we zap it?
//-----------------------------------------------------------------------------
bool CNPC_AttackHelicopter::IsValidZapTarget( CBaseEntity *pTarget )
{
// Don't use the player or vehicle as a zap target, we'll do that ourselves.
if ( pTarget->IsPlayer() || pTarget->GetServerVehicle() )
return false;
if ( pTarget == this )
return false;
if ( !pTarget->IsSolid() )
return false;
Assert( pTarget );
IPhysicsObject *pList[VPHYSICS_MAX_OBJECT_LIST_COUNT];
int count = pTarget->VPhysicsGetObjectList( pList, ARRAYSIZE(pList) );
for ( int i = 0; i < count; i++ )
{
int material = pList[i]->GetMaterialIndex();
const surfacedata_t *pSurfaceData = physprops->GetSurfaceData( material );
// Is flesh or metal? Go for it!
if ( pSurfaceData->game.material == CHAR_TEX_METAL ||
pSurfaceData->game.material == CHAR_TEX_FLESH ||
pSurfaceData->game.material == CHAR_TEX_VENT ||
pSurfaceData->game.material == CHAR_TEX_GRATE ||
pSurfaceData->game.material == CHAR_TEX_COMPUTER ||
pSurfaceData->game.material == CHAR_TEX_BLOODYFLESH ||
pSurfaceData->game.material == CHAR_TEX_ALIENFLESH )
{
return true;
}
}
return false;
}
//------------------------------------------------------------------------------
// Effects
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::CreateZapBeam( const Vector &vecTargetPos )
{
CEffectData data;
data.m_nEntIndex = entindex();
data.m_nAttachmentIndex = 0; // m_nGunTipAttachment;
data.m_vOrigin = vecTargetPos;
data.m_flScale = 5;
DispatchEffect( "TeslaZap", data );
}
void CNPC_AttackHelicopter::CreateEntityZapEffect( CBaseEntity *pEnt )
{
CEffectData data;
data.m_nEntIndex = pEnt->entindex();
data.m_flMagnitude = 10;
data.m_flScale = 1.0f;
DispatchEffect( "TeslaHitboxes", data );
}
//------------------------------------------------------------------------------
// Here's what we do when we *are* firing
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::FireElectricityGun( )
{
if ( m_flNextAttack > gpGlobals->curtime )
return;
EmitSound( "ReallyLoudSpark" );
CBaseEntity *ppEnts[256];
Vector vecCenter = WorldSpaceCenter();
float flRadius = 500.0f;
vecCenter.z -= flRadius * 0.8f;
int nEntCount = UTIL_EntitiesInSphere( ppEnts, 256, vecCenter, flRadius, 0 );
CBaseEntity *ppCandidates[256];
int nCandidateCount = 0;
int i;
for ( i = 0; i < nEntCount; i++ )
{
if ( ppEnts[i] == NULL )
continue;
// Zap metal or flesh things.
if ( !IsValidZapTarget( ppEnts[i] ) )
continue;
ppCandidates[ nCandidateCount++ ] = ppEnts[i];
}
// First, put a bolt in front of the player, at random
float flDist = 1024;
if ( GetEnemy() )
{
Vector vecDelta;
Vector2DSubtract( GetEnemy()->WorldSpaceCenter().AsVector2D(), WorldSpaceCenter().AsVector2D(), vecDelta.AsVector2D() );
vecDelta.z = 0.0f;
flDist = VectorNormalize( vecDelta );
Vector vecPerp( -vecDelta.y, vecDelta.x, 0.0f );
int nBoltCount = (int)(ClampSplineRemapVal( flDist, 256.0f, 1024.0f, 8, 0 ) + 0.5f);
for ( i = 0; i < nBoltCount; ++i )
{
Vector vecTargetPt = GetEnemy()->WorldSpaceCenter();
VectorMA( vecTargetPt, random->RandomFloat( flDist + 100, flDist + 500 ), vecDelta, vecTargetPt );
VectorMA( vecTargetPt, random->RandomFloat( -500, 500 ), vecPerp, vecTargetPt );
vecTargetPt.z += random->RandomFloat( -500, 500 );
CreateZapBeam( vecTargetPt );
}
}
// Next, choose the number of bolts...
int nBoltCount = random->RandomInt( 8, 16 );
for ( i = 0; i < nBoltCount; ++i )
{
if ( (nCandidateCount > 0) && random->RandomFloat( 0.0f, 1.0f ) < 0.6f )
{
--nCandidateCount;
Vector vecTarget;
ppCandidates[nCandidateCount]->CollisionProp()->RandomPointInBounds( vec3_origin, Vector( 1, 1, 1 ), &vecTarget );
CreateZapBeam( vecTarget );
CreateEntityZapEffect( ppCandidates[nCandidateCount] );
}
else
{
// Select random point *on* sphere
Vector vecTargetPt;
float flEffectRadius = random->RandomFloat( flRadius * 1.2, flRadius * 1.5f );
float flTheta = random->RandomFloat( 0.0f, 2.0f * M_PI );
float flPhi = random->RandomFloat( -0.5f * M_PI, 0.5f * M_PI );
vecTargetPt.x = cos(flTheta) * cos(flPhi);
vecTargetPt.y = sin(flTheta) * cos(flPhi);
vecTargetPt.z = sin(flPhi);
vecTargetPt *= flEffectRadius;
vecTargetPt += vecCenter;
CreateZapBeam( vecTargetPt );
}
}
// Finally, put a bolt right at the player, at random
float flHitRatio = ClampSplineRemapVal( flDist, 128.0f, 512.0f, 0.75f, 0.0f );
if ( random->RandomFloat( 0.0f, 1.0f ) < flHitRatio )
{
if ( GetEnemyVehicle() )
{
Vector vecTarget;
GetEnemyVehicle()->CollisionProp()->RandomPointInBounds( vec3_origin, Vector( 1, 1, 1 ), &vecTarget );
CreateZapBeam( vecTarget );
CreateEntityZapEffect( GetEnemyVehicle() );
CTakeDamageInfo info( this, this, 5, DMG_SHOCK );
GetEnemy()->TakeDamage( info );
}
else if ( GetEnemy() )
{
Vector vecTarget;
GetEnemy()->CollisionProp()->RandomPointInBounds( vec3_origin, Vector( 1, 1, 1 ), &vecTarget );
CreateZapBeam( vecTarget );
CTakeDamageInfo info( this, this, 5, DMG_SHOCK );
GetEnemy()->TakeDamage( info );
}
}
m_flNextAttack = gpGlobals->curtime + random->RandomFloat( 0.3f, 1.0f );
}
//------------------------------------------------------------------------------
// Here's what we do when we *are* firing
//------------------------------------------------------------------------------
#define INTERVAL_BETWEEN_HITS 4
bool CNPC_AttackHelicopter::DoGunFiring( const Vector &vBasePos, const Vector &vGunDir, const Vector &vecFireAtPosition )
{
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
float flVolume = controller.SoundGetVolume( m_pGunFiringSound );
if ( flVolume != 1.0f )
{
controller.SoundChangeVolume( m_pGunFiringSound, 1.0, 0.01f );
}
if ( ( m_nAttackMode == ATTACK_MODE_BULLRUSH_VEHICLE ) && ( IsInSecondaryMode( BULLRUSH_MODE_SHOOT_GUN ) ) )
{
ShootAtFacingDirection( vBasePos, vGunDir, m_nRemainingBursts == 0 );
}
else if ( GetEnemyVehicle() )
{
ShootAtVehicle( vBasePos, vecFireAtPosition );
}
else if ( GetEnemy() && GetEnemy()->IsPlayer() )
{
if ( !IsDeadlyShooting() )
{
ShootAtPlayer( vBasePos, vGunDir );
}
else
{
ShootAtFacingDirection( vBasePos, vGunDir, true );
}
}
else
{
ShootAtFacingDirection( vBasePos, vGunDir, false );
}
if ( m_nAttackMode == ATTACK_MODE_BULLRUSH_VEHICLE )
{
if ( --m_nRemainingBursts < 0 )
{
m_nRemainingBursts = INTERVAL_BETWEEN_HITS;
}
return true;
}
--m_nRemainingBursts;
if ( m_nRemainingBursts > 0 )
return true;
controller.SoundChangeVolume( m_pGunFiringSound, 0.0, 0.01f );
float flIdleTime = CHOPPER_GUN_IDLE_TIME;
float flVariance = flIdleTime * 0.1f;
m_flNextAttack = gpGlobals->curtime + m_flIdleTimeDelay + random->RandomFloat(flIdleTime - flVariance, flIdleTime + flVariance);
m_nGunState = GUN_STATE_IDLE;
SetPauseState( PAUSE_NO_PAUSE );
return true;
}
//------------------------------------------------------------------------------
// Is it "fair" to drop this bomb?
//------------------------------------------------------------------------------
#define MIN_BOMB_DISTANCE_SQR ( 600.0f * 600.0f )
bool CNPC_AttackHelicopter::IsBombDropFair( const Vector &vecBombStartPos, const Vector &vecBombVelocity )
{
if ( (m_nAttackMode == ATTACK_MODE_BULLRUSH_VEHICLE) && IsInSecondaryMode( BULLRUSH_MODE_SHOOT_IDLE_PLAYER ) )
return true;
// Can happen if you're noclipping around
if ( !GetEnemy() )
return false;
// If the player is moving slowly, it's fair
if ( GetEnemy()->GetSmoothedVelocity().LengthSqr() < ( CHOPPER_SLOW_BOMB_SPEED * CHOPPER_SLOW_BOMB_SPEED ) )
return true;
// Skip out if we're right above or behind the player.. that's unfair
if ( GetEnemy() && GetEnemy()->IsPlayer() )
{
// How much time will it take to fall?
// dx = 0.5 * a * t^2
Vector vecTarget = GetEnemy()->BodyTarget( GetAbsOrigin(), false );
float dz = vecBombStartPos.z - vecTarget.z;
float dt = (dz > 0.0f) ? sqrt( 2 * dz / GetCurrentGravity() ) : 0.0f;
// Where will the enemy be in that time?
Vector vecEnemyVel = GetEnemy()->GetSmoothedVelocity();
VectorMA( vecTarget, dt, vecEnemyVel, vecTarget );
// Where will the bomb be in that time?
Vector vecBomb;
VectorMA( vecBombStartPos, dt, vecBombVelocity, vecBomb );
float flEnemySpeed = vecEnemyVel.LengthSqr();
flEnemySpeed = clamp( flEnemySpeed, 200.0f, 500.0f );
float flDistFactorSq = RemapVal( flEnemySpeed, 200.0f, 500.0f, 0.3f, 1.0f );
flDistFactorSq *= flDistFactorSq;
// If it's too close, then we're not doing it.
if ( vecBomb.AsVector2D().DistToSqr( vecTarget.AsVector2D() ) < (flDistFactorSq * MIN_BOMB_DISTANCE_SQR) )
return false;
}
return true;
}
//-----------------------------------------------------------------------------
// Purpose: Create the bomb entity and set it up
// Input : &vecPos - Position to spawn at
// &vecVelocity - velocity to spawn with
//-----------------------------------------------------------------------------
CGrenadeHelicopter *CNPC_AttackHelicopter::SpawnBombEntity( const Vector &vecPos, const Vector &vecVelocity )
{
// Create the grenade and set it up
CGrenadeHelicopter *pGrenade = static_cast<CGrenadeHelicopter*>(CreateEntityByName( "grenade_helicopter" ));
pGrenade->SetAbsOrigin( vecPos );
pGrenade->SetOwnerEntity( this );
pGrenade->SetThrower( this );
pGrenade->SetAbsVelocity( vecVelocity );
DispatchSpawn( pGrenade );
pGrenade->SetExplodeOnContact( m_bBombsExplodeOnContact );
#ifdef HL2_EPISODIC
// Disable collisions with the owner's bone followers while we drop
physfollower_t *pFollower = m_BoneFollowerManager.GetBoneFollower( 0 );
if ( pFollower )
{
CBaseEntity *pBoneFollower = pFollower->hFollower;
PhysDisableEntityCollisions( pBoneFollower, pGrenade );
pGrenade->SetCollisionObject( pBoneFollower );
}
#endif // HL2_EPISODIC
return pGrenade;
}
//------------------------------------------------------------------------------
// Actually drops the bomb
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::CreateBomb( bool bCheckForFairness, Vector *pVecVelocity, bool bMegaBomb )
{
if ( m_bBombingSuppressed )
return;
Vector vTipPos;
GetAttachment( m_nBombAttachment, vTipPos );
if ( !CBombSuppressor::CanBomb( vTipPos ) )
return;
// Compute velocity
Vector vecActualVelocity;
if ( !pVecVelocity )
{
Vector vecAcross;
vecActualVelocity = GetAbsVelocity();
CrossProduct( vecActualVelocity, Vector( 0, 0, 1 ), vecAcross );
VectorNormalize( vecAcross );
vecAcross *= random->RandomFloat( 10.0f, 30.0f );
vecAcross *= random->RandomFloat( 0.0f, 1.0f ) < 0.5f ? 1.0f : -1.0f;
// Blat out z component of velocity if it's moving upward....
if ( vecActualVelocity.z > 0 )
{
vecActualVelocity.z = 0.0f;
}
vecActualVelocity += vecAcross;
}
else
{
vecActualVelocity = *pVecVelocity;
}
if ( bCheckForFairness )
{
if ( !IsBombDropFair( vTipPos, vecActualVelocity ) )
return;
}
AddGesture( (Activity)ACT_HELICOPTER_DROP_BOMB );
EmitSound( "NPC_AttackHelicopter.DropMine" );
// Make the bomb and send it off
CGrenadeHelicopter *pGrenade = SpawnBombEntity( vTipPos, vecActualVelocity );
if ( pGrenade && bMegaBomb )
{
pGrenade->AddSpawnFlags( SF_GRENADE_HELICOPTER_MEGABOMB );
}
}
//------------------------------------------------------------------------------
// Drop a bomb at a particular location
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::InputDropBomb( inputdata_t &inputdata )
{
if ( m_flInputDropBombTime > gpGlobals->curtime )
return;
// Prevent two triggers from being hit the same frame
m_flInputDropBombTime = gpGlobals->curtime + 0.01f;
CreateBomb( );
// If we're in the middle of a bomb dropping schedule, wait to drop another bomb.
if ( ShouldDropBombs() )
{
m_flNextAttack = gpGlobals->curtime + 0.5f + random->RandomFloat( 0.3f, 0.6f );
}
}
//------------------------------------------------------------------------------
// Drops a bomb straight downwards
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::InputDropBombStraightDown( inputdata_t &inputdata )
{
if ( m_flInputDropBombTime > gpGlobals->curtime )
return;
// Prevent two triggers from being hit the same frame
m_flInputDropBombTime = gpGlobals->curtime + 0.01f;
Vector vTipPos;
GetAttachment( m_nBombAttachment, vTipPos );
// Make the bomb drop straight down
SpawnBombEntity( vTipPos, vec3_origin );
// If we're in the middle of a bomb dropping schedule, wait to drop another bomb.
if ( ShouldDropBombs() )
{
m_flNextAttack = gpGlobals->curtime + 0.5f + random->RandomFloat( 0.3f, 0.6f );
}
}
//------------------------------------------------------------------------------
// Drop a bomb at a particular location
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::InputDropBombAtTargetInternal( inputdata_t &inputdata, bool bCheckFairness )
{
if ( m_flInputDropBombTime > gpGlobals->curtime )
return;
// Prevent two triggers from being hit the same frame
m_flInputDropBombTime = gpGlobals->curtime + 0.01f;
// Find our specified target
string_t strBombTarget = MAKE_STRING( inputdata.value.String() );
CBaseEntity *pBombEnt = gEntList.FindEntityByName( NULL, strBombTarget );
if ( pBombEnt == NULL )
{
Warning( "%s: Could not find bomb drop target '%s'!\n", GetClassname(), STRING( strBombTarget ) );
return;
}
Vector vTipPos;
GetAttachment( m_nBombAttachment, vTipPos );
// Compute the time it would take to fall to the target
Vector vecTarget = pBombEnt->BodyTarget( GetAbsOrigin(), false );
float dz = vTipPos.z - vecTarget.z;
if ( dz <= 0.0f )
{
Warning("Bomb target %s is above the chopper!\n", STRING( strBombTarget ) );
return;
}
float dt = sqrt( 2 * dz / GetCurrentGravity() );
// Compute the velocity that would make it happen
Vector vecVelocity;
VectorSubtract( vecTarget, vTipPos, vecVelocity );
vecVelocity /= dt;
vecVelocity.z = 0.0f;
if ( bCheckFairness )
{
if ( !IsBombDropFair( vTipPos, vecVelocity ) )
return;
}
// Make the bomb and send it off
SpawnBombEntity( vTipPos, vecVelocity );
// If we're in the middle of a bomb dropping schedule, wait to drop another bomb.
if ( ShouldDropBombs() )
{
m_flNextAttack = gpGlobals->curtime + 1.5f + random->RandomFloat( 0.1f, 0.2f );
}
}
//------------------------------------------------------------------------------
// Drop a bomb at a particular location
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::InputDropBombAtTargetAlways( inputdata_t &inputdata )
{
InputDropBombAtTargetInternal( inputdata, false );
}
//------------------------------------------------------------------------------
// Drop a bomb at a particular location
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::InputDropBombAtTarget( inputdata_t &inputdata )
{
InputDropBombAtTargetInternal( inputdata, true );
}
//------------------------------------------------------------------------------
// Drop a bomb at a particular location
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::InputDropBombDelay( inputdata_t &inputdata )
{
m_flInputDropBombTime = gpGlobals->curtime + inputdata.value.Float();
if ( ShouldDropBombs() )
{
m_flNextAttack = m_flInputDropBombTime;
}
}
//------------------------------------------------------------------------------
// Drop those bombs!
//------------------------------------------------------------------------------
#define MAX_BULLRUSH_BOMB_DISTANCE_SQR ( 3072.0f * 3072.0f )
void CNPC_AttackHelicopter::DropBombs( )
{
// Can't continually fire....
if (m_flNextAttack > gpGlobals->curtime)
return;
// Otherwise, behave as normal.
if ( m_nAttackMode != ATTACK_MODE_BULLRUSH_VEHICLE )
{
if ( GetEnemy() && GetEnemy()->IsPlayer() )
{
if ( GetEnemy()->GetSmoothedVelocity().LengthSqr() > ( CHOPPER_SLOW_BOMB_SPEED * CHOPPER_SLOW_BOMB_SPEED ) )
{
// Don't drop bombs if you are behind the player, unless the player is moving slowly
float flLeadingDistSq = GetLeadingDistance() * 0.75f;
flLeadingDistSq *= flLeadingDistSq;
Vector vecPoint;
ClosestPointToCurrentPath( &vecPoint );
if ( vecPoint.AsVector2D().DistToSqr( GetDesiredPosition().AsVector2D() ) > flLeadingDistSq )
return;
}
}
}
else
{
// Skip out if we're bullrushing but too far from the player
if ( GetEnemy() )
{
if ( GetEnemy()->GetAbsOrigin().AsVector2D().DistToSqr( GetAbsOrigin().AsVector2D() ) > MAX_BULLRUSH_BOMB_DISTANCE_SQR )
return;
}
}
CreateBomb( );
m_flNextAttack = gpGlobals->curtime + 0.5f + random->RandomFloat( 0.3f, 0.6f );
if ( (m_nAttackMode != ATTACK_MODE_BULLRUSH_VEHICLE) )
{
if ( --m_nGrenadeCount <= 0 )
{
m_nGrenadeCount = CHOPPER_BOMB_DROP_COUNT;
m_flNextAttack += random->RandomFloat( 1.5f, 3.0f );
}
}
}
//------------------------------------------------------------------------------
// Should we drop those bombs?
//------------------------------------------------------------------------------
#define BOMB_GRACE_PERIOD 1.5f
#define BOMB_MIN_SPEED 150.0
bool CNPC_AttackHelicopter::ShouldDropBombs( void )
{
if ( IsCarpetBombing() )
return true;
if ( m_nAttackMode == ATTACK_MODE_BULLRUSH_VEHICLE )
{
// Distance determines whether or not we should do this
if ((m_nSecondaryMode == BULLRUSH_MODE_SHOOT_IDLE_PLAYER) && (SecondaryModeTime() >= BULLRUSH_IDLE_PLAYER_FIRE_TIME))
return ShouldBombIdlePlayer();
return (( m_nSecondaryMode == BULLRUSH_MODE_DROP_BOMBS_FIXED_SPEED ) || ( m_nSecondaryMode == BULLRUSH_MODE_DROP_BOMBS_FOLLOW_PLAYER ));
}
if (!IsLeading() || !GetEnemyVehicle())
return false;
if (( m_nAttackMode != ATTACK_MODE_BOMB_VEHICLE ) && ( m_nAttackMode != ATTACK_MODE_ALWAYS_LEAD_VEHICLE ))
return false;
if ( m_nGunState != GUN_STATE_IDLE )
return false;
// This is for bombing. If you get hit, give a grace period to get back to speed
float flSpeedSqr = GetEnemy()->GetSmoothedVelocity().LengthSqr();
if ( flSpeedSqr >= BOMB_MIN_SPEED * BOMB_MIN_SPEED )
{
m_flLastFastTime = gpGlobals->curtime;
}
else
{
if ( ( gpGlobals->curtime - m_flLastFastTime ) < BOMB_GRACE_PERIOD )
return false;
}
float flSpeedAlongPath = TargetSpeedAlongPath();
if ( m_nAttackMode == ATTACK_MODE_BOMB_VEHICLE )
return ( flSpeedAlongPath > -BOMB_MIN_SPEED );
// This is for ALWAYS_LEAD
if ( fabs(flSpeedAlongPath) < 50.0f )
return false;
float flLeadingDist = ComputeDistanceToLeadingPosition( );
flLeadingDist = GetLeadingDistance() - flLeadingDist;
if ( flSpeedAlongPath < 0.0f )
{
return flLeadingDist < 300.0f;
}
else
{
return flLeadingDist > -300.0f;
}
}
//------------------------------------------------------------------------------
// Different bomb-dropping behavior
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::BullrushBombs( )
{
if ( gpGlobals->curtime < m_flNextBullrushBombTime )
return;
if ( m_nBullrushBombMode & 0x1 )
{
CreateBomb( false, NULL, true );
}
else
{
Vector vecAcross;
Vector vecVelocity = GetAbsVelocity();
CrossProduct( vecVelocity, Vector( 0, 0, 1 ), vecAcross );
VectorNormalize( vecAcross );
vecAcross *= random->RandomFloat( 300.0f, 500.0f );
// Blat out z component of velocity if it's moving upward....
if ( vecVelocity.z > 0 )
{
vecVelocity.z = 0.0f;
}
vecVelocity += vecAcross;
CreateBomb( false, &vecVelocity, true );
VectorMA( vecVelocity, -2.0f, vecAcross, vecVelocity );
CreateBomb( false, &vecVelocity, true );
}
m_nBullrushBombMode = !m_nBullrushBombMode;
m_flNextBullrushBombTime = gpGlobals->curtime + 0.2f;
}
//-----------------------------------------------------------------------------
// Purpose: Turn the gun off
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::InputGunOff( inputdata_t &inputdata )
{
BaseClass::InputGunOff( inputdata );
if ( m_pGunFiringSound )
{
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
controller.SoundChangeVolume( m_pGunFiringSound, 0.0, 0.01f );
}
}
//------------------------------------------------------------------------------
// Fire that gun baby!
//------------------------------------------------------------------------------
bool CNPC_AttackHelicopter::FireGun( void )
{
// Do the test electricity gun
if ( HasSpawnFlags(SF_HELICOPTER_ELECTRICAL_DRONE) )
{
FireElectricityGun( );
return true;
}
// HACK: CBaseHelicopter ignores this, and fire forever at the last place it saw the player. Why?
if (( m_nGunState == GUN_STATE_IDLE ) && ( m_nAttackMode != ATTACK_MODE_BULLRUSH_VEHICLE ) && !IsCarpetBombing() )
{
if ( (m_flLastSeen + 1 <= gpGlobals->curtime) || (m_flPrevSeen + m_flGracePeriod > gpGlobals->curtime) )
return false;
}
if ( IsCarpetBombing() )
{
BullrushBombs();
return false;
}
if ( ShouldDropBombs() )
{
DropBombs( );
return false;
}
// Drop those bullrush bombs when shooting...
if ( m_nAttackMode == ATTACK_MODE_BULLRUSH_VEHICLE )
{
if ( IsInSecondaryMode( BULLRUSH_MODE_MEGA_BOMB ) )
{
BullrushBombs( );
return false;
}
// Don't fire if we're bullrushing and we're getting distance
if ( !IsInSecondaryMode( BULLRUSH_MODE_SHOOT_GUN ) && !IsInSecondaryMode(BULLRUSH_MODE_SHOOT_IDLE_PLAYER) )
return false;
// If we're in the grace period on this mode, then don't fire
if ( IsInSecondaryMode( BULLRUSH_MODE_SHOOT_IDLE_PLAYER ) && (SecondaryModeTime() < BULLRUSH_IDLE_PLAYER_FIRE_TIME) )
{
// Stop our gun sound
if ( m_nGunState != GUN_STATE_IDLE )
{
ShutdownGunDuringBullrush();
}
return false;
}
}
// Get gun attachment points
Vector vBasePos;
GetAttachment( m_nGunBaseAttachment, vBasePos );
// Aim perfectly while idle, but after charging, the gun don't move so fast.
Vector vecFireAtPosition;
if ( !GetEnemyVehicle() || (m_nGunState == GUN_STATE_IDLE) )
{
ComputeFireAtPosition( &vecFireAtPosition );
}
else
{
ComputeVehicleFireAtPosition( &vecFireAtPosition );
}
Vector vTargetDir = vecFireAtPosition - vBasePos;
VectorNormalize( vTargetDir );
// Makes the model of the gun point to where we're aiming.
if ( !PoseGunTowardTargetDirection( vTargetDir ) )
return false;
// Are we charging?
if ( m_nGunState == GUN_STATE_CHARGING )
{
if ( !DoGunCharging( ) )
return false;
}
Vector vTipPos;
GetAttachment( m_nGunTipAttachment, vTipPos );
Vector vGunDir = vTipPos - vBasePos;
VectorNormalize( vGunDir );
// Are we firing?
if ( m_nGunState == GUN_STATE_FIRING )
{
return DoGunFiring( vTipPos, vGunDir, vecFireAtPosition );
}
return DoGunIdle( vGunDir, vTargetDir );
}
//-----------------------------------------------------------------------------
// Should we trigger a damage effect?
//-----------------------------------------------------------------------------
inline bool CNPC_AttackHelicopter::ShouldTriggerDamageEffect( int nPrevHealth, int nEffectCount ) const
{
int nPrevRange = (int)( ((float)nPrevHealth / (float)GetMaxHealth()) * nEffectCount );
int nRange = (int)( ((float)GetHealth() / (float)GetMaxHealth()) * nEffectCount );
return ( nRange != nPrevRange );
}
//-----------------------------------------------------------------------------
// Add a smoke trail since we've taken more damage
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::AddSmokeTrail( const Vector &vecPos )
{
if ( m_nSmokeTrailCount == MAX_SMOKE_TRAILS )
return;
// See if there's an attachment for this smoke trail
int nAttachment = LookupAttachment( UTIL_VarArgs( "damage%d", m_nSmokeTrailCount ) );
if ( nAttachment == 0 )
return;
// The final smoke trail is a flaming engine
if ( m_nSmokeTrailCount == 0 || m_nSmokeTrailCount % 2 )
{
CFireTrail *pFireTrail = CFireTrail::CreateFireTrail();
if ( pFireTrail == NULL )
return;
m_hSmokeTrail[m_nSmokeTrailCount] = pFireTrail;
pFireTrail->FollowEntity( this, UTIL_VarArgs( "damage%d", m_nSmokeTrailCount ) );
pFireTrail->SetParent( this, nAttachment );
pFireTrail->SetLocalOrigin( vec3_origin );
pFireTrail->SetMoveType( MOVETYPE_NONE );
pFireTrail->SetLifetime( -1 );
}
else
{
SmokeTrail *pSmokeTrail = SmokeTrail::CreateSmokeTrail();
if( !pSmokeTrail )
return;
m_hSmokeTrail[m_nSmokeTrailCount] = pSmokeTrail;
pSmokeTrail->m_SpawnRate = 48;
pSmokeTrail->m_ParticleLifetime = 0.5f;
pSmokeTrail->m_StartColor.Init(0.15, 0.15, 0.15);
pSmokeTrail->m_EndColor.Init(0.0, 0.0, 0.0);
pSmokeTrail->m_StartSize = 24;
pSmokeTrail->m_EndSize = 80;
pSmokeTrail->m_SpawnRadius = 8;
pSmokeTrail->m_Opacity = 0.2;
pSmokeTrail->m_MinSpeed = 16;
pSmokeTrail->m_MaxSpeed = 64;
pSmokeTrail->SetLifetime(-1);
pSmokeTrail->SetParent( this, nAttachment );
pSmokeTrail->SetLocalOrigin( vec3_origin );
pSmokeTrail->SetMoveType( MOVETYPE_NONE );
}
m_nSmokeTrailCount++;
}
//-----------------------------------------------------------------------------
// Destroy all smoke trails
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::DestroySmokeTrails()
{
for ( int i = m_nSmokeTrailCount; --i >= 0; )
{
UTIL_Remove( m_hSmokeTrail[i] );
m_hSmokeTrail[i] = NULL;
}
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : &vecChunkPos -
//-----------------------------------------------------------------------------
void Chopper_CreateChunk( CBaseEntity *pChopper, const Vector &vecChunkPos, const QAngle &vecChunkAngles, const char *pszChunkName, bool bSmall )
{
// Drop a flaming, smoking chunk.
CGib *pChunk = CREATE_ENTITY( CGib, "gib" );
pChunk->Spawn( pszChunkName );
pChunk->SetBloodColor( DONT_BLEED );
pChunk->SetAbsOrigin( vecChunkPos );
pChunk->SetAbsAngles( vecChunkAngles );
pChunk->SetOwnerEntity( pChopper );
if ( bSmall )
{
pChunk->m_lifeTime = random->RandomFloat( 0.5f, 1.0f );
pChunk->SetSolidFlags( FSOLID_NOT_SOLID );
pChunk->SetSolid( SOLID_BBOX );
pChunk->AddEffects( EF_NODRAW );
pChunk->SetGravity( UTIL_ScaleForGravity( 400 ) );
}
else
{
pChunk->m_lifeTime = 5.0f;
}
pChunk->SetCollisionGroup( COLLISION_GROUP_DEBRIS );
// Set the velocity
Vector vecVelocity;
AngularImpulse angImpulse;
QAngle angles;
angles.x = random->RandomFloat( -70, 20 );
angles.y = random->RandomFloat( 0, 360 );
angles.z = 0.0f;
AngleVectors( angles, &vecVelocity );
vecVelocity *= random->RandomFloat( 550, 800 );
vecVelocity += pChopper->GetAbsVelocity();
angImpulse = RandomAngularImpulse( -180, 180 );
pChunk->SetAbsVelocity( vecVelocity );
if ( bSmall == false )
{
IPhysicsObject *pPhysicsObject = pChunk->VPhysicsInitNormal( SOLID_VPHYSICS, pChunk->GetSolidFlags(), false );
if ( pPhysicsObject )
{
pPhysicsObject->EnableMotion( true );
pPhysicsObject->SetVelocity(&vecVelocity, &angImpulse );
}
}
CFireTrail *pFireTrail = CFireTrail::CreateFireTrail();
if ( pFireTrail == NULL )
return;
pFireTrail->FollowEntity( pChunk, "" );
pFireTrail->SetParent( pChunk, 0 );
pFireTrail->SetLocalOrigin( vec3_origin );
pFireTrail->SetMoveType( MOVETYPE_NONE );
pFireTrail->SetLifetime( pChunk->m_lifeTime );
}
//------------------------------------------------------------------------------
// Pow!
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::ExplodeAndThrowChunk( const Vector &vecExplosionPos )
{
CEffectData data;
data.m_vOrigin = vecExplosionPos;
DispatchEffect( "HelicopterMegaBomb", data );
EmitSound( "BaseExplosionEffect.Sound" );
UTIL_ScreenShake( vecExplosionPos, 25.0, 150.0, 1.0, 750.0f, SHAKE_START );
if(GetCrashPoint() != NULL)
{
// Make it clear that I'm done for.
ExplosionCreate( vecExplosionPos, QAngle(0,0,1), this, 100, 128, false );
}
if ( random->RandomInt( 0, 4 ) )
{
for ( int i = 0; i < 2; i++ )
{
Chopper_CreateChunk( this, vecExplosionPos, RandomAngle(0, 360), g_PropDataSystem.GetRandomChunkModel( "MetalChunks" ), true );
}
}
else
{
Chopper_CreateChunk( this, vecExplosionPos, RandomAngle(0, 360), s_pChunkModelName[random->RandomInt( 0, CHOPPER_MAX_SMALL_CHUNKS - 1 )], false );
}
}
//-----------------------------------------------------------------------------
// Drop a corpse!
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::DropCorpse( int nDamage )
{
// Don't drop another corpse if the next guy's not out on the gun yet
if ( m_flLastCorpseFall > gpGlobals->curtime )
return;
// Clamp damage to prevent ridiculous ragdoll velocity
if( nDamage > 250.0f )
nDamage = 250.0f;
m_flLastCorpseFall = gpGlobals->curtime + 3.0;
// Spawn a ragdoll combine guard
float forceScale = nDamage * 75 * 4;
Vector vecForceVector = RandomVector(-1,1);
vecForceVector.z = 0.5;
vecForceVector *= forceScale;
CBaseEntity *pGib = CreateRagGib( "models/combine_soldier.mdl", GetAbsOrigin(), GetAbsAngles(), vecForceVector );
if ( pGib )
{
pGib->SetOwnerEntity( this );
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::TraceAttack( const CTakeDamageInfo &info, const Vector &vecDir, trace_t *ptr, CDmgAccumulator *pAccumulator )
{
// Take no damage from trace attacks unless it's blast damage. RadiusDamage() sometimes calls
// TraceAttack() as a means for delivering blast damage. Usually when the explosive penetrates
// the target. (RPG missiles do this sometimes).
if ( ( info.GetDamageType() & DMG_AIRBOAT ) ||
( info.GetInflictor()->Classify() == CLASS_MISSILE ) ||
( info.GetAttacker()->Classify() == CLASS_MISSILE ) )
{
BaseClass::BaseClass::TraceAttack( info, vecDir, ptr, pAccumulator );
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
int CNPC_AttackHelicopter::OnTakeDamage( const CTakeDamageInfo &info )
{
// We don't take blast damage from anything but the airboat or missiles (or myself!)
if( info.GetInflictor() != this )
{
if ( ( ( info.GetDamageType() & DMG_AIRBOAT ) == 0 ) &&
( info.GetInflictor()->Classify() != CLASS_MISSILE ) &&
( info.GetAttacker()->Classify() != CLASS_MISSILE ) )
return 0;
}
if ( m_bIndestructible )
{
if ( GetHealth() < info.GetDamage() )
return 0;
}
// helicopter takes extra damage from its own grenades
CGrenadeHelicopter *pGren = dynamic_cast<CGrenadeHelicopter *>(info.GetInflictor());
if ( pGren && info.GetAttacker() && info.GetAttacker()->IsPlayer() )
{
CTakeDamageInfo fudgedInfo = info;
float damage;
if( g_pGameRules->IsSkillLevel(SKILL_EASY) )
{
damage = GetMaxHealth() / sk_helicopter_num_bombs1.GetFloat();
}
else if( g_pGameRules->IsSkillLevel(SKILL_HARD) )
{
damage = GetMaxHealth() / sk_helicopter_num_bombs3.GetFloat();
}
else // Medium, or unspecified
{
damage = GetMaxHealth() / sk_helicopter_num_bombs2.GetFloat();
}
damage = ceilf( damage );
fudgedInfo.SetDamage( damage );
fudgedInfo.SetMaxDamage( damage );
return BaseClass::OnTakeDamage( fudgedInfo );
}
return BaseClass::OnTakeDamage( info );
}
//-----------------------------------------------------------------------------
// Purpose: Take damage from trace attacks if they hit the gunner
//-----------------------------------------------------------------------------
int CNPC_AttackHelicopter::OnTakeDamage_Alive( const CTakeDamageInfo &info )
{
int nPrevHealth = GetHealth();
if ( ( info.GetInflictor() != NULL ) && ( info.GetInflictor()->GetOwnerEntity() != NULL ) && ( info.GetInflictor()->GetOwnerEntity() == this ) )
{
// Don't take damage from my own bombs. (Unless the player grabbed them and threw them back)
return 0;
}
// Chain
int nRetVal = BaseClass::OnTakeDamage_Alive( info );
if( info.GetDamageType() & DMG_BLAST )
{
// Apply a force push that makes us look like we're reacting to the damage
Vector damageDir = info.GetDamageForce();
VectorNormalize( damageDir );
ApplyAbsVelocityImpulse( damageDir * 500.0f );
// Knock the helicopter off of the level, too.
Vector vecRight, vecForce;
float flDot;
GetVectors( NULL, &vecRight, NULL );
vecForce = info.GetDamageForce();
VectorNormalize( vecForce );
flDot = DotProduct( vecForce, vecRight );
m_flGoalRollDmg = random->RandomFloat( 10, 30 );
if( flDot <= 0.0f )
{
// Missile hit the right side.
m_flGoalRollDmg *= -1;
}
}
// Spawn damage effects
if ( nPrevHealth != GetHealth() )
{
// Give the badly damaged call to say we're going to mega bomb soon
if ( m_nAttackMode == ATTACK_MODE_BULLRUSH_VEHICLE )
{
if (( nPrevHealth > m_flNextMegaBombHealth ) && (GetHealth() <= m_flNextMegaBombHealth) )
{
EmitSound( "NPC_AttackHelicopter.BadlyDamagedAlert" );
}
}
if ( ShouldTriggerDamageEffect( nPrevHealth, MAX_SMOKE_TRAILS ) )
{
AddSmokeTrail( info.GetDamagePosition() );
}
if ( ShouldTriggerDamageEffect( nPrevHealth, MAX_CORPSES ) )
{
if ( nPrevHealth != GetMaxHealth() )
{
DropCorpse( info.GetDamage() );
}
}
if ( ShouldTriggerDamageEffect( nPrevHealth, MAX_EXPLOSIONS ) )
{
ExplodeAndThrowChunk( info.GetDamagePosition() );
}
int nPrevPercent = (int)(100.0f * nPrevHealth / GetMaxHealth());
int nCurrPercent = (int)(100.0f * GetHealth() / GetMaxHealth());
if (( (nPrevPercent + 9) / 10 ) != ( (nCurrPercent + 9) / 10 ))
{
m_OnHealthChanged.Set( nCurrPercent, this, this );
}
}
return nRetVal;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void Chopper_BecomeChunks( CBaseEntity *pChopper )
{
QAngle vecChunkAngles = pChopper->GetAbsAngles();
Vector vecForward, vecUp;
pChopper->GetVectors( &vecForward, NULL, &vecUp );
#ifdef HL2_EPISODIC
CNPC_AttackHelicopter *pAttackHelicopter;
pAttackHelicopter = dynamic_cast<CNPC_AttackHelicopter*>(pChopper);
if( pAttackHelicopter != NULL )
{
// New for EP2, we may be tailspinning, (crashing) and playing an animation that is spinning
// our root bone, which means our model is not facing the way our entity is facing. So we have
// to do some attachment point math to get the proper angles to use for computing the relative
// positions of the gibs. The attachment points called DAMAGE0 is properly oriented and attached
// to the chopper body so we can use its angles.
int iAttach = pAttackHelicopter->LookupAttachment( "damage0" );
Vector vecAttachPos;
if( iAttach > -1 )
{
pAttackHelicopter->GetAttachment(iAttach, vecAttachPos, vecChunkAngles );
AngleVectors( vecChunkAngles, &vecForward, NULL, &vecUp );
}
}
#endif//HL2_EPISODIC
Vector vecChunkPos = pChopper->GetAbsOrigin();
Vector vecRight(0,0,0);
if( hl2_episodic.GetBool() )
{
// We need to get a right hand vector to toss the cockpit and tail pieces
// so their motion looks like a continuation of the tailspin animation
// that the chopper plays before crashing.
pChopper->GetVectors( NULL, &vecRight, NULL );
}
// Body
CHelicopterChunk *pBodyChunk = CHelicopterChunk::CreateHelicopterChunk( vecChunkPos, vecChunkAngles, pChopper->GetAbsVelocity(), HELICOPTER_CHUNK_BODY, CHUNK_BODY );
Chopper_CreateChunk( pChopper, vecChunkPos, RandomAngle( 0, 360 ), s_pChunkModelName[random->RandomInt( 0, CHOPPER_MAX_CHUNKS - 1 )], false );
vecChunkPos = pChopper->GetAbsOrigin() + ( vecForward * 100.0f ) + ( vecUp * -38.0f );
// Cockpit
CHelicopterChunk *pCockpitChunk = CHelicopterChunk::CreateHelicopterChunk( vecChunkPos, vecChunkAngles, pChopper->GetAbsVelocity() + vecRight * -800.0f, HELICOPTER_CHUNK_COCKPIT, CHUNK_COCKPIT );
Chopper_CreateChunk( pChopper, vecChunkPos, RandomAngle( 0, 360 ), s_pChunkModelName[random->RandomInt( 0, CHOPPER_MAX_CHUNKS - 1 )], false );
pCockpitChunk->m_hMaster = pBodyChunk;
vecChunkPos = pChopper->GetAbsOrigin() + ( vecForward * -175.0f );
// Tail
CHelicopterChunk *pTailChunk = CHelicopterChunk::CreateHelicopterChunk( vecChunkPos, vecChunkAngles, pChopper->GetAbsVelocity() + vecRight * 800.0f, HELICOPTER_CHUNK_TAIL, CHUNK_TAIL );
Chopper_CreateChunk( pChopper, vecChunkPos, RandomAngle( 0, 360 ), s_pChunkModelName[random->RandomInt( 0, CHOPPER_MAX_CHUNKS - 1 )], false );
pTailChunk->m_hMaster = pBodyChunk;
// Constrain all the pieces together loosely
IPhysicsObject *pBodyObject = pBodyChunk->VPhysicsGetObject();
Assert( pBodyObject );
IPhysicsObject *pCockpitObject = pCockpitChunk->VPhysicsGetObject();
Assert( pCockpitObject );
IPhysicsObject *pTailObject = pTailChunk->VPhysicsGetObject();
Assert( pTailObject );
IPhysicsConstraintGroup *pGroup = NULL;
// Create the constraint
constraint_fixedparams_t fixed;
fixed.Defaults();
fixed.InitWithCurrentObjectState( pBodyObject, pTailObject );
fixed.constraint.Defaults();
pBodyChunk->m_pTailConstraint = physenv->CreateFixedConstraint( pBodyObject, pTailObject, pGroup, fixed );
fixed.Defaults();
fixed.InitWithCurrentObjectState( pBodyObject, pCockpitObject );
fixed.constraint.Defaults();
pBodyChunk->m_pCockpitConstraint = physenv->CreateFixedConstraint( pBodyObject, pCockpitObject, pGroup, fixed );
}
//-----------------------------------------------------------------------------
// Purpose: Start us crashing
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::Event_Killed( const CTakeDamageInfo &info )
{
if( m_lifeState == LIFE_ALIVE )
{
m_OnShotDown.FireOutput( this, this );
}
m_lifeState = LIFE_DYING;
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
controller.SoundChangeVolume( m_pGunFiringSound, 0.0, 0.1f );
if( GetCrashPoint() == NULL )
{
CBaseEntity *pCrashPoint = gEntList.FindEntityByClassname( NULL, "info_target_helicopter_crash" );
if( pCrashPoint != NULL )
{
m_hCrashPoint.Set( pCrashPoint );
SetDesiredPosition( pCrashPoint->GetAbsOrigin() );
// Start the failing engine sound
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
controller.SoundDestroy( m_pRotorSound );
CPASAttenuationFilter filter( this );
m_pRotorSound = controller.SoundCreate( filter, entindex(), "NPC_AttackHelicopter.EngineFailure" );
controller.Play( m_pRotorSound, 1.0, 100 );
// Tailspin!!
SetActivity( ACT_HELICOPTER_CRASHING );
// Intentionally returning with m_lifeState set to LIFE_DYING
return;
}
}
Chopper_BecomeChunks( this );
StopLoopingSounds();
m_lifeState = LIFE_DEAD;
EmitSound( "NPC_CombineGunship.Explode" );
SetThink( &CNPC_AttackHelicopter::SUB_Remove );
SetNextThink( gpGlobals->curtime + 0.1f );
AddEffects( EF_NODRAW );
// Makes the slower rotors fade back in
SetStartupTime( gpGlobals->curtime + 99.0f );
m_iHealth = 0;
m_takedamage = DAMAGE_NO;
m_OnDeath.FireOutput( info.GetAttacker(), this );
}
//------------------------------------------------------------------------------
// Creates the breakable husk of an attack chopper
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::CreateChopperHusk()
{
// We're embedded into the ground
CBaseEntity *pCorpse = CreateEntityByName( "prop_physics" );
pCorpse->SetAbsOrigin( GetAbsOrigin() );
pCorpse->SetAbsAngles( GetAbsAngles() );
pCorpse->SetModel( CHOPPER_MODEL_CORPSE_NAME );
pCorpse->AddSpawnFlags( SF_PHYSPROP_MOTIONDISABLED );
pCorpse->Spawn();
pCorpse->SetMoveType( MOVETYPE_NONE );
}
//-----------------------------------------------------------------------------
// Think!
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::PrescheduleThink( void )
{
if ( m_flGoalRollDmg != 0.0f )
{
m_flGoalRollDmg = UTIL_Approach( 0, m_flGoalRollDmg, 2.0f );
}
switch( m_lifeState )
{
case LIFE_DYING:
{
if( GetCrashPoint() != NULL )
{
// Stay on this, no matter what.
SetDesiredPosition( GetCrashPoint()->WorldSpaceCenter() );
}
if ( random->RandomInt( 0, 4 ) == 0 )
{
Vector explodePoint;
CollisionProp()->RandomPointInBounds( Vector(0.25,0.25,0.25), Vector(0.75,0.75,0.75), &explodePoint );
ExplodeAndThrowChunk( explodePoint );
}
}
break;
}
BaseClass::PrescheduleThink();
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
float CNPC_AttackHelicopter::UpdatePerpPathDistance( float flMaxPathOffset )
{
if ( !IsLeading() || !GetEnemy() )
{
m_flCurrPathOffset = 0.0f;
return 0.0f;
}
float flNewPathOffset = TargetDistanceToPath();
// Make bomb dropping more interesting
if ( ShouldDropBombs() )
{
float flSpeedAlongPath = TargetSpeedAlongPath();
if ( flSpeedAlongPath > 10.0f )
{
float flLeadTime = GetLeadingDistance() / flSpeedAlongPath;
flLeadTime = clamp( flLeadTime, 0.0f, 2.0f );
flNewPathOffset += 0.25 * flLeadTime * TargetSpeedAcrossPath();
}
flSpeedAlongPath = clamp( flSpeedAlongPath, 100.0f, 500.0f );
float flSinHeight = SimpleSplineRemapVal( flSpeedAlongPath, 100.0f, 500.0f, 0.0f, 200.0f );
flNewPathOffset += flSinHeight * sin( 2.0f * M_PI * (gpGlobals->curtime / 6.0f) );
}
if ( (flMaxPathOffset != 0.0f) && (flNewPathOffset > flMaxPathOffset) )
{
flNewPathOffset = flMaxPathOffset;
}
float flMaxChange = 1000.0f * (gpGlobals->curtime - GetLastThink());
if ( fabs( flNewPathOffset - m_flCurrPathOffset ) < flMaxChange )
{
m_flCurrPathOffset = flNewPathOffset;
}
else
{
float flSign = (m_flCurrPathOffset < flNewPathOffset) ? 1.0f : -1.0f;
m_flCurrPathOffset += flSign * flMaxChange;
}
return m_flCurrPathOffset;
}
//-----------------------------------------------------------------------------
// Computes the max speed + acceleration:
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::GetMaxSpeedAndAccel( float *pMaxSpeed, float *pAccelRate )
{
*pAccelRate = CHOPPER_ACCEL_RATE;
*pMaxSpeed = GetMaxSpeed();
if ( GetEnemyVehicle() )
{
*pAccelRate *= 9.0f;
}
}
//-----------------------------------------------------------------------------
// Computes the acceleration:
//-----------------------------------------------------------------------------
#define HELICOPTER_GRAVITY 384
#define HELICOPTER_DT 0.1f
#define HELICOPTER_MIN_DZ_DAMP -500.0f
#define HELICOPTER_MAX_DZ_DAMP -1000.0f
#define HELICOPTER_FORCE_BLEND 0.8f
#define HELICOPTER_FORCE_BLEND_VEHICLE 0.2f
void CNPC_AttackHelicopter::ComputeVelocity( const Vector &vecTargetPosition,
float flAdditionalHeight, float flMinDistFromSegment, float flMaxDistFromSegment, Vector *pVecAccel )
{
Vector deltaPos;
VectorSubtract( vecTargetPosition, GetAbsOrigin(), deltaPos );
// calc goal linear accel to hit deltaPos in dt time.
// This is solving the equation xf = 0.5 * a * dt^2 + vo * dt + xo
float dt = 1.0f;
pVecAccel->x = 2.0f * (deltaPos.x - GetAbsVelocity().x * dt) / (dt * dt);
pVecAccel->y = 2.0f * (deltaPos.y - GetAbsVelocity().y * dt) / (dt * dt);
pVecAccel->z = 2.0f * (deltaPos.z - GetAbsVelocity().z * dt) / (dt * dt) + HELICOPTER_GRAVITY;
float flDistFromPath = 0.0f;
Vector vecPoint, vecDelta;
if ( flMaxDistFromSegment != 0.0f )
{
// Also, add in a little force to get us closer to our current line segment if we can
ClosestPointToCurrentPath( &vecPoint );
if ( flAdditionalHeight != 0.0f )
{
Vector vecEndPoint, vecClosest;
vecEndPoint = vecPoint;
vecEndPoint.z += flAdditionalHeight;
CalcClosestPointOnLineSegment( GetAbsOrigin(), vecPoint, vecEndPoint, vecClosest );
vecPoint = vecClosest;
}
VectorSubtract( vecPoint, GetAbsOrigin(), vecDelta );
flDistFromPath = VectorNormalize( vecDelta );
if ( flDistFromPath > flMaxDistFromSegment )
{
// Strongly constrain to an n unit pipe around the current path
// by damping out all impulse forces that would push us further from the pipe
float flAmount = (flDistFromPath - flMaxDistFromSegment) / 200.0f;
flAmount = clamp( flAmount, 0, 1 );
VectorMA( *pVecAccel, flAmount * 200.0f, vecDelta, *pVecAccel );
}
}
// Apply avoidance forces
if ( !HasSpawnFlags( SF_HELICOPTER_IGNORE_AVOID_FORCES ) )
{
Vector vecAvoidForce;
CAvoidSphere::ComputeAvoidanceForces( this, 350.0f, 2.0f, &vecAvoidForce );
*pVecAccel += vecAvoidForce;
CAvoidBox::ComputeAvoidanceForces( this, 350.0f, 2.0f, &vecAvoidForce );
*pVecAccel += vecAvoidForce;
}
// don't fall faster than 0.2G or climb faster than 2G
pVecAccel->z = clamp( pVecAccel->z, HELICOPTER_GRAVITY * 0.2f, HELICOPTER_GRAVITY * 2.0f );
// The lift factor owing to horizontal movement
float flHorizLiftFactor = fabs( pVecAccel->x ) * 0.10f + fabs( pVecAccel->y ) * 0.10f;
// If we're way above the path, dampen horizontal lift factor
float flNewHorizLiftFactor = clamp( deltaPos.z, HELICOPTER_MAX_DZ_DAMP, HELICOPTER_MIN_DZ_DAMP );
flNewHorizLiftFactor = SimpleSplineRemapVal( flNewHorizLiftFactor, HELICOPTER_MIN_DZ_DAMP, HELICOPTER_MAX_DZ_DAMP, flHorizLiftFactor, 2.5f * (HELICOPTER_GRAVITY * 0.2) );
float flDampening = (flNewHorizLiftFactor != 0.0f) ? (flNewHorizLiftFactor / flHorizLiftFactor) : 1.0f;
if ( flDampening < 1.0f )
{
pVecAccel->x *= flDampening;
pVecAccel->y *= flDampening;
flHorizLiftFactor = flNewHorizLiftFactor;
}
Vector forward, right, up;
GetVectors( &forward, &right, &up );
// First, attenuate the current force
float flForceBlend = GetEnemyVehicle() ? HELICOPTER_FORCE_BLEND_VEHICLE : HELICOPTER_FORCE_BLEND;
m_flForce *= flForceBlend;
// Now add force based on our acceleration factors
m_flForce += ( pVecAccel->z + flHorizLiftFactor ) * HELICOPTER_DT * (1.0f - flForceBlend);
// The force is always *locally* upward based; we pitch + roll the chopper to get movement
Vector vecImpulse;
VectorMultiply( up, m_flForce, vecImpulse );
// NOTE: These have to be done *before* the additional path distance drag forces are applied below
ApplySidewaysDrag( right );
ApplyGeneralDrag();
// If LIFE_DYING, maintain control as long as we're flying to a crash point.
if ( m_lifeState != LIFE_DYING || (m_lifeState == LIFE_DYING && GetCrashPoint() != NULL) )
{
vecImpulse.z += -HELICOPTER_GRAVITY * HELICOPTER_DT;
if ( flMinDistFromSegment != 0.0f && ( flDistFromPath > flMinDistFromSegment ) )
{
Vector vecVelDir = GetAbsVelocity();
// Strongly constrain to an n unit pipe around the current path
// by damping out all impulse forces that would push us further from the pipe
float flDot = DotProduct( vecImpulse, vecDelta );
if ( flDot < 0.0f )
{
VectorMA( vecImpulse, -flDot * 0.1f, vecDelta, vecImpulse );
}
// Also apply an extra impulse to compensate for the current velocity
flDot = DotProduct( vecVelDir, vecDelta );
if ( flDot < 0.0f )
{
VectorMA( vecImpulse, -flDot * 0.1f, vecDelta, vecImpulse );
}
}
}
else
{
// No more upward lift...
vecImpulse.z = -HELICOPTER_GRAVITY * HELICOPTER_DT;
// Damp the horizontal impulses; we should pretty much be falling ballistically
vecImpulse.x *= 0.1f;
vecImpulse.y *= 0.1f;
}
// Add in our velocity pulse for this frame
ApplyAbsVelocityImpulse( vecImpulse );
}
//-----------------------------------------------------------------------------
// Computes the max speed + acceleration:
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::ComputeAngularVelocity( const Vector &vecGoalUp, const Vector &vecFacingDirection )
{
QAngle goalAngAccel;
if ( m_lifeState != LIFE_DYING || (m_lifeState == LIFE_DYING && GetCrashPoint() != NULL) )
{
Vector forward, right, up;
GetVectors( &forward, &right, &up );
Vector goalUp = vecGoalUp;
VectorNormalize( goalUp );
// calc goal orientation to hit linear accel forces
float goalPitch = RAD2DEG( asin( DotProduct( forward, goalUp ) ) );
float goalYaw = UTIL_VecToYaw( vecFacingDirection );
float goalRoll = RAD2DEG( asin( DotProduct( right, goalUp ) ) + m_flGoalRollDmg );
goalPitch *= 0.75f;
// clamp goal orientations
goalPitch = clamp( goalPitch, -30, 45 );
goalRoll = clamp( goalRoll, -45, 45 );
// calc angular accel needed to hit goal pitch in dt time.
float dt = 0.6;
goalAngAccel.x = 2.0 * (AngleDiff( goalPitch, AngleNormalize( GetAbsAngles().x ) ) - GetLocalAngularVelocity().x * dt) / (dt * dt);
goalAngAccel.y = 2.0 * (AngleDiff( goalYaw, AngleNormalize( GetAbsAngles().y ) ) - GetLocalAngularVelocity().y * dt) / (dt * dt);
goalAngAccel.z = 2.0 * (AngleDiff( goalRoll, AngleNormalize( GetAbsAngles().z ) ) - GetLocalAngularVelocity().z * dt) / (dt * dt);
goalAngAccel.x = clamp( goalAngAccel.x, -300, 300 );
//goalAngAccel.y = clamp( goalAngAccel.y, -60, 60 );
goalAngAccel.y = clamp( goalAngAccel.y, -120, 120 );
goalAngAccel.z = clamp( goalAngAccel.z, -300, 300 );
}
else
{
goalAngAccel.x = 0;
goalAngAccel.y = random->RandomFloat( 50, 120 );
goalAngAccel.z = 0;
}
// limit angular accel changes to similate mechanical response times
QAngle angAccelAccel;
float dt = 0.1;
angAccelAccel.x = (goalAngAccel.x - m_vecAngAcceleration.x) / dt;
angAccelAccel.y = (goalAngAccel.y - m_vecAngAcceleration.y) / dt;
angAccelAccel.z = (goalAngAccel.z - m_vecAngAcceleration.z) / dt;
angAccelAccel.x = clamp( angAccelAccel.x, -1000, 1000 );
angAccelAccel.y = clamp( angAccelAccel.y, -1000, 1000 );
angAccelAccel.z = clamp( angAccelAccel.z, -1000, 1000 );
// DevMsg( "pitch %6.1f (%6.1f:%6.1f) ", goalPitch, GetLocalAngles().x, m_vecAngVelocity.x );
// DevMsg( "roll %6.1f (%6.1f:%6.1f) : ", goalRoll, GetLocalAngles().z, m_vecAngVelocity.z );
// DevMsg( "%6.1f %6.1f %6.1f : ", goalAngAccel.x, goalAngAccel.y, goalAngAccel.z );
// DevMsg( "%6.0f %6.0f %6.0f\n", angAccelAccel.x, angAccelAccel.y, angAccelAccel.z );
m_vecAngAcceleration += angAccelAccel * 0.1;
QAngle angVel = GetLocalAngularVelocity();
angVel += m_vecAngAcceleration * 0.1;
angVel.y = clamp( angVel.y, -120, 120 );
// Fix up pitch and yaw to tend toward small values
if ( m_lifeState == LIFE_DYING && GetCrashPoint() == NULL )
{
float flPitchDiff = random->RandomFloat( -5, 5 ) - GetAbsAngles().x;
angVel.x = flPitchDiff * 0.1f;
float flRollDiff = random->RandomFloat( -5, 5 ) - GetAbsAngles().z;
angVel.z = flRollDiff * 0.1f;
}
SetLocalAngularVelocity( angVel );
float flAmt = clamp( angVel.y, -30, 30 );
float flRudderPose = RemapVal( flAmt, -30, 30, 45, -45 );
SetPoseParameter( "rudder", flRudderPose );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::FlightDirectlyOverhead( void )
{
Vector vecTargetPosition = m_vecTargetPosition;
CBaseEntity *pEnemy = GetEnemy();
if ( HasEnemy() && FVisible( pEnemy ) )
{
if ( GetEnemy()->IsPlayer() )
{
CBaseEntity *pEnemyVehicle = assert_cast<CBasePlayer*>(GetEnemy())->GetVehicleEntity();
if ( pEnemyVehicle )
{
Vector vecEnemyVel = pEnemyVehicle->GetSmoothedVelocity();
Vector vecRelativePosition;
VectorSubtract( GetAbsOrigin(), pEnemyVehicle->GetAbsOrigin(), vecRelativePosition );
float flDist = VectorNormalize( vecRelativePosition );
float flEnemySpeed = VectorNormalize( vecEnemyVel );
float flDot = DotProduct( vecRelativePosition, vecEnemyVel );
float flSpeed = GetMaxSpeed() * 0.3f; //GetAbsVelocity().Length();
float a = flSpeed * flSpeed - flEnemySpeed * flEnemySpeed;
float b = 2.0f * flEnemySpeed * flDist * flDot;
float c = - flDist * flDist;
float flDiscrim = b * b - 4 * a * c;
if ( flDiscrim >= 0 )
{
float t = ( -b + sqrt( flDiscrim ) ) / (2 * a);
t = clamp( t, 0.0f, 4.0f );
VectorMA( pEnemyVehicle->GetAbsOrigin(), t * flEnemySpeed, vecEnemyVel, vecTargetPosition );
}
}
}
}
// if ( GetCurrentPathTargetPosition() )
// {
// vecTargetPosition.z = GetCurrentPathTargetPosition()->z;
// }
NDebugOverlay::Cross3D( vecTargetPosition, -Vector(32,32,32), Vector(32,32,32), 0, 0, 255, true, 0.1f );
UpdateFacingDirection( vecTargetPosition );
Vector accel;
ComputeVelocity( vecTargetPosition, 0.0f, 0.0f, 0.0f, &accel );
ComputeAngularVelocity( accel, m_vecDesiredFaceDir );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::Flight( void )
{
if( GetFlags() & FL_ONGROUND )
{
// This would be really bad.
SetGroundEntity( NULL );
}
// Determine the distances we must lie from the path
float flMaxPathOffset = MaxDistanceFromCurrentPath();
float flPerpDist = UpdatePerpPathDistance( flMaxPathOffset );
float flMinDistFromSegment, flMaxDistFromSegment;
if ( !IsLeading() )
{
flMinDistFromSegment = 0.0f;
flMaxDistFromSegment = 0.0f;
}
else
{
flMinDistFromSegment = fabs(flPerpDist) + 100.0f;
flMaxDistFromSegment = fabs(flPerpDist) + 200.0f;
if ( flMaxPathOffset != 0.0 )
{
if ( flMaxDistFromSegment > flMaxPathOffset - 100.0f )
{
flMaxDistFromSegment = flMaxPathOffset - 100.0f;
}
if ( flMinDistFromSegment > flMaxPathOffset - 200.0f )
{
flMinDistFromSegment = flMaxPathOffset - 200.0f;
}
}
}
float maxSpeed, accelRate;
GetMaxSpeedAndAccel( &maxSpeed, &accelRate );
Vector vecTargetPosition;
float flCurrentSpeed = GetAbsVelocity().Length();
float flDist = MIN( flCurrentSpeed + accelRate, maxSpeed );
float dt = 1.0f;
ComputeActualTargetPosition( flDist, dt, flPerpDist, &vecTargetPosition );
// Raise high in the air when doing the shooting attack
float flAdditionalHeight = 0.0f;
if ( m_nAttackMode == ATTACK_MODE_BULLRUSH_VEHICLE )
{
flAdditionalHeight = clamp( m_flBullrushAdditionalHeight, 0.0f, flMaxPathOffset );
vecTargetPosition.z += flAdditionalHeight;
}
Vector accel;
UpdateFacingDirection( vecTargetPosition );
ComputeVelocity( vecTargetPosition, flAdditionalHeight, flMinDistFromSegment, flMaxDistFromSegment, &accel );
ComputeAngularVelocity( accel, m_vecDesiredFaceDir );
}
//------------------------------------------------------------------------------
// Updates the facing direction
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::UpdateFacingDirection( const Vector &vecActualDesiredPosition )
{
bool bIsBullrushing = ( m_nAttackMode == ATTACK_MODE_BULLRUSH_VEHICLE );
bool bSeenTargetRecently = HasSpawnFlags( SF_HELICOPTER_AGGRESSIVE ) || ( m_flLastSeen + 5 > gpGlobals->curtime );
if ( GetEnemy() && !bIsBullrushing )
{
if ( !IsLeading() )
{
if( IsCarpetBombing() && hl2_episodic.GetBool() )
{
m_vecDesiredFaceDir = vecActualDesiredPosition - GetAbsOrigin();
}
else if ( !IsCrashing() && bSeenTargetRecently )
{
// If we've seen the target recently, face the target.
m_vecDesiredFaceDir = m_vecTargetPosition - GetAbsOrigin();
}
else
{
// Remain facing the way you were facing...
}
}
else
{
if ( ShouldDropBombs() || IsCarpetBombing() )
{
m_vecDesiredFaceDir = vecActualDesiredPosition - GetAbsOrigin();
}
else
{
m_vecDesiredFaceDir = m_vecTargetPosition - GetAbsOrigin();
}
}
}
else
{
// Face our desired position
float flDistSqr = vecActualDesiredPosition.AsVector2D().DistToSqr( GetAbsOrigin().AsVector2D() );
if ( flDistSqr <= 50 * 50 )
{
if (( flDistSqr > 1 * 1 ) && bSeenTargetRecently && IsInSecondaryMode( BULLRUSH_MODE_SHOOT_IDLE_PLAYER ) )
{
m_vecDesiredFaceDir = m_vecTargetPosition - GetAbsOrigin();
m_vecDesiredFaceDir.z = 0.0f;
}
else
{
GetVectors( &m_vecDesiredFaceDir, NULL, NULL );
}
}
else
{
m_vecDesiredFaceDir = vecActualDesiredPosition - GetAbsOrigin();
}
}
VectorNormalize( m_vecDesiredFaceDir );
}
//------------------------------------------------------------------------------
// Purpose :
//------------------------------------------------------------------------------
#define ENEMY_CREEP_RATE 400
float CNPC_AttackHelicopter::CreepTowardEnemy( float flSpeed, float flMinSpeed, float flMaxSpeed, float flMinDist, float flMaxDist )
{
float dt = gpGlobals->curtime - GetLastThink();
float flEnemyCreepDist = ENEMY_CREEP_RATE * dt;
// When the player is slow, creep toward him within a second or two
float flLeadingDist = ClampSplineRemapVal( flSpeed, flMinSpeed, flMaxSpeed, flMinDist, flMaxDist );
float flCurrentDist = GetLeadingDistance( );
if ( fabs(flLeadingDist - flCurrentDist) > flEnemyCreepDist )
{
float flSign = ( flLeadingDist < flCurrentDist ) ? -1.0f : 1.0f;
flLeadingDist = flCurrentDist + flSign * flEnemyCreepDist;
}
return flLeadingDist;
}
#define MIN_ENEMY_SPEED 300
//------------------------------------------------------------------------------
// Computes how far to lead the player when bombing
//------------------------------------------------------------------------------
float CNPC_AttackHelicopter::ComputeBombingLeadingDistance( float flSpeed, float flSpeedAlongPath, bool bEnemyInVehicle )
{
if ( ( flSpeed <= MIN_ENEMY_SPEED ) && bEnemyInVehicle )
{
return CreepTowardEnemy( flSpeed, 0.0f, MIN_ENEMY_SPEED, 0.0f, 1000.0f );
}
return ClampSplineRemapVal( flSpeedAlongPath, 200.0f, 600.0f, 1000.0f, 2000.0f );
}
//------------------------------------------------------------------------------
// Computes how far to lead the player when bullrushing
//------------------------------------------------------------------------------
float CNPC_AttackHelicopter::ComputeBullrushLeadingDistance( float flSpeed, float flSpeedAlongPath, bool bEnemyInVehicle )
{
switch ( m_nSecondaryMode )
{
case BULLRUSH_MODE_WAIT_FOR_ENEMY:
return 0.0f;
case BULLRUSH_MODE_GET_DISTANCE:
return m_bRushForward ? -CHOPPER_BULLRUSH_MODE_DISTANCE : CHOPPER_BULLRUSH_MODE_DISTANCE;
case BULLRUSH_MODE_DROP_BOMBS_FOLLOW_PLAYER:
// return m_bRushForward ? 1500.0f : -1500.0f;
return ComputeBombingLeadingDistance( flSpeed, flSpeedAlongPath, bEnemyInVehicle );
case BULLRUSH_MODE_SHOOT_IDLE_PLAYER:
return 0.0f;
case BULLRUSH_MODE_DROP_BOMBS_FIXED_SPEED:
return m_bRushForward ? 7000 : -7000;
case BULLRUSH_MODE_MEGA_BOMB:
return m_bRushForward ? CHOPPER_BULLRUSH_MODE_DISTANCE : -CHOPPER_BULLRUSH_MODE_DISTANCE;
case BULLRUSH_MODE_SHOOT_GUN:
{
float flLeadDistance = 1000.f - CHOPPER_BULLRUSH_ENEMY_BOMB_DISTANCE;
return m_bRushForward ? flLeadDistance : -flLeadDistance;
}
}
Assert(0);
return 0.0f;
}
//------------------------------------------------------------------------------
// Secondary mode
//------------------------------------------------------------------------------
inline void CNPC_AttackHelicopter::SetSecondaryMode( int nMode, bool bRetainTime )
{
m_nSecondaryMode = nMode;
if (!bRetainTime)
{
m_flSecondaryModeStartTime = gpGlobals->curtime;
}
}
inline bool CNPC_AttackHelicopter::IsInSecondaryMode( int nMode )
{
return m_nSecondaryMode == nMode;
}
inline float CNPC_AttackHelicopter::SecondaryModeTime( ) const
{
return gpGlobals->curtime - m_flSecondaryModeStartTime;
}
//------------------------------------------------------------------------------
// Switch to idle
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::SwitchToBullrushIdle( void )
{
// Put us directly into idle gun state (we're in firing state)
m_flNextAttack = gpGlobals->curtime;
m_nGunState = GUN_STATE_IDLE;
m_nRemainingBursts = 0;
m_flBullrushAdditionalHeight = 0.0f;
SetPauseState( PAUSE_NO_PAUSE );
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
controller.SoundChangeVolume( m_pGunFiringSound, 0.0, 0.1f );
}
//------------------------------------------------------------------------------
// Should the chopper shoot the idle player?
//------------------------------------------------------------------------------
bool CNPC_AttackHelicopter::ShouldShootIdlePlayerInBullrush()
{
// Once he starts shooting, then don't stop until the player is moving pretty fast
float flSpeedSqr = IsInSecondaryMode( BULLRUSH_MODE_SHOOT_IDLE_PLAYER ) ? CHOPPER_BULLRUSH_SLOW_SHOOT_SPEED_2_SQ : CHOPPER_BULLRUSH_SLOW_SHOOT_SPEED_SQ;
return ( GetEnemy() && GetEnemy()->GetSmoothedVelocity().LengthSqr() <= flSpeedSqr );
}
//------------------------------------------------------------------------------
// Shutdown shooting during bullrush
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::ShutdownGunDuringBullrush( )
{
// Put us directly into idle gun state (we're in firing state)
m_flNextAttack = gpGlobals->curtime;
m_nGunState = GUN_STATE_IDLE;
m_nRemainingBursts = 0;
SetPauseState( PAUSE_NO_PAUSE );
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
controller.SoundChangeVolume( m_pGunFiringSound, 0.0, 0.1f );
}
#define HELICOPTER_MIN_IDLE_BOMBING_DIST 350.0f
#define HELICOPTER_MIN_IDLE_BOMBING_SPEED 350.0f
//-----------------------------------------------------------------------------
// Purpose:
// Output : Returns true on success, false on failure.
//-----------------------------------------------------------------------------
bool CNPC_AttackHelicopter::ShouldBombIdlePlayer( void )
{
// Must be settled over a position and not moving too quickly to do this
if ( GetAbsVelocity().LengthSqr() > Square(HELICOPTER_MIN_IDLE_BOMBING_SPEED) )
return false;
// Must be within a certain range of the target
float flDistToTargetSqr = (GetEnemy()->WorldSpaceCenter() - GetAbsOrigin()).Length2DSqr();
if ( flDistToTargetSqr < Square(HELICOPTER_MIN_IDLE_BOMBING_DIST) )
return true;
// Can't bomb this
return false;
}
//------------------------------------------------------------------------------
// Update the bullrush state
//------------------------------------------------------------------------------
#define BULLRUSH_GOAL_TOLERANCE 200
#define BULLRUSH_BOMB_MAX_DISTANCE 3500
void CNPC_AttackHelicopter::UpdateBullrushState( void )
{
if ( !GetEnemy() || IsInForcedMove() )
{
if ( !IsInSecondaryMode( BULLRUSH_MODE_WAIT_FOR_ENEMY ) )
{
SwitchToBullrushIdle();
SetSecondaryMode( BULLRUSH_MODE_WAIT_FOR_ENEMY );
}
}
switch( m_nSecondaryMode )
{
case BULLRUSH_MODE_WAIT_FOR_ENEMY:
{
m_flBullrushAdditionalHeight = CHOPPER_BULLRUSH_SHOOTING_VERTICAL_OFFSET;
if ( GetEnemy() && !IsInForcedMove() )
{
// This forces us to not start trying checking positions
// until we have been on the path for a little while
if ( SecondaryModeTime() > 0.3f )
{
float flDistanceToGoal = ComputeDistanceToTargetPosition();
Vector vecPathDir;
CurrentPathDirection( &vecPathDir );
bool bMovingForward = DotProduct2D( GetAbsVelocity().AsVector2D(), vecPathDir.AsVector2D() ) >= 0.0f;
if ( flDistanceToGoal * (bMovingForward ? 1.0f : -1.0f) > 1000 )
{
m_bRushForward = bMovingForward;
SetSecondaryMode( BULLRUSH_MODE_SHOOT_GUN );
SpotlightStartup();
}
else
{
m_bRushForward = !bMovingForward;
SetSecondaryMode( BULLRUSH_MODE_GET_DISTANCE );
}
}
}
else
{
m_flSecondaryModeStartTime = gpGlobals->curtime;
}
}
break;
case BULLRUSH_MODE_GET_DISTANCE:
{
m_flBullrushAdditionalHeight = CHOPPER_BULLRUSH_SHOOTING_VERTICAL_OFFSET;
float flDistanceToGoal = ComputeDistanceToTargetPosition();
if ( m_bRushForward )
{
if ( flDistanceToGoal < (CHOPPER_BULLRUSH_MODE_DISTANCE - 1000) )
break;
}
else
{
if ( flDistanceToGoal > -(CHOPPER_BULLRUSH_MODE_DISTANCE - 1000) )
break;
}
if ( GetHealth() <= m_flNextMegaBombHealth )
{
m_flNextMegaBombHealth -= GetMaxHealth() * g_helicopter_bullrush_mega_bomb_health.GetFloat();
m_flNextBullrushBombTime = gpGlobals->curtime;
SetSecondaryMode( BULLRUSH_MODE_MEGA_BOMB );
EmitSound( "NPC_AttackHelicopter.MegabombAlert" );
}
else
{
SetSecondaryMode( BULLRUSH_MODE_SHOOT_GUN );
SpotlightStartup();
}
}
break;
case BULLRUSH_MODE_MEGA_BOMB:
{
m_flBullrushAdditionalHeight = CHOPPER_BULLRUSH_SHOOTING_VERTICAL_OFFSET;
float flDistanceToGoal = ComputeDistanceToTargetPosition();
if ( m_bRushForward )
{
if ( flDistanceToGoal > -(CHOPPER_BULLRUSH_MODE_DISTANCE - 1000) )
break;
}
else
{
if ( flDistanceToGoal < (CHOPPER_BULLRUSH_MODE_DISTANCE - 1000) )
break;
}
m_bRushForward = !m_bRushForward;
SetSecondaryMode( BULLRUSH_MODE_GET_DISTANCE );
}
break;
case BULLRUSH_MODE_SHOOT_GUN:
{
// When shooting, stop when we cross the player's position
// Then start bombing. Use the fixed speed version if we're too far
// from the enemy or if he's travelling in the opposite direction.
// Otherwise, do the standard bombing behavior for a while.
float flDistanceToGoal = ComputeDistanceToTargetPosition();
float flShootingHeight = CHOPPER_BULLRUSH_SHOOTING_VERTICAL_OFFSET;
float flSwitchToBombDist = CHOPPER_BULLRUSH_ENEMY_BOMB_DISTANCE;
float flDropDownDist = 2000.0f;
if ( m_bRushForward )
{
m_flBullrushAdditionalHeight = ClampSplineRemapVal( flDistanceToGoal,
flSwitchToBombDist, flSwitchToBombDist + flDropDownDist, 0.0f, flShootingHeight );
if ( flDistanceToGoal > flSwitchToBombDist )
break;
}
else
{
m_flBullrushAdditionalHeight = ClampSplineRemapVal( flDistanceToGoal,
-flSwitchToBombDist - flDropDownDist, -flSwitchToBombDist, flShootingHeight, 0.0f );
if ( flDistanceToGoal < -flSwitchToBombDist )
break;
}
if ( ShouldShootIdlePlayerInBullrush() )
{
SetSecondaryMode( BULLRUSH_MODE_SHOOT_IDLE_PLAYER );
}
else
{
ShutdownGunDuringBullrush( );
SetSecondaryMode( BULLRUSH_MODE_DROP_BOMBS_FIXED_SPEED );
}
}
break;
case BULLRUSH_MODE_SHOOT_IDLE_PLAYER:
{
// Shut down our gun if we're switching to bombing
if ( ShouldBombIdlePlayer() )
{
// Must not already be shutdown
if (( m_nGunState != GUN_STATE_IDLE ) && (SecondaryModeTime() >= BULLRUSH_IDLE_PLAYER_FIRE_TIME))
{
ShutdownGunDuringBullrush( );
}
}
// m_nBurstHits = 0;
m_flCircleOfDeathRadius = ClampSplineRemapVal( SecondaryModeTime(), BULLRUSH_IDLE_PLAYER_FIRE_TIME, BULLRUSH_IDLE_PLAYER_FIRE_TIME + 5.0f, 256.0f, 64.0f );
m_flBullrushAdditionalHeight = CHOPPER_BULLRUSH_SHOOTING_VERTICAL_OFFSET;
if ( !ShouldShootIdlePlayerInBullrush() )
{
ShutdownGunDuringBullrush( );
SetSecondaryMode( BULLRUSH_MODE_DROP_BOMBS_FIXED_SPEED );
}
}
break;
case BULLRUSH_MODE_DROP_BOMBS_FOLLOW_PLAYER:
{
m_flBullrushAdditionalHeight = 0.0f;
float flDistanceToGoal = ComputeDistanceToTargetPosition();
if ( fabs( flDistanceToGoal ) > 2000.0f )
{
SetSecondaryMode( BULLRUSH_MODE_DROP_BOMBS_FIXED_SPEED, true );
break;
}
}
// FALL THROUGH!!
case BULLRUSH_MODE_DROP_BOMBS_FIXED_SPEED:
{
float flDistanceToGoal = ComputeDistanceToTargetPosition();
m_flBullrushAdditionalHeight = 0.0f;
if (( SecondaryModeTime() >= CHOPPER_BULLRUSH_ENEMY_BOMB_TIME ) || ( flDistanceToGoal > BULLRUSH_BOMB_MAX_DISTANCE ))
{
m_bRushForward = !m_bRushForward;
SetSecondaryMode( BULLRUSH_MODE_GET_DISTANCE );
}
}
break;
}
}
//------------------------------------------------------------------------------
// Purpose :
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::UpdateEnemyLeading( void )
{
bool bEnemyInVehicle = true;
CBaseEntity *pTarget = GetEnemyVehicle();
if ( !pTarget )
{
bEnemyInVehicle = false;
if ( (m_nAttackMode == ATTACK_MODE_DEFAULT) || !GetEnemy() )
{
EnableLeading( false );
return;
}
pTarget = GetEnemy();
}
EnableLeading( true );
float flLeadingDist = 0.0f;
float flSpeedAlongPath = TargetSpeedAlongPath();
float flSpeed = pTarget->GetSmoothedVelocity().Length();
// Do the test electricity gun
if ( HasSpawnFlags(SF_HELICOPTER_ELECTRICAL_DRONE) )
{
if ( flSpeedAlongPath < 200.0f )
{
flLeadingDist = ClampSplineRemapVal( flSpeedAlongPath, 0.0f, 200.0f, 100.0f, -200.0f );
}
else
{
flLeadingDist = ClampSplineRemapVal( flSpeedAlongPath, 200.0f, 600.0f, -200.0f, -500.0f );
}
SetLeadingDistance( flLeadingDist );
return;
}
switch( m_nAttackMode )
{
case ATTACK_MODE_BULLRUSH_VEHICLE:
flLeadingDist = ComputeBullrushLeadingDistance( flSpeed, flSpeedAlongPath, bEnemyInVehicle );
break;
case ATTACK_MODE_ALWAYS_LEAD_VEHICLE:
if (( flSpeed <= MIN_ENEMY_SPEED ) && (bEnemyInVehicle) )
{
flLeadingDist = CreepTowardEnemy( flSpeed, 0.0f, MIN_ENEMY_SPEED, 0.0f, 1000.0f );
}
else
{
if ( flSpeedAlongPath > 0.0f )
{
flLeadingDist = ClampSplineRemapVal( flSpeedAlongPath, 200.0f, 600.0f, 1000.0f, 2000.0f );
}
else
{
flLeadingDist = ClampSplineRemapVal( flSpeedAlongPath, -600.0f, -200.0f, -2000.0f, -1000.0f );
}
}
break;
case ATTACK_MODE_BOMB_VEHICLE:
flLeadingDist = ComputeBombingLeadingDistance( flSpeed, flSpeedAlongPath, bEnemyInVehicle );
break;
case ATTACK_MODE_DEFAULT:
case ATTACK_MODE_TRAIL_VEHICLE:
if (( flSpeed <= MIN_ENEMY_SPEED ) && (bEnemyInVehicle))
{
flLeadingDist = CreepTowardEnemy( flSpeed, 150.0f, MIN_ENEMY_SPEED, 500.0f, -1000.0f );
}
else
{
flLeadingDist = ClampSplineRemapVal( flSpeedAlongPath, -600.0f, -200.0f, -2500.0f, -1000.0f );
}
break;
}
SetLeadingDistance( flLeadingDist );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *pInfo -
// bAlways -
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::SetTransmit( CCheckTransmitInfo *pInfo, bool bAlways )
{
// Are we already marked for transmission?
if ( pInfo->m_pTransmitEdict->Get( entindex() ) )
return;
BaseClass::SetTransmit( pInfo, bAlways );
// Make our smoke trails always come with us
for ( int i = 0; i < m_nSmokeTrailCount; i++ )
{
m_hSmokeTrail[i]->SetTransmit( pInfo, bAlways );
}
}
//------------------------------------------------------------------------------
// Purpose :
//------------------------------------------------------------------------------
void CNPC_AttackHelicopter::Hunt( void )
{
if ( m_lifeState == LIFE_DEAD )
{
return;
}
if ( m_lifeState == LIFE_DYING )
{
Flight();
UpdatePlayerDopplerShift( );
return;
}
// FIXME: Hack to allow us to change the firing distance
SetFarthestPathDist( GetMaxFiringDistance() );
UpdateEnemy();
// Give free knowledge of the enemy position if the chopper is "aggressive"
if ( HasSpawnFlags( SF_HELICOPTER_AGGRESSIVE ) && GetEnemy() )
{
m_vecTargetPosition = GetEnemy()->WorldSpaceCenter();
}
// Test for state transitions when in bullrush mode
if ( m_nAttackMode == ATTACK_MODE_BULLRUSH_VEHICLE )
{
UpdateBullrushState();
}
UpdateEnemyLeading();
UpdateTrackNavigation( );
Flight();
UpdatePlayerDopplerShift( );
FireWeapons();
if ( !(m_fHelicopterFlags & BITS_HELICOPTER_GUN_ON) )
{
// !!!HACKHACK This is a fairly unsavoury hack that allows the attack
// chopper to continue to carpet bomb even with the gun turned off
// (Normally the chopper will carpet bomb inside FireGun(), but FireGun()
// doesn't get called by the above call to FireWeapons() if the gun is turned off)
// Adding this little exception here lets me avoid going into the CBaseHelicopter and
// making some functions virtual that don't want to be virtual.
if ( IsCarpetBombing() )
{
BullrushBombs();
}
}
#ifdef HL2_EPISODIC
// Update our bone followers
m_BoneFollowerManager.UpdateBoneFollowers(this);
#endif // HL2_EPISODIC
}
//-----------------------------------------------------------------------------
// Purpose: Cache whatever pose parameters we intend to use
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::PopulatePoseParameters( void )
{
m_poseWeapon_Pitch = LookupPoseParameter("weapon_pitch");
m_poseWeapon_Yaw = LookupPoseParameter("weapon_yaw");
m_poseRudder = LookupPoseParameter("rudder");
BaseClass::PopulatePoseParameters();
}
#ifdef HL2_EPISODIC
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CNPC_AttackHelicopter::InitBoneFollowers( void )
{
// Don't do this if we're already loaded
if ( m_BoneFollowerManager.GetNumBoneFollowers() != 0 )
return;
// Init our followers
m_BoneFollowerManager.InitBoneFollowers( this, ARRAYSIZE(pFollowerBoneNames), pFollowerBoneNames );
}
#endif // HL2_EPISODIC
//-----------------------------------------------------------------------------
// Where are how should we avoid?
//-----------------------------------------------------------------------------
AI_BEGIN_CUSTOM_NPC( npc_helicopter, CNPC_AttackHelicopter )
// DECLARE_TASK( )
DECLARE_ACTIVITY( ACT_HELICOPTER_DROP_BOMB );
DECLARE_ACTIVITY( ACT_HELICOPTER_CRASHING );
// DECLARE_CONDITION( COND_ )
//=========================================================
// DEFINE_SCHEDULE
// (
// SCHED_DUMMY,
//
// " Tasks"
// " TASK_FACE_ENEMY 0"
// " "
// " Interrupts"
// )
AI_END_CUSTOM_NPC()
//------------------------------------------------------------------------------
//
// A sensor used to drop bombs only in the correct points
//
//------------------------------------------------------------------------------
LINK_ENTITY_TO_CLASS( npc_helicoptersensor, CBombDropSensor );
BEGIN_DATADESC( CBombDropSensor )
DEFINE_INPUTFUNC( FIELD_VOID, "DropBomb", InputDropBomb ),
DEFINE_INPUTFUNC( FIELD_VOID, "DropBombStraightDown", InputDropBombStraightDown ),
DEFINE_INPUTFUNC( FIELD_STRING, "DropBombAtTargetAlways", InputDropBombAtTargetAlways ),
DEFINE_INPUTFUNC( FIELD_STRING, "DropBombAtTarget", InputDropBombAtTarget ),
DEFINE_INPUTFUNC( FIELD_FLOAT, "DropBombDelay", InputDropBombDelay ),
END_DATADESC()
void CBombDropSensor::Spawn()
{
BaseClass::Spawn();
UTIL_SetSize(this, Vector(-30,-30,-30), Vector(30,30,30) );
SetSolid(SOLID_BBOX);
// Shots pass through
SetCollisionGroup( COLLISION_GROUP_PROJECTILE );
}
// Drop a bomb at a particular location
void CBombDropSensor::InputDropBomb( inputdata_t &inputdata )
{
inputdata_t myVersion = inputdata;
myVersion.pActivator = this;
assert_cast<CNPC_AttackHelicopter*>(GetOwnerEntity())->InputDropBomb( myVersion );
}
void CBombDropSensor::InputDropBombStraightDown( inputdata_t &inputdata )
{
inputdata_t myVersion = inputdata;
myVersion.pActivator = this;
assert_cast<CNPC_AttackHelicopter*>(GetOwnerEntity())->InputDropBombStraightDown( myVersion );
}
void CBombDropSensor::InputDropBombAtTarget( inputdata_t &inputdata )
{
inputdata_t myVersion = inputdata;
myVersion.pActivator = this;
assert_cast<CNPC_AttackHelicopter*>(GetOwnerEntity())->InputDropBombAtTarget( myVersion );
}
void CBombDropSensor::InputDropBombAtTargetAlways( inputdata_t &inputdata )
{
inputdata_t myVersion = inputdata;
myVersion.pActivator = this;
assert_cast<CNPC_AttackHelicopter*>(GetOwnerEntity())->InputDropBombAtTargetAlways( myVersion );
}
void CBombDropSensor::InputDropBombDelay( inputdata_t &inputdata )
{
inputdata_t myVersion = inputdata;
myVersion.pActivator = this;
assert_cast<CNPC_AttackHelicopter*>(GetOwnerEntity())->InputDropBombDelay( myVersion );
}
//------------------------------------------------------------------------------
//
// The bombs the helicopter drops on the player
//
//------------------------------------------------------------------------------
//------------------------------------------------------------------------------
// Save/load
//------------------------------------------------------------------------------
LINK_ENTITY_TO_CLASS( grenade_helicopter, CGrenadeHelicopter );
BEGIN_DATADESC( CGrenadeHelicopter )
DEFINE_FIELD( m_bActivated, FIELD_BOOLEAN ),
DEFINE_FIELD( m_bExplodeOnContact, FIELD_BOOLEAN ),
DEFINE_SOUNDPATCH( m_pWarnSound ),
DEFINE_FIELD( m_hWarningSprite, FIELD_EHANDLE ),
DEFINE_FIELD( m_bBlinkerAtTop, FIELD_BOOLEAN ),
#ifdef HL2_EPISODIC
DEFINE_FIELD( m_flLifetime, FIELD_FLOAT ),
DEFINE_FIELD( m_hCollisionObject, FIELD_EHANDLE ),
DEFINE_FIELD( m_bPickedUp, FIELD_BOOLEAN ),
DEFINE_FIELD( m_flBlinkFastTime, FIELD_TIME ),
DEFINE_INPUTFUNC( FIELD_FLOAT, "ExplodeIn", InputExplodeIn ),
DEFINE_OUTPUT( m_OnPhysGunOnlyPickup, "OnPhysGunOnlyPickup" ),
#endif // HL2_EPISODIC
DEFINE_THINKFUNC( ExplodeThink ),
DEFINE_THINKFUNC( AnimateThink ),
DEFINE_THINKFUNC( RampSoundThink ),
DEFINE_THINKFUNC( WarningBlinkerThink ),
DEFINE_ENTITYFUNC( ExplodeConcussion ),
END_DATADESC()
#define SF_HELICOPTER_GRENADE_DUD (1<<16) // Will not become active on impact, only when launched via physcannon
//------------------------------------------------------------------------------
// Precache
//------------------------------------------------------------------------------
void CGrenadeHelicopter::Precache( void )
{
BaseClass::Precache( );
PrecacheModel( GRENADE_HELICOPTER_MODEL );
PrecacheScriptSound( "ReallyLoudSpark" );
PrecacheScriptSound( "NPC_AttackHelicopterGrenade.Ping" );
PrecacheScriptSound( "NPC_AttackHelicopterGrenade.PingCaptured" );
PrecacheScriptSound( "NPC_AttackHelicopterGrenade.HardImpact" );
}
//------------------------------------------------------------------------------
// Spawn
//------------------------------------------------------------------------------
void CGrenadeHelicopter::Spawn( void )
{
Precache();
// point sized, solid, bouncing
SetCollisionGroup( COLLISION_GROUP_PROJECTILE );
SetModel( GRENADE_HELICOPTER_MODEL );
if ( HasSpawnFlags( SF_HELICOPTER_GRENADE_DUD ) )
{
m_nSkin = (int)SKIN_DUD;
}
if ( !HasSpawnFlags( SF_GRENADE_HELICOPTER_MEGABOMB ) )
{
IPhysicsObject *pPhysicsObject = VPhysicsInitNormal( SOLID_VPHYSICS, GetSolidFlags(), false );
SetMoveType( MOVETYPE_VPHYSICS );
Vector vecAbsVelocity = GetAbsVelocity();
pPhysicsObject->AddVelocity( &vecAbsVelocity, NULL );
}
else
{
SetSolid( SOLID_BBOX );
SetCollisionBounds( Vector( -12.5, -12.5, -12.5 ), Vector( 12.5, 12.5, 12.5 ) );
VPhysicsInitShadow( false, false );
SetMoveType( MOVETYPE_FLYGRAVITY, MOVECOLLIDE_FLY_CUSTOM );
SetElasticity( 0.5f );
AddEffects( EF_NOSHADOW );
}
// We're always being dropped beneath the helicopter; need to not
// be affected by the rotor wash
AddEFlags( EFL_NO_ROTORWASH_PUSH );
// contact grenades arc lower
QAngle angles;
VectorAngles(GetAbsVelocity(), angles );
SetLocalAngles( angles );
SetThink( NULL );
// Tumble in air
QAngle vecAngVel( random->RandomFloat ( -100, -500 ), 0, 0 );
SetLocalAngularVelocity( vecAngVel );
// Explode on contact
SetTouch( &CGrenadeHelicopter::ExplodeConcussion );
// use a lower gravity for grenades to make them easier to see
SetGravity( UTIL_ScaleForGravity( 400 ) );
#ifdef HL2_EPISODIC
m_bPickedUp = false;
m_flLifetime = BOMB_LIFETIME * 2.0;
#endif // HL2_EPISODIC
if ( hl2_episodic.GetBool() )
{
// Disallow this, we'd rather deal with them as physobjects
m_takedamage = DAMAGE_NO;
}
else
{
// Allow player to blow this puppy up in the air
m_takedamage = DAMAGE_YES;
}
m_bActivated = false;
m_pWarnSound = NULL;
m_bExplodeOnContact = false;
m_flDamage = sk_helicopter_grenadedamage.GetFloat();
g_pNotify->AddEntity( this, this );
if( hl2_episodic.GetBool() )
{
SetContextThink( &CGrenadeHelicopter::AnimateThink, gpGlobals->curtime, s_pAnimateThinkContext );
}
}
//------------------------------------------------------------------------------
// On Remve
//------------------------------------------------------------------------------
void CGrenadeHelicopter::UpdateOnRemove()
{
if( m_pWarnSound )
{
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
controller.SoundDestroy( m_pWarnSound );
}
g_pNotify->ClearEntity( this );
BaseClass::UpdateOnRemove();
}
#ifdef HL2_EPISODIC
//------------------------------------------------------------------------------
//------------------------------------------------------------------------------
void CGrenadeHelicopter::InputExplodeIn( inputdata_t &inputdata )
{
m_flLifetime = inputdata.value.Float();
if ( HasSpawnFlags( SF_HELICOPTER_GRENADE_DUD ) )
{
// We are a dud no more!
RemoveSpawnFlags( SF_HELICOPTER_GRENADE_DUD );
m_nSkin = (int)SKIN_REGULAR;
}
m_bActivated = false;
BecomeActive();
}
#endif
//------------------------------------------------------------------------------
// Activate!
//------------------------------------------------------------------------------
void CGrenadeHelicopter::BecomeActive()
{
if ( m_bActivated )
return;
if ( IsMarkedForDeletion() )
return;
m_bActivated = true;
bool bMegaBomb = HasSpawnFlags(SF_GRENADE_HELICOPTER_MEGABOMB);
SetThink( &CGrenadeHelicopter::ExplodeThink );
if ( hl2_episodic.GetBool() )
{
if ( HasSpawnFlags( SF_HELICOPTER_GRENADE_DUD ) == false )
{
SetNextThink( gpGlobals->curtime + GetBombLifetime() );
}
else
{
// NOTE: A dud will not explode after a set time, only when launched!
SetThink( NULL );
return;
}
}
else
{
SetNextThink( gpGlobals->curtime + GetBombLifetime() );
}
if ( !bMegaBomb )
{
SetContextThink( &CGrenadeHelicopter::RampSoundThink, gpGlobals->curtime + GetBombLifetime() - BOMB_RAMP_SOUND_TIME, s_pRampSoundContext );
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
CReliableBroadcastRecipientFilter filter;
m_pWarnSound = controller.SoundCreate( filter, entindex(), "NPC_AttackHelicopterGrenade.Ping" );
controller.Play( m_pWarnSound, 1.0, PITCH_NORM );
}
SetContextThink( &CGrenadeHelicopter::WarningBlinkerThink, gpGlobals->curtime + (GetBombLifetime() - 2.0f), s_pWarningBlinkerContext );
#ifdef HL2_EPISODIC
m_flBlinkFastTime = gpGlobals->curtime + GetBombLifetime() - 1.0f;
#endif//HL2_EPISODIC
}
//------------------------------------------------------------------------------
// Pow!
//------------------------------------------------------------------------------
void CGrenadeHelicopter::RampSoundThink( )
{
if ( m_pWarnSound )
{
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
controller.SoundChangePitch( m_pWarnSound, 140, BOMB_RAMP_SOUND_TIME );
}
SetContextThink( NULL, gpGlobals->curtime, s_pRampSoundContext );
}
//------------------------------------------------------------------------------
//------------------------------------------------------------------------------
void CGrenadeHelicopter::WarningBlinkerThink()
{
#ifndef HL2_EPISODIC
return;
#endif
/*
if( !m_hWarningSprite.Get() )
{
Vector up;
GetVectors( NULL, NULL, &up );
// Light isn't on, so create the sprite.
m_hWarningSprite = CSprite::SpriteCreate( "sprites/redglow1.vmt", GetAbsOrigin() + up * 10.0f, false );
CSprite *pSprite = (CSprite *)m_hWarningSprite.Get();
if( pSprite != NULL )
{
pSprite->SetParent( this, LookupAttachment("top") );
pSprite->SetTransparency( kRenderTransAdd, 255, 255, 255, 255, kRenderFxNone );
pSprite->SetScale( 0.35, 0.0 );
}
m_bBlinkerAtTop = true;
ResetSequence( LookupActivity( "ACT_ARM" ) );
}
else
*/
{
// Just flip it to the other attachment.
if( m_bBlinkerAtTop )
{
//m_hWarningSprite->SetParentAttachment( "SetParentAttachment", "bottom", false );
m_nSkin = (int)SKIN_REGULAR;
m_bBlinkerAtTop = false;
}
else
{
//m_hWarningSprite->SetParentAttachment( "SetParentAttachment", "top", false );
m_nSkin = (int)SKIN_DUD;
m_bBlinkerAtTop = true;
}
}
// Frighten people
CSoundEnt::InsertSound ( SOUND_DANGER, WorldSpaceCenter(), g_helicopter_bomb_danger_radius.GetFloat(), 0.2f, this, SOUNDENT_CHANNEL_REPEATED_DANGER );
#ifdef HL2_EPISODIC
if( gpGlobals->curtime >= m_flBlinkFastTime )
{
SetContextThink( &CGrenadeHelicopter::WarningBlinkerThink, gpGlobals->curtime + 0.1f, s_pWarningBlinkerContext );
}
else
{
SetContextThink( &CGrenadeHelicopter::WarningBlinkerThink, gpGlobals->curtime + 0.2f, s_pWarningBlinkerContext );
}
#endif//HL2_EPISODIC
}
//------------------------------------------------------------------------------
//------------------------------------------------------------------------------
void CGrenadeHelicopter::StopWarningBlinker()
{
if( m_hWarningSprite.Get() )
{
UTIL_Remove( m_hWarningSprite.Get() );
m_hWarningSprite.Set( NULL );
}
}
//------------------------------------------------------------------------------
//------------------------------------------------------------------------------
void CGrenadeHelicopter::AnimateThink()
{
StudioFrameAdvance();
SetContextThink( &CGrenadeHelicopter::AnimateThink, gpGlobals->curtime + 0.1f, s_pAnimateThinkContext );
}
//------------------------------------------------------------------------------
// Entity events... these are events targetted to a particular entity
//------------------------------------------------------------------------------
void CGrenadeHelicopter::OnEntityEvent( EntityEvent_t event, void *pEventData )
{
BaseClass::OnEntityEvent( event, pEventData );
if ( event == ENTITY_EVENT_WATER_TOUCH )
{
BecomeActive();
}
}
//------------------------------------------------------------------------------
// If we hit water, then stop
//------------------------------------------------------------------------------
void CGrenadeHelicopter::PhysicsSimulate( void )
{
Vector vecPrevPosition = GetAbsOrigin();
BaseClass::PhysicsSimulate();
if (!m_bActivated && (GetMoveType() != MOVETYPE_VPHYSICS))
{
if ( GetWaterLevel() > 1 )
{
SetAbsVelocity( vec3_origin );
SetMoveType( MOVETYPE_NONE );
BecomeActive();
}
// Stuck condition, can happen pretty often
if ( vecPrevPosition == GetAbsOrigin() )
{
SetAbsVelocity( vec3_origin );
SetMoveType( MOVETYPE_NONE );
BecomeActive();
}
}
}
//------------------------------------------------------------------------------
// If we hit something, start the timer
//------------------------------------------------------------------------------
void CGrenadeHelicopter::VPhysicsCollision( int index, gamevcollisionevent_t *pEvent )
{
BaseClass::VPhysicsCollision( index, pEvent );
BecomeActive();
#ifndef HL2_EPISODIC // in ep2, don't do this here, do it in Touch()
if ( m_bExplodeOnContact )
{
Vector vecVelocity;
GetVelocity( &vecVelocity, NULL );
DoExplosion( GetAbsOrigin(), vecVelocity );
}
#endif
if( hl2_episodic.GetBool() )
{
float flImpactSpeed = pEvent->preVelocity->Length();
if( flImpactSpeed > 400.0f && pEvent->pEntities[ 1 ]->IsWorld() )
{
EmitSound( "NPC_AttackHelicopterGrenade.HardImpact" );
}
}
}
#if HL2_EPISODIC
//------------------------------------------------------------------------------
// double launch velocity for ep2_outland_08
//------------------------------------------------------------------------------
Vector CGrenadeHelicopter::PhysGunLaunchVelocity( const Vector &forward, float flMass )
{
// return ( striderbuster_shot_velocity.GetFloat() * forward );
return BaseClass::PhysGunLaunchVelocity(forward,flMass) * sk_helicopter_grenaderadius.GetFloat();
}
#endif
//------------------------------------------------------------------------------
//------------------------------------------------------------------------------
float CGrenadeHelicopter::GetBombLifetime()
{
#if HL2_EPISODIC
return m_flLifetime;
#else
return BOMB_LIFETIME;
#endif
}
//------------------------------------------------------------------------------
// Pow!
//------------------------------------------------------------------------------
int CGrenadeHelicopter::OnTakeDamage( const CTakeDamageInfo &info )
{
// We don't take blast damage
if ( info.GetDamageType() & DMG_BLAST )
return 0;
return BaseClass::OnTakeDamage( info );
}
//------------------------------------------------------------------------------
// Pow!
//------------------------------------------------------------------------------
void CGrenadeHelicopter::DoExplosion( const Vector &vecOrigin, const Vector &vecVelocity )
{
ExplosionCreate( GetAbsOrigin(), GetAbsAngles(), GetOwnerEntity() ? GetOwnerEntity() : this, sk_helicopter_grenadedamage.GetFloat(),
sk_helicopter_grenaderadius.GetFloat(), (SF_ENVEXPLOSION_NOSPARKS|SF_ENVEXPLOSION_NODLIGHTS|SF_ENVEXPLOSION_NODECAL|SF_ENVEXPLOSION_NOFIREBALL|SF_ENVEXPLOSION_NOPARTICLES),
sk_helicopter_grenadeforce.GetFloat(), this );
if ( GetShakeAmplitude() )
{
UTIL_ScreenShake( GetAbsOrigin(), GetShakeAmplitude(), 150.0, 1.0, GetShakeRadius(), SHAKE_START );
}
CEffectData data;
// If we're under water do a water explosion
if ( GetWaterLevel() != 0 && (GetWaterType() & CONTENTS_WATER) )
{
data.m_vOrigin = WorldSpaceCenter();
data.m_flMagnitude = 128;
data.m_flScale = 128;
data.m_fFlags = 0;
DispatchEffect( "WaterSurfaceExplosion", data );
}
else
{
// Otherwise do a normal explosion
data.m_vOrigin = GetAbsOrigin();
DispatchEffect( "HelicopterMegaBomb", data );
}
UTIL_Remove( this );
}
//------------------------------------------------------------------------------
// I think I Pow!
//------------------------------------------------------------------------------
void CGrenadeHelicopter::ExplodeThink(void)
{
#ifdef HL2_EPISODIC
// remember if we were thrown by player, we can only determine this prior to explosion
bool bIsThrownByPlayer = IsThrownByPlayer();
int iHealthBefore = 0;
// get the health of the helicopter we came from prior to our explosion
CNPC_AttackHelicopter *pOwner = dynamic_cast<CNPC_AttackHelicopter *>( GetOriginalThrower() );
if ( pOwner )
{
iHealthBefore = pOwner->GetHealth();
}
#endif // HL2_EPISODIC
Vector vecVelocity;
GetVelocity( &vecVelocity, NULL );
DoExplosion( GetAbsOrigin(), vecVelocity );
#ifdef HL2_EPISODIC
// if we were thrown by player, look at health of helicopter after explosion and determine if we damaged it
if ( bIsThrownByPlayer && pOwner && ( iHealthBefore > 0 ) )
{
int iHealthAfter = pOwner->GetHealth();
if ( iHealthAfter == iHealthBefore )
{
// The player threw us, we exploded due to timer, and we did not damage the helicopter that fired us. Send a miss event
SendMissEvent();
}
}
#endif // HL2_EPISODIC
}
//------------------------------------------------------------------------------
// I think I Pow!
//------------------------------------------------------------------------------
void CGrenadeHelicopter::ResolveFlyCollisionCustom( trace_t &trace, Vector &vecVelocity )
{
ResolveFlyCollisionBounce( trace, vecVelocity, 0.1f );
}
//------------------------------------------------------------------------------
// Contact grenade, explode when it touches something
//------------------------------------------------------------------------------
void CGrenadeHelicopter::ExplodeConcussion( CBaseEntity *pOther )
{
if ( !pOther->IsSolid() )
return;
if ( !m_bExplodeOnContact )
{
if ( pOther->IsWorld() )
return;
if ( hl2_episodic.GetBool() )
{
// Don't hit anything other than vehicles
if ( pOther->GetCollisionGroup() != COLLISION_GROUP_VEHICLE )
return;
}
}
#ifdef HL2_EPISODIC
CBaseEntity *pEntityHit = pOther;
if ( pEntityHit->ClassMatches( "phys_bone_follower" ) && pEntityHit->GetOwnerEntity() )
{
pEntityHit = pEntityHit->GetOwnerEntity();
}
if ( ( CLASS_COMBINE_GUNSHIP != pEntityHit->Classify() ) || !pEntityHit->ClassMatches( "npc_helicopter" ) )
{
// We hit something other than a helicopter. If the player threw us, send a miss event
if ( IsThrownByPlayer() )
{
SendMissEvent();
}
}
#endif // HL2_EPISODIC
Vector vecVelocity;
GetVelocity( &vecVelocity, NULL );
DoExplosion( GetAbsOrigin(), vecVelocity );
}
#ifdef HL2_EPISODIC
//-----------------------------------------------------------------------------
// Purpose: The bomb will act differently when picked up by the player
//-----------------------------------------------------------------------------
void CGrenadeHelicopter::OnPhysGunPickup(CBasePlayer *pPhysGunUser, PhysGunPickup_t reason )
{
if ( reason == PICKED_UP_BY_CANNON )
{
if ( !m_bPickedUp )
{
if( m_hWarningSprite.Get() != NULL )
{
UTIL_Remove( m_hWarningSprite );
m_hWarningSprite.Set(NULL);
}
// Turn on
BecomeActive();
// Change the warning sound to a captured sound.
SetContextThink( &CGrenadeHelicopter::RampSoundThink, gpGlobals->curtime + GetBombLifetime() - BOMB_RAMP_SOUND_TIME, s_pRampSoundContext );
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
controller.SoundDestroy( m_pWarnSound );
CReliableBroadcastRecipientFilter filter;
m_pWarnSound = controller.SoundCreate( filter, entindex(), "NPC_AttackHelicopterGrenade.PingCaptured" );
controller.Play( m_pWarnSound, 1.0, PITCH_NORM );
// Reset our counter so the player has more time
SetThink( &CGrenadeHelicopter::ExplodeThink );
SetNextThink( gpGlobals->curtime + GetBombLifetime() );
SetContextThink( &CGrenadeHelicopter::WarningBlinkerThink, gpGlobals->curtime + GetBombLifetime() - 2.0f, s_pWarningBlinkerContext );
#ifdef HL2_EPISODIC
m_nSkin = (int)SKIN_REGULAR;
m_flBlinkFastTime = gpGlobals->curtime + GetBombLifetime() - 1.0f;
#endif//HL2_EPISODIC
// Stop us from sparing damage to the helicopter that dropped us
SetOwnerEntity( pPhysGunUser );
PhysEnableEntityCollisions( this, m_hCollisionObject );
// Don't do this again!
m_bPickedUp = true;
m_OnPhysGunOnlyPickup.FireOutput( pPhysGunUser, this );
}
}
BaseClass::OnPhysGunPickup( pPhysGunUser, reason );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CGrenadeHelicopter::OnPhysGunDrop( CBasePlayer *pPhysGunUser, PhysGunDrop_t reason )
{
if ( reason == LAUNCHED_BY_CANNON )
{
// Enable world touches.
unsigned int flags = VPhysicsGetObject()->GetCallbackFlags();
VPhysicsGetObject()->SetCallbackFlags( flags | CALLBACK_GLOBAL_TOUCH_STATIC );
// Explode on contact
SetTouch( &CGrenadeHelicopter::ExplodeConcussion );
m_bExplodeOnContact = true;
}
BaseClass::OnPhysGunDrop( pPhysGunUser, reason );
}
//-----------------------------------------------------------------------------
// Purpose: Returns if the player threw this grenade w/phys gun
//-----------------------------------------------------------------------------
bool CGrenadeHelicopter::IsThrownByPlayer()
{
// if player is the owner and we're set to explode on contact, then the player threw this grenade.
return ( ( GetOwnerEntity() == UTIL_GetLocalPlayer() ) && m_bExplodeOnContact );
}
//-----------------------------------------------------------------------------
// Purpose: If player threw this grenade, sends a miss event
//-----------------------------------------------------------------------------
void CGrenadeHelicopter::SendMissEvent()
{
// send a miss event
IGameEvent *event = gameeventmanager->CreateEvent( "helicopter_grenade_punt_miss" );
if ( event )
{
gameeventmanager->FireEvent( event );
}
}
#endif // HL2_EPISODIC
//-----------------------------------------------------------------------------
//
// This entity is used to create little force spheres that the helicopters should avoid.
//
//-----------------------------------------------------------------------------
CUtlVector< CAvoidSphere::AvoidSphereHandle_t > CAvoidSphere::s_AvoidSpheres;
#define SF_AVOIDSPHERE_AVOID_BELOW 0x00010000
LINK_ENTITY_TO_CLASS( npc_heli_avoidsphere, CAvoidSphere );
BEGIN_DATADESC( CAvoidSphere )
DEFINE_KEYFIELD( m_flRadius, FIELD_FLOAT, "radius" ),
END_DATADESC()
//-----------------------------------------------------------------------------
// Creates an avoidance sphere
//-----------------------------------------------------------------------------
CBaseEntity *CreateHelicopterAvoidanceSphere( CBaseEntity *pParent, int nAttachment, float flRadius, bool bAvoidBelow )
{
CAvoidSphere *pSphere = static_cast<CAvoidSphere*>(CreateEntityByName( "npc_heli_avoidsphere" ));
pSphere->Init( flRadius );
if ( bAvoidBelow )
{
pSphere->AddSpawnFlags( SF_AVOIDSPHERE_AVOID_BELOW );
}
pSphere->Spawn();
pSphere->SetParent( pParent, nAttachment );
pSphere->SetLocalOrigin( vec3_origin );
pSphere->SetLocalAngles( vec3_angle );
pSphere->SetOwnerEntity( pParent );
return pSphere;
}
//-----------------------------------------------------------------------------
// Init
//-----------------------------------------------------------------------------
void CAvoidSphere::Init( float flRadius )
{
m_flRadius = flRadius;
}
//-----------------------------------------------------------------------------
// Spawn, remove
//-----------------------------------------------------------------------------
void CAvoidSphere::Activate( )
{
BaseClass::Activate();
s_AvoidSpheres.AddToTail( this );
}
void CAvoidSphere::UpdateOnRemove( )
{
s_AvoidSpheres.FindAndRemove( this );
BaseClass::UpdateOnRemove();
}
//-----------------------------------------------------------------------------
// Where are how should we avoid?
//-----------------------------------------------------------------------------
void CAvoidSphere::ComputeAvoidanceForces( CBaseEntity *pEntity, float flEntityRadius,
float flAvoidTime, Vector *pVecAvoidForce )
{
pVecAvoidForce->Init( );
Vector vecEntityDelta;
VectorMultiply( pEntity->GetAbsVelocity(), flAvoidTime, vecEntityDelta );
Vector vecEntityCenter = pEntity->WorldSpaceCenter();
for ( int i = s_AvoidSpheres.Count(); --i >= 0; )
{
CAvoidSphere *pSphere = s_AvoidSpheres[i].Get();
const Vector &vecAvoidCenter = pSphere->WorldSpaceCenter();
// NOTE: This test can be thought of sweeping a sphere through space
// and seeing if it intersects the avoidance sphere
float flTotalRadius = flEntityRadius + pSphere->m_flRadius;
float t1, t2;
if ( !IntersectRayWithSphere( vecEntityCenter, vecEntityDelta,
vecAvoidCenter, flTotalRadius, &t1, &t2 ) )
{
continue;
}
// NOTE: The point of closest approach is at the average t value
Vector vecClosestApproach;
float flAverageT = (t1 + t2) * 0.5f;
VectorMA( vecEntityCenter, flAverageT, vecEntityDelta, vecClosestApproach );
// Add velocity to make it be pushed out away from the sphere center
// without totally counteracting its velocity.
Vector vecDir;
VectorSubtract( vecClosestApproach, vecAvoidCenter, vecDir );
float flZDist = vecDir.z;
float flDist = VectorNormalize( vecDir );
float flDistToTravel;
if ( flDist < 0.01f )
{
flDist = 0.01f;
vecDir.Init( 0, 0, 1 );
flDistToTravel = flTotalRadius;
}
else
{
// make the chopper always avoid *above*
// That means if a force would be applied to push the chopper down,
// figure out a new distance to travel that would push the chopper up.
if ( flZDist < 0.0f && !pSphere->HasSpawnFlags(SF_AVOIDSPHERE_AVOID_BELOW) )
{
Vector vecExitPoint;
vecDir.z = -vecDir.z;
VectorMA( vecAvoidCenter, flTotalRadius, vecDir, vecExitPoint );
VectorSubtract( vecExitPoint, vecClosestApproach, vecDir );
flDistToTravel = VectorNormalize( vecDir );
}
else
{
Assert( flDist <= flTotalRadius );
flDistToTravel = flTotalRadius - flDist;
}
}
// The actual force amount is easy to think about:
// We need to change the position by dx over a time dt, so dv = dx/dt
// But so it doesn't explode, lets clamp t1 to a not-unreasonable time
if ( t1 < 0.25f )
{
t1 = 0.25f;
}
float flForce = 1.25f * flDistToTravel / t1;
vecDir *= flForce;
*pVecAvoidForce += vecDir;
}
}
//-----------------------------------------------------------------------------
//
// This entity is used to create little force boxes that the helicopters should avoid.
//
//-----------------------------------------------------------------------------
CUtlVector< CAvoidBox::AvoidBoxHandle_t > CAvoidBox::s_AvoidBoxes;
#define SF_AVOIDBOX_AVOID_BELOW 0x00010000
LINK_ENTITY_TO_CLASS( npc_heli_avoidbox, CAvoidBox );
BEGIN_DATADESC( CAvoidBox )
END_DATADESC()
//-----------------------------------------------------------------------------
// Spawn, remove
//-----------------------------------------------------------------------------
void CAvoidBox::Spawn( )
{
SetModel( STRING( GetModelName() ) );
SetSolid( SOLID_BSP );
AddSolidFlags( FSOLID_NOT_SOLID );
AddEffects( EF_NODRAW );
}
void CAvoidBox::Activate( )
{
BaseClass::Activate();
s_AvoidBoxes.AddToTail( this );
}
void CAvoidBox::UpdateOnRemove( )
{
s_AvoidBoxes.FindAndRemove( this );
BaseClass::UpdateOnRemove();
}
//-----------------------------------------------------------------------------
// Where are how should we avoid?
//-----------------------------------------------------------------------------
void CAvoidBox::ComputeAvoidanceForces( CBaseEntity *pEntity, float flEntityRadius, float flAvoidTime, Vector *pVecAvoidForce )
{
pVecAvoidForce->Init( );
Vector vecEntityDelta, vecEntityEnd;
VectorMultiply( pEntity->GetAbsVelocity(), flAvoidTime, vecEntityDelta );
Vector vecEntityCenter = pEntity->WorldSpaceCenter();
VectorAdd( vecEntityCenter, vecEntityDelta, vecEntityEnd );
Vector vecVelDir = pEntity->GetAbsVelocity();
VectorNormalize( vecVelDir );
for ( int i = s_AvoidBoxes.Count(); --i >= 0; )
{
CAvoidBox *pBox = s_AvoidBoxes[i].Get();
const Vector &vecAvoidCenter = pBox->WorldSpaceCenter();
// NOTE: This test can be thought of sweeping a sphere through space
// and seeing if it intersects the avoidance box
float flTotalRadius = flEntityRadius + pBox->BoundingRadius();
float t1, t2;
if ( !IntersectInfiniteRayWithSphere( vecEntityCenter, vecEntityDelta,
vecAvoidCenter, flTotalRadius, &t1, &t2 ) )
{
continue;
}
if (( t2 < 0.0f ) || ( t1 > 1.0f ))
continue;
// Unlike the avoid spheres, we also need to make sure the ray intersects the box
Vector vecLocalCenter, vecLocalDelta;
pBox->CollisionProp()->WorldToCollisionSpace( vecEntityCenter, &vecLocalCenter );
pBox->CollisionProp()->WorldDirectionToCollisionSpace( vecEntityDelta, &vecLocalDelta );
Vector vecBoxMin( -flEntityRadius, -flEntityRadius, -flEntityRadius );
Vector vecBoxMax( flEntityRadius, flEntityRadius, flEntityRadius );
vecBoxMin += pBox->CollisionProp()->OBBMins();
vecBoxMax += pBox->CollisionProp()->OBBMaxs();
trace_t tr;
if ( !IntersectRayWithBox( vecLocalCenter, vecLocalDelta, vecBoxMin, vecBoxMax, 0.0f, &tr ) )
continue;
// NOTE: The point of closest approach is at the average t value
Vector vecClosestApproach;
float flAverageT = (t1 + t2) * 0.5f;
VectorMA( vecEntityCenter, flAverageT, vecEntityDelta, vecClosestApproach );
// Add velocity to make it be pushed out away from the sphere center
// without totally counteracting its velocity.
Vector vecDir;
VectorSubtract( vecClosestApproach, vecAvoidCenter, vecDir );
// Limit unnecessary sideways motion
if ( ( tr.plane.type != 3 ) || ( tr.plane.normal[2] > 0.0f ) )
{
vecDir.x *= 0.1f;
vecDir.y *= 0.1f;
}
float flZDist = vecDir.z;
float flDist = VectorNormalize( vecDir );
float flDistToTravel;
if ( flDist < 10.0f )
{
flDist = 10.0f;
vecDir.Init( 0, 0, 1 );
flDistToTravel = flTotalRadius;
}
else
{
// make the chopper always avoid *above*
// That means if a force would be applied to push the chopper down,
// figure out a new distance to travel that would push the chopper up.
if ( flZDist < 0.0f && !pBox->HasSpawnFlags(SF_AVOIDSPHERE_AVOID_BELOW) )
{
Vector vecExitPoint;
vecDir.z = -vecDir.z;
VectorMA( vecAvoidCenter, flTotalRadius, vecDir, vecExitPoint );
VectorSubtract( vecExitPoint, vecClosestApproach, vecDir );
flDistToTravel = VectorNormalize( vecDir );
}
else
{
Assert( flDist <= flTotalRadius );
flDistToTravel = flTotalRadius - flDist;
}
}
// The actual force amount is easy to think about:
// We need to change the position by dx over a time dt, so dv = dx/dt
// But so it doesn't explode, lets clamp t1 to a not-unreasonable time
if ( t1 < 0.25f )
{
t1 = 0.25f;
}
float flForce = 1.5f * flDistToTravel / t1;
vecDir *= flForce;
*pVecAvoidForce += vecDir;
}
}
//-----------------------------------------------------------------------------
//
// This entity is used to create little force boxes that the helicopters should avoid.
//
//-----------------------------------------------------------------------------
CUtlVector< CBombSuppressor::BombSuppressorHandle_t > CBombSuppressor::s_BombSuppressors;
LINK_ENTITY_TO_CLASS( npc_heli_nobomb, CBombSuppressor );
BEGIN_DATADESC( CBombSuppressor )
END_DATADESC()
//-----------------------------------------------------------------------------
// Spawn, remove
//-----------------------------------------------------------------------------
void CBombSuppressor::Spawn( )
{
SetModel( STRING( GetModelName() ) );
SetSolid( SOLID_BSP );
AddSolidFlags( FSOLID_NOT_SOLID );
AddEffects( EF_NODRAW );
}
void CBombSuppressor::Activate( )
{
BaseClass::Activate();
s_BombSuppressors.AddToTail( this );
}
void CBombSuppressor::UpdateOnRemove( )
{
s_BombSuppressors.FindAndRemove( this );
BaseClass::UpdateOnRemove();
}
//-----------------------------------------------------------------------------
// Where are how should we avoid?
//-----------------------------------------------------------------------------
bool CBombSuppressor::CanBomb( const Vector &vecPosition )
{
for ( int i = s_BombSuppressors.Count(); --i >= 0; )
{
CBombSuppressor *pBox = s_BombSuppressors[i].Get();
if ( pBox->CollisionProp()->IsPointInBounds( vecPosition ) )
return false;
}
return true;
}
LINK_ENTITY_TO_CLASS( helicopter_chunk, CHelicopterChunk );
BEGIN_DATADESC( CHelicopterChunk )
DEFINE_THINKFUNC( FallThink ),
DEFINE_FIELD( m_bLanded, FIELD_BOOLEAN ),
DEFINE_FIELD( m_hMaster, FIELD_EHANDLE ),
DEFINE_FIELD( m_nChunkID, FIELD_INTEGER ),
DEFINE_PHYSPTR( m_pTailConstraint ),
DEFINE_PHYSPTR( m_pCockpitConstraint ),
END_DATADESC()
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CHelicopterChunk::Spawn( void )
{
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CHelicopterChunk::FallThink( void )
{
if ( m_bLanded )
{
SetThink( NULL );
return;
}
if ( random->RandomInt( 0, 8 ) == 0 )
{
CEffectData data;
data.m_vOrigin = GetAbsOrigin() + RandomVector( -64, 64 );
DispatchEffect( "HelicopterMegaBomb", data );
EmitSound( "BaseExplosionEffect.Sound" );
}
SetNextThink( gpGlobals->curtime + 0.1f );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : index -
// *pEvent -
//-----------------------------------------------------------------------------
void CHelicopterChunk::VPhysicsCollision( int index, gamevcollisionevent_t *pEvent )
{
BaseClass::VPhysicsCollision( index, pEvent );
if ( m_bLanded == false )
{
int otherIndex = !index;
CBaseEntity *pOther = pEvent->pEntities[otherIndex];
if ( !pOther )
return;
if ( pOther->IsWorld() )
{
CollisionCallback( this );
m_bLanded = true;
SetThink( NULL );
}
}
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *pCaller -
//-----------------------------------------------------------------------------
void CHelicopterChunk::CollisionCallback( CHelicopterChunk *pCaller )
{
if ( m_bLanded )
return;
if ( m_hMaster != NULL )
{
m_hMaster->CollisionCallback( this );
}
else
{
// Break our other constraints
if ( m_pTailConstraint )
{
physenv->DestroyConstraint( m_pTailConstraint );
m_pTailConstraint = NULL;
}
if ( m_pCockpitConstraint )
{
physenv->DestroyConstraint( m_pCockpitConstraint );
m_pCockpitConstraint = NULL;
}
// Add a dust cloud
AR2Explosion *pExplosion = AR2Explosion::CreateAR2Explosion( GetAbsOrigin() );
if ( pExplosion != NULL )
{
pExplosion->SetLifetime( 10 );
}
// Make a loud noise
EmitSound( "NPC_AttackHelicopter.Crash" );
m_bLanded = true;
}
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : &vecPos -
// &vecAngles -
// &vecVelocity -
// *pszModelName -
// Output : CHelicopterChunk
//-----------------------------------------------------------------------------
CHelicopterChunk *CHelicopterChunk::CreateHelicopterChunk( const Vector &vecPos, const QAngle &vecAngles, const Vector &vecVelocity, const char *pszModelName, int chunkID )
{
// Drop a flaming, smoking chunk.
CHelicopterChunk *pChunk = CREATE_ENTITY( CHelicopterChunk, "helicopter_chunk" );
if ( pChunk == NULL )
return NULL;
pChunk->Spawn();
pChunk->SetAbsOrigin( vecPos );
pChunk->SetAbsAngles( vecAngles );
pChunk->SetModel( pszModelName );
pChunk->m_nChunkID = chunkID;
pChunk->SetCollisionGroup( COLLISION_GROUP_INTERACTIVE );
IPhysicsObject *pPhysicsObject = pChunk->VPhysicsInitNormal( SOLID_VPHYSICS, pChunk->GetSolidFlags(), false );
// Set the velocity
if ( pPhysicsObject )
{
pPhysicsObject->EnableMotion( true );
Vector vecChunkVelocity;
AngularImpulse angImpulse;
vecChunkVelocity = vecVelocity;
angImpulse = vec3_origin;
pPhysicsObject->SetVelocity(&vecChunkVelocity, &angImpulse );
}
pChunk->SetThink( &CHelicopterChunk::FallThink );
pChunk->SetNextThink( gpGlobals->curtime + 0.1f );
pChunk->m_bLanded = false;
SmokeTrail *pSmokeTrail = SmokeTrail::CreateSmokeTrail();
pSmokeTrail->FollowEntity( pChunk, "damage" );
pSmokeTrail->m_SpawnRate = 4;
pSmokeTrail->m_ParticleLifetime = 2.0f;
pSmokeTrail->m_StartColor.Init( 0.7f, 0.7f, 0.7f );
pSmokeTrail->m_EndColor.Init( 0.6, 0.6, 0.6 );
pSmokeTrail->m_StartSize = 32;
pSmokeTrail->m_EndSize = 64;
pSmokeTrail->m_SpawnRadius= 8;
pSmokeTrail->m_MinSpeed = 0;
pSmokeTrail->m_MaxSpeed = 8;
pSmokeTrail->m_Opacity = 0.35f;
CFireTrail *pFireTrail = CFireTrail::CreateFireTrail();
if ( pFireTrail == NULL )
return pChunk;
pFireTrail->FollowEntity( pChunk, "damage" );
pFireTrail->SetParent( pChunk, 1 );
pFireTrail->SetLocalOrigin( vec3_origin );
pFireTrail->SetMoveType( MOVETYPE_NONE );
pFireTrail->SetLifetime( 10.0f );
return pChunk;
}