source-engine/game/server/portal/npc_security_camera.cpp

1167 lines
30 KiB
C++
Raw Permalink Normal View History

2022-04-16 09:05:19 +00:00
//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $NoKeywords: $
//=============================================================================//
#include "cbase.h"
#include "ai_basenpc.h"
#include "ai_senses.h"
#include "ai_memory.h"
#include "engine/IEngineSound.h"
#include "ammodef.h"
#include "Sprite.h"
#include "hl2/hl2_player.h"
#include "soundenvelope.h"
#include "explode.h"
#include "IEffects.h"
#include "animation.h"
#include "props.h"
#include "rope.h"
#include "rope_shared.h"
#include "basehlcombatweapon_shared.h"
#include "iservervehicle.h"
#include "physics_prop_ragdoll.h"
#include "portal_util_shared.h"
#include "prop_portal.h"
#include "portal_player.h"
#include "world.h"
#include "ai_baseactor.h" // for Glados ent playing VCDs
#include "sceneentity.h" // precacheing vcds
// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"
#define SECURITY_CAMERA_MODEL "models/props/security_camera.mdl"
#define SECURITY_CAMERA_BC_YAW "aim_yaw"
#define SECURITY_CAMERA_BC_PITCH "aim_pitch"
#define SECURITY_CAMERA_RANGE 1500
#define SECURITY_CAMERA_SPREAD VECTOR_CONE_2DEGREES
#define SECURITY_CAMERA_MAX_WAIT 5
#define SECURITY_CAMERA_PING_TIME 1.0f //LPB!!
#define SECURITY_CAMERA_NUM_ROPES 2
#define SECURITY_CAMERA_GLOW_SPRITE "sprites/glow1.vmt"
//Aiming variables
#define SECURITY_CAMERA_MAX_NOHARM_PERIOD 0.0f
#define SECURITY_CAMERA_MAX_GRACE_PERIOD 3.0f
//Spawnflags
#define SF_SECURITY_CAMERA_AUTOACTIVATE 0x00000020
#define SF_SECURITY_CAMERA_STARTINACTIVE 0x00000040
#define SF_SECURITY_CAMERA_NEVERRETIRE 0x00000080
#define SF_SECURITY_CAMERA_OUT_OF_AMMO 0x00000100
#define CAMERA_DESTROYED_SCENE_1 "scenes/general/generic_security_camera_destroyed-1.vcd"
#define CAMERA_DESTROYED_SCENE_2 "scenes/general/generic_security_camera_destroyed-2.vcd"
#define CAMERA_DESTROYED_SCENE_3 "scenes/general/generic_security_camera_destroyed-3.vcd"
#define CAMERA_DESTROYED_SCENE_4 "scenes/general/generic_security_camera_destroyed-4.vcd"
#define CAMERA_DESTROYED_SCENE_5 "scenes/general/generic_security_camera_destroyed-5.vcd"
//Heights
#define SECURITY_CAMERA_YAW_SPEED 7.0f
#define SECURITY_CAMERA_TOTAL_TO_KNOCK_DOWN 33
//Turret states
enum turretState_e
{
TURRET_SEARCHING,
TURRET_AUTO_SEARCHING,
TURRET_ACTIVE,
TURRET_DEPLOYING,
TURRET_RETIRING,
TURRET_DEAD,
};
// Forces glados actor to play reaction scenes when player dismounts camera.
void PlayDismountSounds( void );
//
// Security Camera
//
class CNPC_SecurityCamera : public CNPCBaseInteractive<CAI_BaseNPC>, public CDefaultPlayerPickupVPhysics
{
DECLARE_CLASS( CNPC_SecurityCamera, CNPCBaseInteractive<CAI_BaseNPC> );
public:
CNPC_SecurityCamera( void );
~CNPC_SecurityCamera( void );
void Precache( void );
virtual void CreateSounds( void );
virtual void StopLoopingSounds( void );
virtual void Spawn( void );
virtual void Activate( void );
bool CreateVPhysics( void );
virtual void UpdateOnRemove( void );
virtual void NotifySystemEvent( CBaseEntity *pNotify, notify_system_event_t eventType, const notify_system_event_params_t &params );
virtual int ObjectCaps( void );
void Use( CBaseEntity *pActivator, CBaseEntity *pCaller, USE_TYPE useType, float value );
// Think functions
void Retire( void );
void Deploy( void );
void ActiveThink( void );
void SearchThink( void );
void DeathThink( void );
// Inputs
void InputToggle( inputdata_t &inputdata );
void InputEnable( inputdata_t &inputdata );
void InputDisable( inputdata_t &inputdata );
void InputRagdoll( inputdata_t &inputdata );
void SetLastSightTime();
int OnTakeDamage( const CTakeDamageInfo &inputInfo );
virtual void PlayerPenetratingVPhysics( void );
bool OnAttemptPhysGunPickup( CBasePlayer *pPhysGunUser, PhysGunPickup_t reason );
bool ShouldSavePhysics() { return true; }
virtual bool CanBeAnEnemyOf( CBaseEntity *pEnemy );
Class_T Classify( void )
{
if( m_bEnabled )
return CLASS_COMBINE;
return CLASS_NONE;
}
bool FVisible( CBaseEntity *pEntity, int traceMask = MASK_BLOCKLOS, CBaseEntity **ppBlocker = NULL );
Vector EyeOffset( Activity nActivity )
{
Vector vForward;
GetVectors( &vForward, 0, 0 );
return vForward * 10.0f;
}
Vector EyePosition( void )
{
return GetAbsOrigin() + EyeOffset(GetActivity());
}
protected:
bool PreThink( turretState_e state );
void Ping( void );
void Toggle( void );
void Enable( void );
void Disable( void );
void RopesOn( void );
void RopesOff( void );
void EyeOn( void );
void EyeOff( void );
bool UpdateFacing( void );
private:
CHandle<CRopeKeyframe> m_hRopes[ SECURITY_CAMERA_NUM_ROPES ];
CHandle<CSprite> m_hEyeGlow;
bool m_bAutoStart;
bool m_bActive; //Denotes the turret is deployed and looking for targets
bool m_bBlinkState;
bool m_bEnabled; //Denotes whether the turret is able to deploy or not
float m_flLastSight;
float m_flPingTime;
QAngle m_vecGoalAngles;
QAngle m_vecCurrentAngles;
Vector m_vNoisePos;
int m_iTicksTillNextNoise;
CSoundPatch *m_pMovementSound;
COutputEvent m_OnDeploy;
COutputEvent m_OnRetire;
DECLARE_DATADESC();
};
//Datatable
BEGIN_DATADESC( CNPC_SecurityCamera )
DEFINE_ARRAY( m_hRopes, FIELD_EHANDLE, SECURITY_CAMERA_NUM_ROPES ),
DEFINE_FIELD( m_hEyeGlow, FIELD_EHANDLE ),
DEFINE_FIELD( m_bAutoStart, FIELD_BOOLEAN ),
DEFINE_FIELD( m_bActive, FIELD_BOOLEAN ),
DEFINE_FIELD( m_bBlinkState, FIELD_BOOLEAN ),
DEFINE_FIELD( m_bEnabled, FIELD_BOOLEAN ),
DEFINE_FIELD( m_flLastSight, FIELD_TIME ),
DEFINE_FIELD( m_flPingTime, FIELD_TIME ),
DEFINE_FIELD( m_vecGoalAngles, FIELD_VECTOR ),
DEFINE_FIELD( m_vecCurrentAngles, FIELD_VECTOR ),
DEFINE_FIELD( m_vNoisePos, FIELD_VECTOR ),
DEFINE_FIELD( m_iTicksTillNextNoise, FIELD_INTEGER ),
DEFINE_SOUNDPATCH( m_pMovementSound ),
DEFINE_THINKFUNC( Retire ),
DEFINE_THINKFUNC( Deploy ),
DEFINE_THINKFUNC( ActiveThink ),
DEFINE_THINKFUNC( SearchThink ),
DEFINE_THINKFUNC( DeathThink ),
// Inputs
DEFINE_INPUTFUNC( FIELD_VOID, "Toggle", InputToggle ),
DEFINE_INPUTFUNC( FIELD_VOID, "Enable", InputEnable ),
DEFINE_INPUTFUNC( FIELD_VOID, "Disable", InputDisable ),
DEFINE_INPUTFUNC( FIELD_VOID, "Ragdoll", InputRagdoll ),
DEFINE_OUTPUT( m_OnDeploy, "OnDeploy" ),
DEFINE_OUTPUT( m_OnRetire, "OnRetire" ),
END_DATADESC()
LINK_ENTITY_TO_CLASS( npc_security_camera, CNPC_SecurityCamera );
//-----------------------------------------------------------------------------
// Constructor
//-----------------------------------------------------------------------------
CNPC_SecurityCamera::CNPC_SecurityCamera( void )
{
m_bActive = false;
m_bAutoStart = false;
m_flPingTime = 0;
m_flLastSight = 0;
m_bBlinkState = false;
m_bEnabled = false;
m_vecCurrentAngles = QAngle( 0.0f, 0.0f, 0.0f );
m_vecGoalAngles.Init();
m_vNoisePos = Vector( 0.0f, 0.0f, 0.0f );
m_iTicksTillNextNoise = 5;
m_pMovementSound = NULL;
m_hEyeGlow = NULL;
}
CNPC_SecurityCamera::~CNPC_SecurityCamera( void )
{
}
//-----------------------------------------------------------------------------
// Purpose: Precache
//-----------------------------------------------------------------------------
void CNPC_SecurityCamera::Precache( void )
{
PrecacheModel( SECURITY_CAMERA_MODEL );
PrecacheScriptSound( "Portalgun.pedestal_rotate_loop" );
// Scenes for when the player dismounts a security camera. Spoken only if Aperture_AI actor is in the
PrecacheInstancedScene( CAMERA_DESTROYED_SCENE_1 );
PrecacheInstancedScene( CAMERA_DESTROYED_SCENE_2 );
PrecacheInstancedScene( CAMERA_DESTROYED_SCENE_3 );
PrecacheInstancedScene( CAMERA_DESTROYED_SCENE_4 );
PrecacheInstancedScene( CAMERA_DESTROYED_SCENE_5 );
BaseClass::Precache();
}
void CNPC_SecurityCamera::CreateSounds()
{
if (!m_pMovementSound)
{
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
CPASAttenuationFilter filter( this );
m_pMovementSound = controller.SoundCreate( filter, entindex(), "Portalgun.pedestal_rotate_loop" );
controller.Play( m_pMovementSound, 0, 100 );
}
}
void CNPC_SecurityCamera::StopLoopingSounds()
{
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
controller.SoundDestroy( m_pMovementSound );
m_pMovementSound = NULL;
BaseClass::StopLoopingSounds();
}
//-----------------------------------------------------------------------------
// Purpose: Spawn the entity
//-----------------------------------------------------------------------------
void CNPC_SecurityCamera::Spawn( void )
{
Precache();
SetModel( SECURITY_CAMERA_MODEL );
BaseClass::Spawn();
m_HackedGunPos = Vector( 0, 0, 12.75 );
SetViewOffset( EyeOffset( ACT_IDLE ) );
m_flFieldOfView = VIEW_FIELD_FULL;
m_takedamage = DAMAGE_NO;
m_iHealth = 1000;
m_bloodColor = BLOOD_COLOR_MECH;
SetSolid( SOLID_BBOX );
AddSolidFlags( FSOLID_NOT_STANDABLE );
SetCollisionBounds( Vector( -16.0f, -16.0f, -16.0f ), Vector( 16.0f, 16.0f, 16.0f ) );
RemoveFlag( FL_AIMTARGET );
AddEFlags( EFL_NO_DISSOLVE );
SetPoseParameter( SECURITY_CAMERA_BC_YAW, 0 );
SetPoseParameter( SECURITY_CAMERA_BC_PITCH, 0 );
//Set our autostart state
m_bAutoStart = !!( m_spawnflags & SF_SECURITY_CAMERA_AUTOACTIVATE );
m_bEnabled = ( ( m_spawnflags & SF_SECURITY_CAMERA_STARTINACTIVE ) == false );
//Do we start active?
if ( m_bAutoStart && m_bEnabled )
{
SetThink( &CNPC_SecurityCamera::SearchThink );
}
//Stagger our starting times
SetNextThink( gpGlobals->curtime + random->RandomFloat( 0.1f, 0.3f ) );
CreateVPhysics();
}
void CNPC_SecurityCamera::Activate( void )
{
BaseClass::Activate();
CreateSounds();
RopesOn();
EyeOn();
}
bool CNPC_SecurityCamera::CreateVPhysics( void )
{
IPhysicsObject *pPhysics = VPhysicsInitNormal( SOLID_VPHYSICS, FSOLID_NOT_STANDABLE, false );
if ( !pPhysics )
DevMsg( "npc_turret_floor unable to spawn physics object!\n" );
else
pPhysics->EnableMotion( false );
return true;
}
void CNPC_SecurityCamera::UpdateOnRemove( void )
{
RopesOff();
EyeOff();
BaseClass::UpdateOnRemove();
}
void CNPC_SecurityCamera::NotifySystemEvent(CBaseEntity *pNotify, notify_system_event_t eventType, const notify_system_event_params_t &params )
{
// On teleport, we record a pointer to the portal we are arriving at
if ( eventType == NOTIFY_EVENT_TELEPORT )
{
RopesOff();
RopesOn();
}
BaseClass::NotifySystemEvent( pNotify, eventType, params );
}
int CNPC_SecurityCamera::ObjectCaps( void )
{
IPhysicsObject *pPhysics = VPhysicsGetObject();
if ( !pPhysics || !pPhysics->IsMotionEnabled() )
return BaseClass::ObjectCaps();
return ( BaseClass::ObjectCaps() | FCAP_USE_IN_RADIUS | FCAP_USE_ONGROUND | FCAP_IMPULSE_USE );
}
void CNPC_SecurityCamera::Use( CBaseEntity *pActivator, CBaseEntity *pCaller, USE_TYPE useType, float value )
{
CBasePlayer *pPlayer = ToBasePlayer( pActivator );
if ( pPlayer )
pPlayer->PickupObject( this );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
int CNPC_SecurityCamera::OnTakeDamage( const CTakeDamageInfo &inputInfo )
{
if ( !m_takedamage )
return 0;
CTakeDamageInfo info = inputInfo;
if ( m_bActive == false )
info.ScaleDamage( 0.1f );
m_iHealth -= info.GetDamage();
if ( m_iHealth <= 0 )
{
m_iHealth = 0;
m_takedamage = DAMAGE_NO;
RemoveFlag( FL_NPC ); // why are they set in the first place???
ExplosionCreate( GetAbsOrigin(), GetLocalAngles(), this, 100, 100, false );
SetThink( &CNPC_SecurityCamera::DeathThink );
StopSound( "NPC_SecurityCamera.Alert" );
m_OnDamaged.FireOutput( info.GetInflictor(), this );
SetNextThink( gpGlobals->curtime + 0.1f );
return 0;
}
return 1;
}
//-----------------------------------------------------------------------------
// Purpose: We override this code because otherwise we start to move into the
// tricky realm of player avoidance. Since we don't go through the
// normal NPC thinking but we ARE an NPC (...) we miss a bunch of
// book keeping. This means we can become invisible and then never
// reappear.
//-----------------------------------------------------------------------------
void CNPC_SecurityCamera::PlayerPenetratingVPhysics( void )
{
// We don't care!
}
bool CNPC_SecurityCamera::OnAttemptPhysGunPickup( CBasePlayer *pPhysGunUser, PhysGunPickup_t reason )
{
return !m_bActive;
}
//-----------------------------------------------------------------------------
// Purpose: Shut down
//-----------------------------------------------------------------------------
void CNPC_SecurityCamera::Retire( void )
{
if ( PreThink( TURRET_RETIRING ) )
return;
//Level out the turret
m_vecGoalAngles = GetAbsAngles();
SetNextThink( gpGlobals->curtime );
//Set ourselves to close
if ( m_bActive )
{
//Notify of the retraction
m_OnRetire.FireOutput( NULL, this );
}
m_bActive = false;
m_flLastSight = 0;
SetThink( &CNPC_SecurityCamera::SUB_DoNothing );
}
//-----------------------------------------------------------------------------
// Purpose: Start up
//-----------------------------------------------------------------------------
void CNPC_SecurityCamera::Deploy( void )
{
if ( PreThink( TURRET_DEPLOYING ) )
return;
m_vecGoalAngles = GetAbsAngles();
SetNextThink( gpGlobals->curtime );
if ( !m_bActive )
{
m_bActive = true;
//Notify we're deploying
m_OnDeploy.FireOutput( NULL, this );
}
m_flPlaybackRate = 0;
SetThink( &CNPC_SecurityCamera::SearchThink );
//EmitSound( "NPC_SecurityCamera.Move" );
SetLastSightTime();
}
//-----------------------------------------------------------------------------
//-----------------------------------------------------------------------------
void CNPC_SecurityCamera::SetLastSightTime()
{
if( HasSpawnFlags( SF_SECURITY_CAMERA_NEVERRETIRE ) )
{
m_flLastSight = FLT_MAX;
}
else
{
m_flLastSight = gpGlobals->curtime + SECURITY_CAMERA_MAX_WAIT;
}
}
//-----------------------------------------------------------------------------
// Purpose: Causes the turret to face its desired angles
//-----------------------------------------------------------------------------
bool CNPC_SecurityCamera::UpdateFacing( void )
{
bool bMoved = false;
if ( m_vecCurrentAngles.x < m_vecGoalAngles.x )
{
m_vecCurrentAngles.x += SECURITY_CAMERA_YAW_SPEED;
if ( m_vecCurrentAngles.x > m_vecGoalAngles.x )
m_vecCurrentAngles.x = m_vecGoalAngles.x;
bMoved = true;
}
if ( m_vecCurrentAngles.y < m_vecGoalAngles.y )
{
m_vecCurrentAngles.y += SECURITY_CAMERA_YAW_SPEED;
if ( m_vecCurrentAngles.y > m_vecGoalAngles.y )
m_vecCurrentAngles.y = m_vecGoalAngles.y;
bMoved = true;
}
if ( m_vecCurrentAngles.x > m_vecGoalAngles.x )
{
m_vecCurrentAngles.x -= SECURITY_CAMERA_YAW_SPEED;
if ( m_vecCurrentAngles.x < m_vecGoalAngles.x )
m_vecCurrentAngles.x = m_vecGoalAngles.x;
bMoved = true;
}
if ( m_vecCurrentAngles.y > m_vecGoalAngles.y )
{
m_vecCurrentAngles.y -= SECURITY_CAMERA_YAW_SPEED;
if ( m_vecCurrentAngles.y < m_vecGoalAngles.y )
m_vecCurrentAngles.y = m_vecGoalAngles.y;
bMoved = true;
}
if ( bMoved )
{
if ( m_pMovementSound )
{
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
controller.SoundChangeVolume( m_pMovementSound, RandomFloat( 0.7f, 0.9f ), 0.05f );
}
// Update pitch
int iPose = LookupPoseParameter( SECURITY_CAMERA_BC_PITCH );
SetPoseParameter( iPose, m_vecCurrentAngles.x );
// Update yaw
iPose = LookupPoseParameter( SECURITY_CAMERA_BC_YAW );
SetPoseParameter( iPose, m_vecCurrentAngles.y );
InvalidateBoneCache();
}
else
{
if ( m_pMovementSound )
{
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
controller.SoundChangeVolume( m_pMovementSound, 0.0f, 0.05f );
}
}
return bMoved;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *pEntity -
// Output : Returns true on success, false on failure.
//-----------------------------------------------------------------------------
bool CNPC_SecurityCamera::FVisible( CBaseEntity *pEntity, int traceMask, CBaseEntity **ppBlocker )
{
CBaseEntity *pHitEntity = NULL;
if ( BaseClass::FVisible( pEntity, traceMask, &pHitEntity ) )
return true;
// If we hit something that's okay to hit anyway, still fire
if ( pHitEntity && pHitEntity->MyCombatCharacterPointer() )
{
if (IRelationType(pHitEntity) == D_HT)
return true;
}
if (ppBlocker)
{
*ppBlocker = pHitEntity;
}
return false;
}
//-----------------------------------------------------------------------------
// Purpose: Allows the turret to fire on targets if they're visible
//-----------------------------------------------------------------------------
void CNPC_SecurityCamera::ActiveThink( void )
{
//Allow descended classes a chance to do something before the think function
if ( PreThink( TURRET_ACTIVE ) )
return;
//Update our think time
SetNextThink( gpGlobals->curtime + 0.1f );
CBaseEntity *pEnemy = GetEnemy();
//If we've become inactive, go back to searching
if ( m_bActive == false || !pEnemy )
{
SetEnemy( NULL );
SetLastSightTime();
SetThink( &CNPC_SecurityCamera::SearchThink );
m_vecGoalAngles = GetAbsAngles();
return;
}
//Get our shot positions
Vector vecMid = EyePosition();
Vector vecMidEnemy = pEnemy->GetAbsOrigin();
//Store off our last seen location
UpdateEnemyMemory( pEnemy, vecMidEnemy );
//Look for our current enemy
bool bEnemyVisible = pEnemy->IsAlive() && FInViewCone( pEnemy ) && FVisible( pEnemy );
//Calculate dir and dist to enemy
Vector vecDirToEnemy = vecMidEnemy - vecMid;
float flDistToEnemy = VectorNormalize( vecDirToEnemy );
CProp_Portal *pPortal = NULL;
if ( pEnemy->IsAlive() )
{
pPortal = FInViewConeThroughPortal( pEnemy );
if ( pPortal && FVisibleThroughPortal( pPortal, pEnemy ) )
{
// Translate our target across the portal
Vector vecMidEnemyTransformed;
UTIL_Portal_PointTransform( pPortal->m_hLinkedPortal->MatrixThisToLinked(), vecMidEnemy, vecMidEnemyTransformed );
//Calculate dir and dist to enemy
Vector vecDirToEnemyTransformed = vecMidEnemyTransformed - vecMid;
float flDistToEnemyTransformed = VectorNormalize( vecDirToEnemyTransformed );
// If it's not visible through normal means or the enemy is closer through the portal, use the translated info
if ( !bEnemyVisible || flDistToEnemyTransformed < flDistToEnemy )
{
bEnemyVisible = true;
vecMidEnemy = vecMidEnemyTransformed;
vecDirToEnemy = vecDirToEnemyTransformed;
flDistToEnemy = flDistToEnemyTransformed;
}
else
{
pPortal = NULL;
}
}
else
{
pPortal = NULL;
}
}
// Add noise to the look position
--m_iTicksTillNextNoise;
if ( m_iTicksTillNextNoise <= 0 && flDistToEnemy < 256.0f )
{
m_vNoisePos.x = RandomFloat( -8.0f, 8.0f );
m_vNoisePos.y = RandomFloat( -8.0f, 8.0f );
m_vNoisePos.z = RandomFloat( 0.0f, 32.0f );
m_iTicksTillNextNoise = RandomInt( 5, 30 );
}
//We want to look at the enemy's eyes so we don't jitter
Vector vEnemyEyes = pEnemy->EyePosition();
if ( pPortal )
{
UTIL_Portal_PointTransform( pPortal->m_hLinkedPortal->MatrixThisToLinked(), vEnemyEyes, vEnemyEyes );
}
Vector vecDirToEnemyEyes = ( vEnemyEyes + m_vNoisePos ) - vecMid;
VectorNormalize( vecDirToEnemyEyes );
QAngle vecAnglesToEnemy;
VectorAngles( vecDirToEnemyEyes, vecAnglesToEnemy );
Vector vForward, vRight, vUp;
GetVectors( &vForward, &vRight, &vUp );
vecAnglesToEnemy.x = acosf( vecDirToEnemyEyes.Dot( -vUp ) ) * ( 180.0f / M_PI );
Vector vProjectedDirToEnemyEyes = vecDirToEnemyEyes - vecDirToEnemyEyes.Dot( vUp ) * vUp;
VectorNormalize( vProjectedDirToEnemyEyes );
if ( vProjectedDirToEnemyEyes.IsZero() )
vecAnglesToEnemy.y = m_vecGoalAngles.y;
else
{
if ( vProjectedDirToEnemyEyes.Dot( vForward ) > 0.0f )
vecAnglesToEnemy.y = acosf( vProjectedDirToEnemyEyes.Dot( vRight ) ) * ( 180.0f / M_PI ) - 90.0f;
else
vecAnglesToEnemy.y = -acosf( vProjectedDirToEnemyEyes.Dot( vRight ) ) * ( 180.0f / M_PI ) - 90.0f;
}
vecAnglesToEnemy.y = AngleNormalize( vecAnglesToEnemy.y );
//Current enemy is not visible
if ( ( bEnemyVisible == false ) || ( flDistToEnemy > SECURITY_CAMERA_RANGE ) )
{
if ( gpGlobals->curtime > m_flLastSight )
{
// Should we look for a new target?
ClearEnemyMemory();
SetEnemy( NULL );
SetLastSightTime();
SetThink( &CNPC_SecurityCamera::SearchThink );
m_vecGoalAngles = GetAbsAngles();
return;
}
bEnemyVisible = false;
}
//If we can see our enemy, face it
if ( bEnemyVisible )
{
m_vecGoalAngles.y = vecAnglesToEnemy.y;
m_vecGoalAngles.x = vecAnglesToEnemy.x;
m_flLastSight = gpGlobals->curtime + 0.5f;
}
//Turn to face
UpdateFacing();
// Update rope positions
for ( int iRope = 0; iRope < SECURITY_CAMERA_NUM_ROPES; ++iRope )
{
if ( m_hRopes[ iRope ] )
{
m_hRopes[ iRope ]->EndpointsChanged();
}
}
}
//-----------------------------------------------------------------------------
// Purpose: Target doesn't exist or has eluded us, so search for one
//-----------------------------------------------------------------------------
void CNPC_SecurityCamera::SearchThink( void )
{
//Allow descended classes a chance to do something before the think function
if ( PreThink( TURRET_SEARCHING ) )
return;
SetNextThink( gpGlobals->curtime + 0.1f );
//If our enemy has died, pick a new enemy
if ( ( GetEnemy() != NULL ) && ( GetEnemy()->IsAlive() == false ) )
{
SetEnemy( NULL );
}
//Acquire the target
if ( GetEnemy() == NULL )
{
CBaseEntity *pEnemy = NULL;
//CBasePlayer *pPlayer = UTIL_PlayerByIndex( 1 );
for( int i = 1; i <= gpGlobals->maxClients; ++i )
{
CBasePlayer *pPlayer = UTIL_PlayerByIndex( i );
if ( pPlayer && pPlayer->IsAlive() )
{
if ( FInViewCone( pPlayer ) && FVisible( pPlayer ) )
{
pEnemy = pPlayer;
break;
}
else
{
CProp_Portal *pPortal = FInViewConeThroughPortal( pPlayer );
if ( pPortal && FVisibleThroughPortal( pPortal, pPlayer ) )
{
pEnemy = pPlayer;
break;
}
}
}
}
if ( pEnemy )
{
SetEnemy( pEnemy );
}
}
//If we've found a target follow it
if ( GetEnemy() != NULL )
{
m_flLastSight = 0;
m_bActive = true;
SetThink( &CNPC_SecurityCamera::ActiveThink );
//EmitSound( "NPC_CeilingTurret.Active" );
return;
}
--m_iTicksTillNextNoise;
if ( m_iTicksTillNextNoise <= 0 )
{
//Display that we're scanning
m_vecGoalAngles.x = RandomFloat( -10.0f, 30.0f );
m_vecGoalAngles.y = RandomFloat( -80.0f, 80.0f );
m_iTicksTillNextNoise = RandomInt( 10, 35 );
}
//Turn and ping
//UpdateFacing();
Ping();
}
//-----------------------------------------------------------------------------
// Purpose: Allows a generic think function before the others are called
// Input : state - which state the turret is currently in
//-----------------------------------------------------------------------------
bool CNPC_SecurityCamera::PreThink( turretState_e state )
{
CheckPVSCondition();
//Animate
StudioFrameAdvance();
//Do not interrupt current think function
return false;
}
//-----------------------------------------------------------------------------
// Purpose: Make a pinging noise so the player knows where we are
//-----------------------------------------------------------------------------
void CNPC_SecurityCamera::Ping( void )
{
//See if it's time to ping again
if ( m_flPingTime > gpGlobals->curtime )
return;
//Ping!
//EmitSound( "NPC_CeilingTurret.Ping" );
m_flPingTime = gpGlobals->curtime + SECURITY_CAMERA_PING_TIME;
}
//-----------------------------------------------------------------------------
// Purpose: Toggle the turret's state
//-----------------------------------------------------------------------------
void CNPC_SecurityCamera::Toggle( void )
{
//Toggle the state
if ( m_bEnabled )
{
Disable();
}
else
{
Enable();
}
}
//-----------------------------------------------------------------------------
// Purpose: Enable the turret and deploy
//-----------------------------------------------------------------------------
void CNPC_SecurityCamera::Enable( void )
{
m_bEnabled = true;
// if the turret is flagged as an autoactivate turret, re-enable its ability open self.
if ( m_spawnflags & SF_SECURITY_CAMERA_AUTOACTIVATE )
{
m_bAutoStart = true;
}
SetThink( &CNPC_SecurityCamera::Deploy );
SetNextThink( gpGlobals->curtime + 0.05f );
}
//-----------------------------------------------------------------------------
// Purpose: Retire the turret until enabled again
//-----------------------------------------------------------------------------
void CNPC_SecurityCamera::Disable( void )
{
m_bEnabled = false;
m_bAutoStart = false;
SetEnemy( NULL );
SetThink( &CNPC_SecurityCamera::Retire );
SetNextThink( gpGlobals->curtime + 0.1f );
}
void CNPC_SecurityCamera::RopesOn( void )
{
for ( int iRope = 0; iRope < SECURITY_CAMERA_NUM_ROPES; ++iRope )
{
// Make a rope if it doesn't exist
if ( !m_hRopes[ iRope ] )
{
CFmtStr str;
int iStartIndex = LookupAttachment( str.sprintf( "Wire%i_A", iRope + 1 ) );
int iEndIndex = LookupAttachment( str.sprintf( "Wire%i_B", iRope + 1 ) );
m_hRopes[ iRope ] = CRopeKeyframe::Create( this, this, iStartIndex, iEndIndex );
if ( m_hRopes[ iRope ] )
{
m_hRopes[ iRope ]->m_Width = 0.7;
m_hRopes[ iRope ]->m_nSegments = ROPE_MAX_SEGMENTS;
m_hRopes[ iRope ]->EnableWind( false );
m_hRopes[ iRope ]->SetupHangDistance( 9.0f );
m_hRopes[ iRope ]->m_bConstrainBetweenEndpoints = true;
}
}
}
}
void CNPC_SecurityCamera::RopesOff( void )
{
for ( int iRope = 0; iRope < SECURITY_CAMERA_NUM_ROPES; ++iRope )
{
// Remove rope if it's alive
if ( m_hRopes[ iRope ] )
{
UTIL_Remove( m_hRopes[ iRope ] );
m_hRopes[ iRope ] = NULL;
}
}
}
void CNPC_SecurityCamera::EyeOn( void )
{
if ( !m_hEyeGlow )
{
// Create our eye sprite
m_hEyeGlow = CSprite::SpriteCreate( SECURITY_CAMERA_GLOW_SPRITE, GetLocalOrigin(), false );
if ( !m_hEyeGlow )
return;
m_hEyeGlow->SetTransparency( kRenderWorldGlow, 255, 0, 0, 128, kRenderFxNoDissipation );
m_hEyeGlow->SetAttachment( this, LookupAttachment( "light" ) );
m_hEyeGlow->SetScale( 0.3f, 1.0f );
}
}
void CNPC_SecurityCamera::EyeOff( void )
{
if ( m_hEyeGlow != NULL )
{
UTIL_Remove( m_hEyeGlow );
m_hEyeGlow = NULL;
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CNPC_SecurityCamera::InputToggle( inputdata_t &inputdata )
{
Toggle();
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CNPC_SecurityCamera::InputEnable( inputdata_t &inputdata )
{
Enable();
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CNPC_SecurityCamera::InputDisable( inputdata_t &inputdata )
{
Disable();
}
void CNPC_SecurityCamera::InputRagdoll( inputdata_t &inputdata )
{
if ( !m_bEnabled )
return;
// Leave decal on wall (may want to disable this once decal for where cam touches wall is made)
Vector vForward;
GetVectors( &vForward, NULL, NULL );
trace_t tr;
UTIL_TraceLine ( GetAbsOrigin() + 10.0f * vForward, GetAbsOrigin() -60.0f * vForward, MASK_SHOT, this, COLLISION_GROUP_NONE, &tr );
if ( tr.m_pEnt )
UTIL_DecalTrace( &tr, "SecurityCamera.Detachment" );
// Disable it's AI
Disable();
SetThink( &CNPC_SecurityCamera::DeathThink );
EyeOff();
// Make it move
IPhysicsObject *pPhysics = VPhysicsGetObject();
if ( !pPhysics || pPhysics->IsMotionEnabled() )
return;
pPhysics->EnableMotion( true );
pPhysics->Wake();
PlayDismountSounds();
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CNPC_SecurityCamera::DeathThink( void )
{
if ( PreThink( TURRET_DEAD ) )
return;
// Level out our angles
m_vecGoalAngles.x = 120.0f;
m_vecGoalAngles.y = 0.0f;
SetNextThink( gpGlobals->curtime + 0.1f );
if ( m_lifeState != LIFE_DEAD )
{
m_lifeState = LIFE_DEAD;
//EmitSound( "NPC_CeilingTurret.Die" );
}
// lots of smoke
Vector pos;
CollisionProp()->RandomPointInBounds( vec3_origin, Vector( 1, 1, 1 ), &pos );
CBroadcastRecipientFilter filter;
te->Smoke( filter, 0.0, &pos, g_sModelIndexSmoke, 2.5, 10 );
g_pEffects->Sparks( pos );
if ( !UpdateFacing() )
{
m_flPlaybackRate = 0;
SetThink( NULL );
}
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *pEnemy -
// Output : Returns true on success, false on failure.
//-----------------------------------------------------------------------------
bool CNPC_SecurityCamera::CanBeAnEnemyOf( CBaseEntity *pEnemy )
{
// If we're out of ammo, make friendly companions ignore us
if ( m_spawnflags & SF_SECURITY_CAMERA_OUT_OF_AMMO )
{
if ( pEnemy->Classify() == CLASS_PLAYER_ALLY_VITAL )
return false;
}
return BaseClass::CanBeAnEnemyOf( pEnemy );
}
void PlayDismountSounds()
{
// Play GLaDOS's audio reaction
CPortal_Player* pPlayer = ToPortalPlayer( UTIL_PlayerByIndex( 1 ) );
CAI_BaseActor* pGlaDOS = (CAI_BaseActor*)gEntList.FindEntityByName( NULL, "Aperture_AI" );
if ( !pPlayer || !pGlaDOS )
{
DevMsg( 2, "Could not play CNPC_SecurityCamera dismount scene, make sure actor named 'Aperture_AI' is present in map.\n" );
return;
}
IGameEvent *event = gameeventmanager->CreateEvent( "security_camera_detached" );
if ( event )
{
gameeventmanager->FireEvent( event );
}
// If glados is currently talking, don't let her talk over herself or interrupt a potentially important speech.
// Should we play the dismount sound after she's done? or is that too disjointed from the camera dismounting act to make sense...
if ( IsRunningScriptedScene( pGlaDOS, false ) )
{
return;
}
pPlayer->IncNumCamerasDetatched();
int iNumCamerasDetatched = pPlayer->GetNumCamerasDetatched();
// If they've knocked down every one possible, play special '1' sound.
if ( iNumCamerasDetatched == SECURITY_CAMERA_TOTAL_TO_KNOCK_DOWN )
{
InstancedScriptedScene( pGlaDOS, CAMERA_DESTROYED_SCENE_1 );
}
else // iNumCamerasDetatched < SECURITY_CAMERA_TOTAL_TO_KNOCK_DOWN
{
// Play different sounds based on progress towards security camera knockdown total.
switch ( iNumCamerasDetatched )
{
case 1:
InstancedScriptedScene( pGlaDOS, CAMERA_DESTROYED_SCENE_2 );
break;
case 2:
InstancedScriptedScene( pGlaDOS, CAMERA_DESTROYED_SCENE_3 );
break;
case 3:
InstancedScriptedScene( pGlaDOS, CAMERA_DESTROYED_SCENE_4 );
break;
default:
InstancedScriptedScene( pGlaDOS, CAMERA_DESTROYED_SCENE_5 );
break;
}
}
}