2013-12-02 19:31:46 -08:00
//========= 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 ;
}