1784 lines
43 KiB
C++
1784 lines
43 KiB
C++
//========= Copyright Valve Corporation, All rights reserved. ============//
|
|
//
|
|
// Purpose: Implements d0g, the loving and caring head crushing Alyx companion.
|
|
//
|
|
//=============================================================================//
|
|
|
|
#include "cbase.h"
|
|
#include "npcevent.h"
|
|
#include "ai_basenpc.h"
|
|
#include "ai_network.h"
|
|
#include "ai_navigator.h"
|
|
#include "ai_motor.h"
|
|
#include "ai_hull.h"
|
|
#include "beam_shared.h"
|
|
#include "ai_baseactor.h"
|
|
#include "npc_rollermine.h"
|
|
#include "saverestore_utlvector.h"
|
|
#include "physics_bone_follower.h"
|
|
#include "Sprite.h"
|
|
#include "ai_behavior_follow.h"
|
|
#include "collisionutils.h"
|
|
|
|
// memdbgon must be the last include file in a .cpp file!!!
|
|
#include "tier0/memdbgon.h"
|
|
|
|
#define EFFECT_COUNT 4
|
|
|
|
extern ConVar ai_debug_avoidancebounds;
|
|
|
|
class CNPC_Dog : public CAI_BaseActor
|
|
{
|
|
public:
|
|
DECLARE_DATADESC();
|
|
DECLARE_CLASS( CNPC_Dog, CAI_BaseActor );
|
|
Class_T Classify ( void );
|
|
void Spawn( void );
|
|
void Precache( void );
|
|
void StartTask( const Task_t *pTask );
|
|
void HandleAnimEvent( animevent_t *pEvent );
|
|
int SelectSchedule( void );
|
|
|
|
bool FindPhysicsObject( const char *pPickupName, CBaseEntity *pIgnore = NULL );
|
|
void RunTask( const Task_t *pTask );
|
|
void CreateBeams( void );
|
|
void ClearBeams( void );
|
|
|
|
void PrescheduleThink( void );
|
|
|
|
bool CanTargetSeeMe( void );
|
|
|
|
Vector FacingPosition( void ) { return WorldSpaceCenter(); }
|
|
float GetHeadDebounce( void ) { return 0.8; } // how much of previous head turn to use
|
|
|
|
void InputSetPickupTarget( inputdata_t &inputdata );
|
|
void InputStartCatchThrowBehavior( inputdata_t &inputdata );
|
|
void InputStopCatchThrowBehavior( inputdata_t &inputdata );
|
|
void InputPlayerPickupObject( inputdata_t &inputdata );
|
|
|
|
void InputStartWaitAndCatch( inputdata_t &inputdata );
|
|
void InputStopWaitAndCatch( inputdata_t &inputdata );
|
|
void InputSetThrowArcModifier( inputdata_t &inputdata );
|
|
void InputSetThrowTarget( inputdata_t &inputdata );
|
|
|
|
void InputTurnBoneFollowersOff( inputdata_t &inputdata );
|
|
void InputTurnBoneFollowersOn( inputdata_t &inputdata );
|
|
|
|
void CleanCatchAndThrow( bool bClearTimers = true );
|
|
void SetTurnActivity ( void );
|
|
void ThrowObject( const char *pAttachmentName );
|
|
void PickupOrCatchObject( const char *pAttachmentName );
|
|
void PullObject( bool bMantain );
|
|
void SetupThrowTarget( void );
|
|
|
|
void GatherConditions( void );
|
|
|
|
Disposition_t IRelationType( CBaseEntity *pTarget );
|
|
|
|
int OnTakeDamage_Alive( const CTakeDamageInfo &info );
|
|
|
|
void MantainBoneFollowerCollisionGroups( int CollisionGroup );
|
|
virtual void SetPlayerAvoidState( void );
|
|
|
|
protected:
|
|
enum
|
|
{
|
|
COND_DOG_LOST_PHYSICS_ENTITY = BaseClass::NEXT_CONDITION,
|
|
|
|
NEXT_CONDITION,
|
|
};
|
|
|
|
protected:
|
|
float m_flNextSwat;
|
|
float m_flTimeToCatch;
|
|
float m_flTimeToPull;
|
|
EHANDLE m_hPhysicsEnt;
|
|
EHANDLE m_hThrowTarget;
|
|
|
|
int m_iPhysGunAttachment;
|
|
bool m_bDoCatchThrowBehavior;
|
|
bool m_bDoWaitforObjectBehavior;
|
|
string_t m_sObjectName;
|
|
|
|
COutputEvent m_OnThrow;
|
|
COutputEvent m_OnCatch;
|
|
COutputEvent m_OnPickup;
|
|
|
|
float m_flThrowArcModifier;
|
|
int m_iContainerMoveType;
|
|
float m_flNextRouteTime;
|
|
|
|
bool m_bHasObject;
|
|
bool m_bBeamEffects;
|
|
|
|
CUtlVector< CHandle <CBaseEntity> > m_hUnreachableObjects;
|
|
|
|
// Contained Bone Follower manager
|
|
CBoneFollowerManager m_BoneFollowerManager;
|
|
|
|
bool CreateVPhysics( void );
|
|
void UpdateOnRemove( void );
|
|
void NPCThink( void );
|
|
void Event_Killed( const CTakeDamageInfo &info );
|
|
|
|
void CreateSprites( void );
|
|
void ClearSprites( void );
|
|
CHandle<CSprite> m_hGlowSprites[EFFECT_COUNT];
|
|
CHandle<CBeam> m_hBeams[EFFECT_COUNT]; //This is temp.
|
|
|
|
virtual bool CreateBehaviors( void );
|
|
CAI_FollowBehavior m_FollowBehavior;
|
|
|
|
bool m_bBoneFollowersActive;
|
|
|
|
|
|
protected:
|
|
|
|
DEFINE_CUSTOM_AI;
|
|
};
|
|
|
|
LINK_ENTITY_TO_CLASS( npc_dog, CNPC_Dog );
|
|
|
|
BEGIN_DATADESC( CNPC_Dog )
|
|
DEFINE_EMBEDDED( m_BoneFollowerManager ),
|
|
// m_FollowBehavior
|
|
DEFINE_FIELD( m_flNextSwat, FIELD_TIME ),
|
|
DEFINE_FIELD( m_flTimeToCatch, FIELD_TIME ),
|
|
DEFINE_FIELD( m_flTimeToPull, FIELD_TIME ),
|
|
DEFINE_FIELD( m_hPhysicsEnt, FIELD_EHANDLE ),
|
|
DEFINE_FIELD( m_hThrowTarget, FIELD_EHANDLE ),
|
|
DEFINE_FIELD( m_iPhysGunAttachment, FIELD_INTEGER ),
|
|
DEFINE_FIELD( m_bDoCatchThrowBehavior, FIELD_BOOLEAN ),
|
|
DEFINE_FIELD( m_bDoWaitforObjectBehavior, FIELD_BOOLEAN ),
|
|
DEFINE_FIELD( m_sObjectName, FIELD_STRING ),
|
|
DEFINE_FIELD( m_flThrowArcModifier, FIELD_FLOAT ),
|
|
DEFINE_FIELD( m_flNextRouteTime, FIELD_TIME ),
|
|
DEFINE_FIELD( m_bHasObject, FIELD_BOOLEAN ),
|
|
DEFINE_FIELD( m_iContainerMoveType, FIELD_INTEGER ),
|
|
DEFINE_FIELD( m_bBeamEffects, FIELD_BOOLEAN ),
|
|
DEFINE_FIELD( m_bBoneFollowersActive, FIELD_BOOLEAN ),
|
|
DEFINE_UTLVECTOR( m_hUnreachableObjects, FIELD_EHANDLE ),
|
|
DEFINE_AUTO_ARRAY( m_hGlowSprites, FIELD_EHANDLE ),
|
|
DEFINE_AUTO_ARRAY( m_hBeams, FIELD_EHANDLE ),
|
|
DEFINE_INPUTFUNC( FIELD_STRING, "SetPickupTarget", InputSetPickupTarget ),
|
|
DEFINE_INPUTFUNC( FIELD_STRING, "StartCatchThrowBehavior", InputStartCatchThrowBehavior ),
|
|
DEFINE_INPUTFUNC( FIELD_STRING, "StopCatchThrowBehavior", InputStopCatchThrowBehavior ),
|
|
DEFINE_INPUTFUNC( FIELD_VOID, "PlayerPickupObject", InputPlayerPickupObject ),
|
|
DEFINE_INPUTFUNC( FIELD_VOID, "StartWaitAndCatch", InputStartWaitAndCatch ),
|
|
DEFINE_INPUTFUNC( FIELD_VOID, "StopWaitAndCatch", InputStopWaitAndCatch ),
|
|
DEFINE_INPUTFUNC( FIELD_FLOAT, "SetThrowArcModifier", InputSetThrowArcModifier ),
|
|
DEFINE_INPUTFUNC( FIELD_STRING, "SetThrowTarget", InputSetThrowTarget ),
|
|
DEFINE_INPUTFUNC( FIELD_VOID, "TurnBoneFollowersOff", InputTurnBoneFollowersOff ),
|
|
DEFINE_INPUTFUNC( FIELD_VOID, "TurnBoneFollowersOn", InputTurnBoneFollowersOn ),
|
|
DEFINE_OUTPUT( m_OnThrow, "OnDogThrow"),
|
|
DEFINE_OUTPUT( m_OnCatch, "OnDogCatch"),
|
|
DEFINE_OUTPUT( m_OnPickup, "OnDogPickup"),
|
|
|
|
END_DATADESC()
|
|
|
|
#define DOG_PHYSOBJ_MOVE_TO_DIST 96
|
|
#define DOG_PULL_DISTANCE 200
|
|
#define DOG_CATCH_DISTANCE 48
|
|
#define DOG_PULL_VELOCITY_MOD 0.1f
|
|
#define DOG_PULL_ANGULARIMP_MOD 0.8f
|
|
#define DOG_PULL_TO_GUN_VEL_MOD 2.0f
|
|
#define DOG_MAX_THROW_MASS 250.0f
|
|
#define DOG_PHYSGUN_ATTACHMENT_NAME "physgun"
|
|
|
|
// These bones have physics shadows
|
|
static const char *pFollowerBoneNames[] =
|
|
{
|
|
// head
|
|
"Dog_Model.Eye",
|
|
"Dog_Model.Pelvis",
|
|
};
|
|
|
|
enum
|
|
{
|
|
SCHED_DOG_FIND_OBJECT = LAST_SHARED_SCHEDULE,
|
|
SCHED_DOG_CATCH_OBJECT,
|
|
SCHED_DOG_WAIT_THROW_OBJECT,
|
|
};
|
|
|
|
//=========================================================
|
|
// tasks
|
|
//=========================================================
|
|
enum
|
|
{
|
|
TASK_DOG_DELAY_SWAT = LAST_SHARED_TASK,
|
|
TASK_DOG_GET_PATH_TO_PHYSOBJ,
|
|
TASK_DOG_PICKUP_ITEM,
|
|
TASK_DOG_LAUNCH_ITEM,
|
|
TASK_DOG_FACE_OBJECT,
|
|
TASK_DOG_WAIT_FOR_OBJECT,
|
|
TASK_DOG_CATCH_OBJECT,
|
|
TASK_DOG_WAIT_FOR_TARGET_TO_FACE,
|
|
TASK_DOG_SETUP_THROW_TARGET,
|
|
};
|
|
|
|
int ACT_DOG_THROW;
|
|
int ACT_DOG_PICKUP;
|
|
int ACT_DOG_WAITING;
|
|
int ACT_DOG_CATCH;
|
|
|
|
int AE_DOG_THROW;
|
|
int AE_DOG_PICKUP;
|
|
int AE_DOG_CATCH;
|
|
int AE_DOG_PICKUP_NOEFFECT;
|
|
|
|
ConVar dog_max_wait_time( "dog_max_wait_time", "7" );
|
|
ConVar dog_debug( "dog_debug", "0" );
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Classify - indicates this NPC's place in the
|
|
// relationship table.
|
|
//-----------------------------------------------------------------------------
|
|
Class_T CNPC_Dog::Classify ( void )
|
|
{
|
|
return CLASS_PLAYER_ALLY_VITAL;
|
|
}
|
|
|
|
bool CNPC_Dog::CreateBehaviors( void )
|
|
{
|
|
AddBehavior( &m_FollowBehavior );
|
|
|
|
return BaseClass::CreateBehaviors();
|
|
}
|
|
|
|
Disposition_t CNPC_Dog::IRelationType( CBaseEntity *pTarget )
|
|
{
|
|
if ( NPC_Rollermine_IsRollermine( pTarget ) )
|
|
{
|
|
if ( pTarget->HasSpawnFlags( SF_ROLLERMINE_FRIENDLY ) )
|
|
return D_LI;
|
|
}
|
|
|
|
return BaseClass::IRelationType( pTarget );
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
//---------------------------------------------------------
|
|
bool CNPC_Dog::CreateVPhysics( void )
|
|
{
|
|
BaseClass::CreateVPhysics();
|
|
|
|
if ( m_bBoneFollowersActive == true && !m_BoneFollowerManager.GetNumBoneFollowers() )
|
|
{
|
|
m_BoneFollowerManager.InitBoneFollowers( this, ARRAYSIZE(pFollowerBoneNames), pFollowerBoneNames );
|
|
}
|
|
return true;
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
//---------------------------------------------------------
|
|
void CNPC_Dog::UpdateOnRemove( void )
|
|
{
|
|
m_BoneFollowerManager.DestroyBoneFollowers();
|
|
BaseClass::UpdateOnRemove();
|
|
}
|
|
|
|
void CNPC_Dog::GatherConditions( void )
|
|
{
|
|
if ( IsInAScript() )
|
|
{
|
|
ClearSenseConditions();
|
|
return;
|
|
}
|
|
|
|
BaseClass::GatherConditions();
|
|
}
|
|
|
|
int CNPC_Dog::OnTakeDamage_Alive( const CTakeDamageInfo &info )
|
|
{
|
|
if ( IsInAScript() )
|
|
return 0;
|
|
|
|
return BaseClass::OnTakeDamage_Alive( info );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// This function checks if Dog's collision group doesn't match his bone follower's and fixes them up.
|
|
//-----------------------------------------------------------------------------
|
|
void CNPC_Dog::MantainBoneFollowerCollisionGroups( int iCollisionGroup )
|
|
{
|
|
if ( m_bBoneFollowersActive == false )
|
|
return;
|
|
|
|
physfollower_t* pBone = m_BoneFollowerManager.GetBoneFollower( 0 );
|
|
|
|
if ( pBone && pBone->hFollower && pBone->hFollower->GetCollisionGroup() != iCollisionGroup )
|
|
{
|
|
for ( int i = 0; i < m_BoneFollowerManager.GetNumBoneFollowers(); i++ )
|
|
{
|
|
pBone = m_BoneFollowerManager.GetBoneFollower( i );
|
|
|
|
if ( pBone && pBone->hFollower )
|
|
{
|
|
pBone->hFollower->SetCollisionGroup( iCollisionGroup );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void CNPC_Dog::SetPlayerAvoidState( void )
|
|
{
|
|
bool bIntersectingBoneFollowers = false;
|
|
bool bIntersectingNPCBox = false;
|
|
|
|
Vector vNothing;
|
|
|
|
GetSequenceLinearMotion( GetSequence(), &vNothing );
|
|
bool bIsMoving = ( IsMoving() || ( vNothing != vec3_origin ) );
|
|
|
|
//If we are coming out of a script, check if we are stuck inside the player.
|
|
if ( m_bPerformAvoidance || ( ShouldPlayerAvoid() && bIsMoving ) )
|
|
{
|
|
trace_t trace;
|
|
Vector vMins, vMaxs;
|
|
Vector vWorldMins, vWorldMaxs;
|
|
Vector vPlayerMins, vPlayerMaxs;
|
|
physfollower_t *pBone;
|
|
int i;
|
|
|
|
CBasePlayer *pLocalPlayer = AI_GetSinglePlayer();
|
|
|
|
if ( pLocalPlayer )
|
|
{
|
|
vWorldMins = WorldAlignMins();
|
|
vWorldMaxs = WorldAlignMaxs();
|
|
|
|
vPlayerMins = pLocalPlayer->GetAbsOrigin() + pLocalPlayer->WorldAlignMins();
|
|
vPlayerMaxs = pLocalPlayer->GetAbsOrigin() + pLocalPlayer->WorldAlignMaxs();
|
|
|
|
// check if the player intersects the bounds of any of the bone followers
|
|
for ( i = 0; i < m_BoneFollowerManager.GetNumBoneFollowers(); i++ )
|
|
{
|
|
pBone = m_BoneFollowerManager.GetBoneFollower( i );
|
|
if ( pBone && pBone->hFollower )
|
|
{
|
|
pBone->hFollower->CollisionProp()->WorldSpaceSurroundingBounds( &vMins, &vMaxs );
|
|
if ( IsBoxIntersectingBox( vMins, vMaxs, vPlayerMins, vPlayerMaxs ) )
|
|
{
|
|
bIntersectingBoneFollowers = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
bIntersectingNPCBox = IsBoxIntersectingBox( GetAbsOrigin() + vWorldMins, GetAbsOrigin() + vWorldMaxs, vPlayerMins, vPlayerMaxs );
|
|
|
|
if ( ai_debug_avoidancebounds.GetBool() )
|
|
{
|
|
int iRed = ( bIntersectingNPCBox == true ) ? 255 : 0;
|
|
|
|
NDebugOverlay::Box( GetAbsOrigin(), vWorldMins, vWorldMaxs, iRed, 0, 255, 64, 0.1 );
|
|
|
|
// draw the bounds of the bone followers
|
|
for ( i = 0; i < m_BoneFollowerManager.GetNumBoneFollowers(); i++ )
|
|
{
|
|
pBone = m_BoneFollowerManager.GetBoneFollower( i );
|
|
if ( pBone && pBone->hFollower )
|
|
{
|
|
pBone->hFollower->CollisionProp()->WorldSpaceSurroundingBounds( &vMins, &vMaxs );
|
|
iRed = ( IsBoxIntersectingBox( vMins, vMaxs, vPlayerMins, vPlayerMaxs ) ) ? 255 : 0;
|
|
|
|
NDebugOverlay::Box( vec3_origin, vMins, vMaxs, iRed, 0, 255, 64, 0.1 );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
m_bPlayerAvoidState = ShouldPlayerAvoid();
|
|
m_bPerformAvoidance = bIntersectingNPCBox || bIntersectingBoneFollowers;
|
|
|
|
if ( GetCollisionGroup() == COLLISION_GROUP_NPC || GetCollisionGroup() == COLLISION_GROUP_NPC_ACTOR )
|
|
{
|
|
if ( bIntersectingNPCBox == true )
|
|
{
|
|
SetCollisionGroup( COLLISION_GROUP_NPC_ACTOR );
|
|
}
|
|
else
|
|
{
|
|
SetCollisionGroup( COLLISION_GROUP_NPC );
|
|
}
|
|
|
|
if ( bIntersectingBoneFollowers == true )
|
|
{
|
|
MantainBoneFollowerCollisionGroups( COLLISION_GROUP_NPC_ACTOR );
|
|
}
|
|
else
|
|
{
|
|
MantainBoneFollowerCollisionGroups( COLLISION_GROUP_NPC );
|
|
}
|
|
}
|
|
}
|
|
//---------------------------------------------------------
|
|
//---------------------------------------------------------
|
|
void CNPC_Dog::NPCThink( void )
|
|
{
|
|
BaseClass::NPCThink();
|
|
|
|
if ( m_hPhysicsEnt == NULL )
|
|
{
|
|
ClearBeams();
|
|
m_bHasObject = false;
|
|
}
|
|
|
|
if ( m_bHasObject == true )
|
|
{
|
|
RelaxAim();
|
|
PullObject( true );
|
|
}
|
|
|
|
|
|
// update follower bones
|
|
m_BoneFollowerManager.UpdateBoneFollowers(this);
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
//---------------------------------------------------------
|
|
void CNPC_Dog::Event_Killed( const CTakeDamageInfo &info )
|
|
{
|
|
m_BoneFollowerManager.DestroyBoneFollowers();
|
|
BaseClass::Event_Killed( info );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Spawn
|
|
//-----------------------------------------------------------------------------
|
|
void CNPC_Dog::Spawn( void )
|
|
{
|
|
m_bBoneFollowersActive = true;
|
|
|
|
Precache();
|
|
|
|
BaseClass::Spawn();
|
|
|
|
SetModel( "models/dog.mdl" );
|
|
|
|
SetHullType( HULL_WIDE_HUMAN );
|
|
SetHullSizeNormal();
|
|
|
|
SetSolid( SOLID_BBOX );
|
|
AddSolidFlags( FSOLID_NOT_STANDABLE );
|
|
SetMoveType( MOVETYPE_STEP );
|
|
SetBloodColor( BLOOD_COLOR_MECH );
|
|
|
|
m_iHealth = 999;
|
|
m_flFieldOfView = 0.5;// indicates the width of this NPC's forward view cone ( as a dotproduct result )
|
|
m_NPCState = NPC_STATE_NONE;
|
|
|
|
m_takedamage = DAMAGE_NO;
|
|
|
|
CapabilitiesAdd( bits_CAP_MOVE_GROUND | bits_CAP_OPEN_DOORS | bits_CAP_TURN_HEAD | bits_CAP_ANIMATEDFACE );
|
|
CapabilitiesAdd( bits_CAP_FRIENDLY_DMG_IMMUNE );
|
|
|
|
NPCInit();
|
|
|
|
m_iPhysGunAttachment = LookupAttachment( DOG_PHYSGUN_ATTACHMENT_NAME );
|
|
|
|
m_bDoCatchThrowBehavior = false;
|
|
m_bDoWaitforObjectBehavior = false;
|
|
m_bHasObject = false;
|
|
m_bBeamEffects = true;
|
|
|
|
m_flThrowArcModifier = 1.0f;
|
|
|
|
m_flNextSwat = gpGlobals->curtime;
|
|
m_flNextRouteTime = gpGlobals->curtime;
|
|
}
|
|
|
|
|
|
void CNPC_Dog::PrescheduleThink( void )
|
|
{
|
|
BaseClass::PrescheduleThink();
|
|
|
|
if ( m_hPhysicsEnt )
|
|
{
|
|
IPhysicsObject *pPhysObj = m_hPhysicsEnt->VPhysicsGetObject();
|
|
|
|
if ( pPhysObj && pPhysObj->GetGameFlags() & FVPHYSICS_PLAYER_HELD )
|
|
{
|
|
m_hPhysicsEnt->SetOwnerEntity( NULL );
|
|
}
|
|
}
|
|
|
|
if ( m_flTimeToCatch < gpGlobals->curtime )
|
|
m_flTimeToCatch = 0.0f;
|
|
|
|
|
|
if ( GetIdealActivity() == ACT_IDLE )
|
|
{
|
|
if ( m_hPhysicsEnt && m_bHasObject == true )
|
|
{
|
|
SetIdealActivity( (Activity)ACT_DOG_WAITING );
|
|
}
|
|
}
|
|
}
|
|
|
|
int CNPC_Dog::SelectSchedule ( void )
|
|
{
|
|
ClearCondition( COND_DOG_LOST_PHYSICS_ENTITY );
|
|
|
|
if ( GetState() == NPC_STATE_SCRIPT || IsInAScript() )
|
|
return BaseClass::SelectSchedule();
|
|
|
|
if ( BehaviorSelectSchedule() )
|
|
return BaseClass::SelectSchedule();
|
|
|
|
if ( m_bDoWaitforObjectBehavior == true )
|
|
{
|
|
if ( m_hPhysicsEnt )
|
|
return SCHED_DOG_CATCH_OBJECT;
|
|
}
|
|
|
|
if ( m_bDoCatchThrowBehavior == true )
|
|
{
|
|
if ( m_flTimeToCatch < 0.1 && m_flNextSwat <= gpGlobals->curtime )
|
|
{
|
|
return SCHED_DOG_FIND_OBJECT;
|
|
}
|
|
|
|
if ( m_flTimeToCatch > gpGlobals->curtime && m_hPhysicsEnt )
|
|
return SCHED_DOG_CATCH_OBJECT;
|
|
}
|
|
else
|
|
{
|
|
if ( m_hPhysicsEnt )
|
|
{
|
|
if ( m_bHasObject == true )
|
|
{
|
|
return SCHED_DOG_WAIT_THROW_OBJECT;
|
|
}
|
|
}
|
|
}
|
|
|
|
return BaseClass::SelectSchedule();
|
|
}
|
|
|
|
void CNPC_Dog::PullObject( bool bMantain )
|
|
{
|
|
if ( m_hPhysicsEnt == NULL )
|
|
{
|
|
TaskFail( "Ack! No Phys Object!");
|
|
return;
|
|
}
|
|
|
|
IPhysicsObject *pPhysObj = m_hPhysicsEnt->VPhysicsGetObject();
|
|
|
|
if ( pPhysObj == NULL )
|
|
{
|
|
TaskFail( "Pulling object with no Phys Object?!" );
|
|
return;
|
|
}
|
|
|
|
if( pPhysObj->GetGameFlags() & FVPHYSICS_PLAYER_HELD )
|
|
{
|
|
m_bHasObject = false;
|
|
ClearBeams();
|
|
TaskFail("Player Grabbed Ball");
|
|
return;
|
|
}
|
|
|
|
CreateBeams();
|
|
|
|
Vector vGunPos;
|
|
GetAttachment( m_iPhysGunAttachment, vGunPos );
|
|
float flDistance = ( vGunPos - m_hPhysicsEnt->WorldSpaceCenter() ).Length();
|
|
|
|
if ( bMantain == false )
|
|
{
|
|
if ( flDistance <= DOG_CATCH_DISTANCE )
|
|
{
|
|
m_hPhysicsEnt->SetOwnerEntity( this );
|
|
|
|
GetNavigator()->StopMoving();
|
|
|
|
//Fire Output!
|
|
m_OnPickup.FireOutput( this, this );
|
|
|
|
m_bHasObject = true;
|
|
ClearBeams();
|
|
TaskComplete();
|
|
return;
|
|
}
|
|
}
|
|
|
|
Vector vDir = ( vGunPos - m_hPhysicsEnt->WorldSpaceCenter() );
|
|
|
|
Vector vCurrentVel;
|
|
float flCurrentVel;
|
|
AngularImpulse vCurrentAI;
|
|
|
|
pPhysObj->GetVelocity( &vCurrentVel, &vCurrentAI );
|
|
flCurrentVel = vCurrentVel.Length();
|
|
|
|
VectorNormalize( vCurrentVel );
|
|
VectorNormalize( vDir );
|
|
|
|
float flVelMod = DOG_PULL_VELOCITY_MOD;
|
|
|
|
if ( bMantain == true )
|
|
flVelMod *= 2;
|
|
|
|
vCurrentVel = vCurrentVel * flCurrentVel * flVelMod;
|
|
|
|
vCurrentAI = vCurrentAI * DOG_PULL_ANGULARIMP_MOD;
|
|
pPhysObj->SetVelocity( &vCurrentVel, &vCurrentAI );
|
|
|
|
vDir = vDir * flDistance * (DOG_PULL_TO_GUN_VEL_MOD * 2);
|
|
|
|
Vector vAngle( 0, 0, 0 );
|
|
pPhysObj->AddVelocity( &vDir, &vAngle );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Precache - precaches all resources this NPC needs
|
|
//-----------------------------------------------------------------------------
|
|
void CNPC_Dog::Precache( void )
|
|
{
|
|
PrecacheModel( "models/dog.mdl" );
|
|
|
|
PrecacheScriptSound( "Weapon_PhysCannon.Launch" );
|
|
|
|
PrecacheModel( "sprites/orangelight1.vmt" );
|
|
PrecacheModel( "sprites/physcannon_bluelight2.vmt" );
|
|
PrecacheModel( "sprites/glow04_noz.vmt" );
|
|
|
|
BaseClass::Precache();
|
|
}
|
|
|
|
void CNPC_Dog::CleanCatchAndThrow( bool bClearTimers )
|
|
{
|
|
if ( m_hPhysicsEnt )
|
|
{
|
|
if ( m_bHasObject == true )
|
|
{
|
|
IPhysicsObject *pPhysObj = m_hPhysicsEnt->VPhysicsGetObject();
|
|
|
|
m_hPhysicsEnt->SetParent( NULL );
|
|
m_hPhysicsEnt->SetOwnerEntity( NULL );
|
|
|
|
Vector vGunPos;
|
|
QAngle angGunAngles;
|
|
GetAttachment( m_iPhysGunAttachment, vGunPos, angGunAngles );
|
|
|
|
if ( pPhysObj )
|
|
{
|
|
pPhysObj->Wake();
|
|
pPhysObj->RemoveShadowController();
|
|
pPhysObj->SetPosition( vGunPos, angGunAngles, true );
|
|
}
|
|
else
|
|
{
|
|
Warning( "CleanCatchAndThrow: m_hPhysicsEnt->VPhysicsGetObject == NULL!\n" );
|
|
}
|
|
|
|
m_hPhysicsEnt->SetMoveType( (MoveType_t)m_iContainerMoveType );
|
|
|
|
if ( pPhysObj )
|
|
{
|
|
pPhysObj->RecheckCollisionFilter();
|
|
}
|
|
|
|
ClearBeams();
|
|
}
|
|
|
|
m_hPhysicsEnt = NULL;
|
|
}
|
|
|
|
if ( bClearTimers == true )
|
|
{
|
|
m_bDoCatchThrowBehavior = false;
|
|
m_bDoWaitforObjectBehavior = false;
|
|
m_flTimeToCatch = 0.0f;
|
|
m_flNextSwat = 0.0f;
|
|
|
|
SetCondition( COND_DOG_LOST_PHYSICS_ENTITY );
|
|
}
|
|
}
|
|
|
|
void CNPC_Dog::InputPlayerPickupObject ( inputdata_t &inputdata )
|
|
{
|
|
if ( m_bDoWaitforObjectBehavior == true )
|
|
{
|
|
if ( m_hPhysicsEnt != inputdata.pCaller )
|
|
{
|
|
if ( m_hPhysicsEnt != NULL )
|
|
CleanCatchAndThrow( false );
|
|
|
|
//Reset this cause CleanCatchAndThrow clears it.
|
|
m_bDoWaitforObjectBehavior = true;
|
|
m_hPhysicsEnt = inputdata.pCaller;
|
|
}
|
|
}
|
|
else if ( m_bDoCatchThrowBehavior == true )
|
|
{
|
|
if ( m_sObjectName != NULL_STRING )
|
|
{
|
|
if ( m_hPhysicsEnt != inputdata.pCaller )
|
|
{
|
|
if ( m_hPhysicsEnt != NULL )
|
|
CleanCatchAndThrow( false );
|
|
|
|
//Reset this cause CleanCatchAndThrow clears it.
|
|
m_bDoCatchThrowBehavior = true;
|
|
m_hPhysicsEnt = inputdata.pCaller;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void CNPC_Dog::InputSetThrowArcModifier( inputdata_t &inputdata )
|
|
{
|
|
m_flThrowArcModifier = inputdata.value.Float();
|
|
}
|
|
|
|
void CNPC_Dog::InputSetPickupTarget( inputdata_t &inputdata )
|
|
{
|
|
CleanCatchAndThrow( false );
|
|
FindPhysicsObject( inputdata.value.String() );
|
|
}
|
|
|
|
void CNPC_Dog::InputStartWaitAndCatch( inputdata_t &inputdata )
|
|
{
|
|
CleanCatchAndThrow();
|
|
m_bDoWaitforObjectBehavior = true;
|
|
}
|
|
|
|
void CNPC_Dog::InputStopWaitAndCatch( inputdata_t &inputdata )
|
|
{
|
|
CleanCatchAndThrow();
|
|
}
|
|
|
|
void CNPC_Dog::InputStartCatchThrowBehavior( inputdata_t &inputdata )
|
|
{
|
|
CleanCatchAndThrow();
|
|
|
|
m_sObjectName = MAKE_STRING( inputdata.value.String() );
|
|
m_bDoCatchThrowBehavior = true;
|
|
|
|
m_flTimeToCatch = 0.0f;
|
|
m_flNextSwat = 0.0f;
|
|
|
|
FindPhysicsObject( inputdata.value.String() );
|
|
}
|
|
|
|
void CNPC_Dog::InputStopCatchThrowBehavior( inputdata_t &inputdata )
|
|
{
|
|
m_bDoCatchThrowBehavior = false;
|
|
|
|
m_flTimeToCatch = 0.0f;
|
|
m_flNextSwat = 0.0f;
|
|
m_sObjectName = NULL_STRING;
|
|
|
|
CleanCatchAndThrow();
|
|
}
|
|
|
|
void CNPC_Dog::InputSetThrowTarget( inputdata_t &inputdata )
|
|
{
|
|
m_hThrowTarget = gEntList.FindEntityByName( NULL, inputdata.value.String(), NULL, inputdata.pActivator, inputdata.pCaller );
|
|
}
|
|
|
|
void CNPC_Dog::SetTurnActivity( void )
|
|
{
|
|
BaseClass::SetTurnActivity();
|
|
|
|
if ( GetIdealActivity() == ACT_IDLE )
|
|
{
|
|
if ( m_hPhysicsEnt && m_bHasObject == true )
|
|
SetIdealActivity( (Activity)ACT_DOG_WAITING );
|
|
}
|
|
}
|
|
|
|
void CNPC_Dog::ThrowObject( const char *pAttachmentName )
|
|
{
|
|
if ( m_hPhysicsEnt )
|
|
{
|
|
m_bHasObject = false;
|
|
|
|
IPhysicsObject *pPhysObj = m_hPhysicsEnt->VPhysicsGetObject();
|
|
|
|
if ( pPhysObj )
|
|
{
|
|
Vector vGunPos;
|
|
QAngle angGunAngles;
|
|
|
|
AngularImpulse angVelocity = RandomAngularImpulse( -250 , -250 ) / pPhysObj->GetMass();
|
|
|
|
InvalidateBoneCache();
|
|
|
|
int iAttachment = LookupAttachment( pAttachmentName );
|
|
|
|
if ( iAttachment == 0 )
|
|
iAttachment = m_iPhysGunAttachment;
|
|
|
|
GetAttachment( iAttachment, vGunPos, angGunAngles );
|
|
|
|
pPhysObj->Wake();
|
|
|
|
if ( pPhysObj->GetShadowController() )
|
|
{
|
|
m_hPhysicsEnt->SetParent( NULL );
|
|
m_hPhysicsEnt->SetMoveType( (MoveType_t)m_iContainerMoveType );
|
|
m_hPhysicsEnt->SetOwnerEntity( this );
|
|
|
|
pPhysObj->RemoveShadowController();
|
|
pPhysObj->SetPosition( m_hPhysicsEnt->GetLocalOrigin(), m_hPhysicsEnt->GetLocalAngles(), true );
|
|
|
|
pPhysObj->RecheckCollisionFilter();
|
|
pPhysObj->RecheckContactPoints();
|
|
}
|
|
|
|
if ( m_hThrowTarget == NULL )
|
|
m_hThrowTarget = AI_GetSinglePlayer();
|
|
|
|
Vector vThrowDirection;
|
|
|
|
if ( m_hThrowTarget )
|
|
{
|
|
Vector vThrowOrigin = m_hThrowTarget->GetAbsOrigin();
|
|
|
|
if ( m_hThrowTarget->IsPlayer() )
|
|
vThrowOrigin = vThrowOrigin + Vector( random->RandomFloat( -128, 128 ), random->RandomFloat( -128, 128 ), 0 );
|
|
|
|
Vector vecToss = VecCheckToss( this, vGunPos, vThrowOrigin, m_flThrowArcModifier, 1.0f, true );
|
|
|
|
if( vecToss == vec3_origin )
|
|
{
|
|
// Fix up an impossible throw so dog will at least toss the box in the target's general direction instead of dropping it.
|
|
// Also toss it up in the air so it will fall down and break. (Just throw the box up at a 45 degree angle)
|
|
Vector forward, up;
|
|
GetVectors( &forward, NULL, &up );
|
|
|
|
vecToss = forward + up;
|
|
VectorNormalize( vecToss );
|
|
|
|
vecToss *= pPhysObj->GetMass() * 30.0f;
|
|
}
|
|
|
|
vThrowDirection = vecToss + ( m_hThrowTarget->GetSmoothedVelocity() / 2 );
|
|
|
|
Vector vLinearDrag;
|
|
|
|
Vector unitVel = vThrowDirection;
|
|
VectorNormalize( unitVel );
|
|
|
|
float flTest = 1000 / vThrowDirection.Length();
|
|
|
|
float flDrag = pPhysObj->CalculateLinearDrag( vThrowDirection );
|
|
vThrowDirection = vThrowDirection + ( unitVel * ( flDrag * flDrag ) ) / flTest;
|
|
|
|
pPhysObj->SetVelocity( &vThrowDirection, &angVelocity );
|
|
|
|
m_flTimeToCatch = gpGlobals->curtime + dog_max_wait_time.GetFloat();
|
|
|
|
//Don't start pulling until the object is away from me.
|
|
//We base the time on the throw velocity.
|
|
m_flTimeToPull = gpGlobals->curtime + ( 1000 / vThrowDirection.Length() );
|
|
}
|
|
|
|
//Fire Output!
|
|
m_OnThrow.FireOutput( this, this );
|
|
|
|
ClearBeams();
|
|
|
|
if ( m_bBeamEffects == true )
|
|
{
|
|
EmitSound( "Weapon_PhysCannon.Launch" );
|
|
|
|
CBeam *pBeam = CBeam::BeamCreate( "sprites/orangelight1.vmt", 1.8 );
|
|
|
|
if ( pBeam != NULL )
|
|
{
|
|
pBeam->PointEntInit( m_hPhysicsEnt->WorldSpaceCenter(), this );
|
|
pBeam->SetEndAttachment( m_iPhysGunAttachment );
|
|
pBeam->SetWidth( 6.4 );
|
|
pBeam->SetEndWidth( 12.8 );
|
|
pBeam->SetBrightness( 255 );
|
|
pBeam->SetColor( 255, 255, 255 );
|
|
pBeam->LiveForTime( 0.2f );
|
|
pBeam->RelinkBeam();
|
|
pBeam->SetNoise( 2 );
|
|
}
|
|
|
|
Vector shotDir = ( m_hPhysicsEnt->WorldSpaceCenter() - vGunPos );
|
|
VectorNormalize( shotDir );
|
|
|
|
CPVSFilter filter( m_hPhysicsEnt->WorldSpaceCenter() );
|
|
te->GaussExplosion( filter, 0.0f, m_hPhysicsEnt->WorldSpaceCenter() - ( shotDir * 4.0f ), RandomVector(-1.0f, 1.0f), 0 );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void CNPC_Dog::PickupOrCatchObject( const char *pAttachmentName )
|
|
{
|
|
if ( m_hPhysicsEnt )
|
|
{
|
|
InvalidateBoneCache();
|
|
|
|
int iAttachment = LookupAttachment( pAttachmentName );
|
|
|
|
if ( iAttachment == 0 )
|
|
iAttachment = m_iPhysGunAttachment;
|
|
|
|
// Move physobject to shadow
|
|
IPhysicsObject *pPhysicsObject = m_hPhysicsEnt->VPhysicsGetObject();
|
|
if ( pPhysicsObject )
|
|
{
|
|
pPhysicsObject->SetShadow( 1e4, 1e4, false, false );
|
|
pPhysicsObject->UpdateShadow( GetAbsOrigin(), GetAbsAngles(), false, 0 );
|
|
}
|
|
|
|
m_iContainerMoveType = m_hPhysicsEnt->GetMoveType();
|
|
m_hPhysicsEnt->SetMoveType( MOVETYPE_NONE );
|
|
|
|
m_hPhysicsEnt->SetParent( this, iAttachment );
|
|
|
|
m_hPhysicsEnt->SetLocalOrigin( vec3_origin );
|
|
m_hPhysicsEnt->SetLocalAngles( vec3_angle );
|
|
|
|
m_hPhysicsEnt->SetGroundEntity( NULL );
|
|
|
|
|
|
if ( m_hPhysicsEnt->GetOwnerEntity() == NULL )
|
|
m_hPhysicsEnt->SetOwnerEntity( this );
|
|
|
|
if ( pPhysicsObject )
|
|
pPhysicsObject->RecheckCollisionFilter();
|
|
|
|
m_bHasObject = true;
|
|
|
|
//Fire Output!
|
|
m_OnPickup.FireOutput( this, this );
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// HandleAnimEvent - catches the NPC-specific messages
|
|
// that occur when tagged animation frames are played.
|
|
//-----------------------------------------------------------------------------
|
|
void CNPC_Dog::HandleAnimEvent( animevent_t *pEvent )
|
|
{
|
|
if ( pEvent->event == AE_DOG_THROW )
|
|
{
|
|
ThrowObject( pEvent->options );
|
|
return;
|
|
}
|
|
|
|
if ( pEvent->event == AE_DOG_PICKUP || pEvent->event == AE_DOG_CATCH || pEvent->event == AE_DOG_PICKUP_NOEFFECT )
|
|
{
|
|
if ( pEvent->event == AE_DOG_PICKUP_NOEFFECT )
|
|
m_bBeamEffects = false;
|
|
else
|
|
m_bBeamEffects = true;
|
|
|
|
PickupOrCatchObject( pEvent->options );
|
|
return;
|
|
}
|
|
|
|
BaseClass::HandleAnimEvent( pEvent );
|
|
}
|
|
|
|
void CNPC_Dog::ClearBeams( void )
|
|
{
|
|
ClearSprites();
|
|
|
|
// Turn off sprites
|
|
for ( int i = 0; i < EFFECT_COUNT; i++ )
|
|
{
|
|
if ( m_hBeams[i] != NULL )
|
|
{
|
|
UTIL_Remove( m_hBeams[i] );
|
|
m_hBeams[i] = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
void CNPC_Dog::ClearSprites( void )
|
|
{
|
|
// Turn off sprites
|
|
for ( int i = 0; i < EFFECT_COUNT; i++ )
|
|
{
|
|
if ( m_hGlowSprites[i] != NULL )
|
|
{
|
|
UTIL_Remove( m_hGlowSprites[i] );
|
|
m_hGlowSprites[i] = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
void CNPC_Dog::CreateSprites( void )
|
|
{
|
|
//Create the glow sprites
|
|
for ( int i = 0; i < EFFECT_COUNT; i++ )
|
|
{
|
|
if ( m_hGlowSprites[i] )
|
|
continue;
|
|
|
|
const char *attachNames[] =
|
|
{
|
|
"physgun",
|
|
"thumb",
|
|
"pinky",
|
|
"index",
|
|
};
|
|
|
|
m_hGlowSprites[i] = CSprite::SpriteCreate( "sprites/glow04_noz.vmt", GetAbsOrigin(), false );
|
|
|
|
m_hGlowSprites[i]->SetAttachment( this, LookupAttachment( attachNames[i] ) );
|
|
m_hGlowSprites[i]->SetTransparency( kRenderGlow, 255, 128, 0, 64, kRenderFxNoDissipation );
|
|
m_hGlowSprites[i]->SetBrightness( 255, 0.2f );
|
|
m_hGlowSprites[i]->SetScale( 0.55f, 0.2f );
|
|
}
|
|
}
|
|
|
|
void CNPC_Dog::CreateBeams( void )
|
|
{
|
|
if ( m_bBeamEffects == false )
|
|
{
|
|
ClearBeams();
|
|
return;
|
|
}
|
|
|
|
CreateSprites();
|
|
|
|
for ( int i = 0; i < EFFECT_COUNT; i++ )
|
|
{
|
|
if ( m_hBeams[i] )
|
|
continue;
|
|
|
|
const char *attachNames[] =
|
|
{
|
|
"physgun",
|
|
"thumb",
|
|
"pinky",
|
|
"index",
|
|
};
|
|
|
|
m_hBeams[i] = CBeam::BeamCreate( "sprites/physcannon_bluelight2.vmt", 5.0 );
|
|
|
|
m_hBeams[i]->EntsInit( m_hPhysicsEnt, this );
|
|
m_hBeams[i]->SetEndAttachment( LookupAttachment( attachNames[i] ) );
|
|
m_hBeams[i]->SetBrightness( 255 );
|
|
m_hBeams[i]->SetColor( 255, 255, 255 );
|
|
m_hBeams[i]->SetNoise( 5.5 );
|
|
m_hBeams[i]->SetRenderMode( kRenderTransAdd );
|
|
}
|
|
|
|
}
|
|
|
|
bool CNPC_Dog::FindPhysicsObject( const char *pPickupName, CBaseEntity *pIgnore )
|
|
{
|
|
CBaseEntity *pEnt = NULL;
|
|
CBaseEntity *pNearest = NULL;
|
|
float flDist;
|
|
IPhysicsObject *pPhysObj = NULL;
|
|
float flNearestDist = 99999;
|
|
|
|
if ( pPickupName != NULL && strlen( pPickupName ) > 0 )
|
|
{
|
|
pEnt = gEntList.FindEntityByName( NULL, pPickupName );
|
|
|
|
if ( m_hUnreachableObjects.Find( pEnt ) == -1 )
|
|
{
|
|
m_bHasObject = false;
|
|
m_hPhysicsEnt = pEnt;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
while ( ( pEnt = gEntList.FindEntityByClassname( pEnt, "prop_physics" ) ) != NULL )
|
|
{
|
|
//We don't want this one.
|
|
if ( pEnt == pIgnore )
|
|
continue;
|
|
|
|
if ( m_hUnreachableObjects.Find( pEnt ) != -1 )
|
|
continue;
|
|
|
|
pPhysObj = pEnt->VPhysicsGetObject();
|
|
|
|
if( pPhysObj == NULL )
|
|
continue;
|
|
|
|
if ( pPhysObj->GetMass() > DOG_MAX_THROW_MASS )
|
|
continue;
|
|
|
|
Vector center = pEnt->WorldSpaceCenter();
|
|
flDist = UTIL_DistApprox2D( GetAbsOrigin(), center );
|
|
|
|
vcollide_t *pCollide = modelinfo->GetVCollide( pEnt->GetModelIndex() );
|
|
|
|
if ( pCollide == NULL )
|
|
continue;
|
|
|
|
if ( pPhysObj->GetGameFlags() & FVPHYSICS_PLAYER_HELD )
|
|
continue;
|
|
|
|
if ( pPhysObj->IsMoveable() == false )
|
|
continue;
|
|
|
|
if ( pEnt->GetCollisionGroup() == COLLISION_GROUP_DEBRIS ||
|
|
pEnt->GetCollisionGroup() == COLLISION_GROUP_INTERACTIVE_DEBRIS )
|
|
continue;
|
|
|
|
if ( center.z > EyePosition().z )
|
|
continue;
|
|
|
|
if ( flDist >= flNearestDist )
|
|
continue;
|
|
|
|
if ( FVisible( pEnt ) == false )
|
|
continue;
|
|
|
|
pNearest = pEnt;
|
|
flNearestDist = flDist;
|
|
}
|
|
|
|
m_bHasObject = false;
|
|
m_hPhysicsEnt = pNearest;
|
|
|
|
if ( dog_debug.GetBool() == true )
|
|
{
|
|
if ( pNearest )
|
|
NDebugOverlay::Box( pNearest->WorldSpaceCenter(), pNearest->CollisionProp()->OBBMins(), pNearest->CollisionProp()->OBBMaxs(), 255, 0, 255, true, 3 );
|
|
}
|
|
|
|
if( m_hPhysicsEnt == NULL )
|
|
{
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Can me enemy see me?
|
|
//-----------------------------------------------------------------------------
|
|
bool CNPC_Dog::CanTargetSeeMe( void )
|
|
{
|
|
CBaseEntity *pEntity = m_hThrowTarget;
|
|
|
|
if ( pEntity )
|
|
{
|
|
if ( pEntity->IsPlayer() == false )
|
|
return true;
|
|
|
|
CBasePlayer *pPlayer = dynamic_cast<CBasePlayer*>( pEntity );
|
|
|
|
if ( pPlayer )
|
|
{
|
|
if ( m_hPhysicsEnt )
|
|
{
|
|
if ( pPlayer->FVisible( m_hPhysicsEnt ) == false )
|
|
return false;
|
|
}
|
|
|
|
if ( pPlayer->FInViewCone( this ) )
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
//---------------------------------------------------------
|
|
//---------------------------------------------------------
|
|
void CNPC_Dog::RunTask( const Task_t *pTask )
|
|
{
|
|
switch( pTask->iTask )
|
|
{
|
|
|
|
case TASK_DOG_PICKUP_ITEM:
|
|
{
|
|
PullObject( false );
|
|
}
|
|
break;
|
|
|
|
case TASK_DOG_GET_PATH_TO_PHYSOBJ:
|
|
{
|
|
//Check this cause our object might have been deleted.
|
|
if ( m_hPhysicsEnt == NULL )
|
|
FindPhysicsObject( NULL );
|
|
|
|
//And if we still can't find anything, then just go away.
|
|
if ( m_hPhysicsEnt == NULL )
|
|
{
|
|
TaskFail( "Can't find an object I like!" );
|
|
return;
|
|
}
|
|
|
|
IPhysicsObject *pPhysicsObject = m_hPhysicsEnt->VPhysicsGetObject();
|
|
|
|
Vector vecGoalPos;
|
|
Vector vecDir;
|
|
|
|
vecDir = GetLocalOrigin() - m_hPhysicsEnt->WorldSpaceCenter();
|
|
VectorNormalize(vecDir);
|
|
vecDir.z = 0;
|
|
|
|
if ( m_hPhysicsEnt->GetOwnerEntity() == NULL )
|
|
m_hPhysicsEnt->SetOwnerEntity( this );
|
|
|
|
if ( pPhysicsObject )
|
|
pPhysicsObject->RecheckCollisionFilter();
|
|
|
|
vecGoalPos = m_hPhysicsEnt->WorldSpaceCenter() + (vecDir * DOG_PHYSOBJ_MOVE_TO_DIST );
|
|
|
|
bool bBuiltRoute = false;
|
|
|
|
//If I'm near my goal, then just walk to it.
|
|
Activity aActivity = ACT_RUN;
|
|
|
|
if ( ( vecGoalPos - GetLocalOrigin() ).Length() <= 128 )
|
|
aActivity = ACT_WALK;
|
|
|
|
bBuiltRoute = GetNavigator()->SetGoal( AI_NavGoal_t( vecGoalPos, aActivity ), AIN_NO_PATH_TASK_FAIL );
|
|
|
|
if ( bBuiltRoute == true )
|
|
TaskComplete();
|
|
else
|
|
{
|
|
m_flTimeToCatch = gpGlobals->curtime + 0.1;
|
|
m_flNextRouteTime = gpGlobals->curtime + 0.3;
|
|
m_flNextSwat = gpGlobals->curtime + 0.1;
|
|
|
|
if ( m_hUnreachableObjects.Find( m_hPhysicsEnt ) == -1 )
|
|
m_hUnreachableObjects.AddToTail( m_hPhysicsEnt );
|
|
|
|
m_hPhysicsEnt = NULL;
|
|
|
|
GetNavigator()->ClearGoal();
|
|
}
|
|
}
|
|
break;
|
|
|
|
case TASK_WAIT:
|
|
{
|
|
if ( IsWaitFinished() )
|
|
{
|
|
TaskComplete();
|
|
}
|
|
|
|
if ( m_hPhysicsEnt )
|
|
{
|
|
if ( m_bHasObject == false )
|
|
{
|
|
GetMotor()->SetIdealYawToTarget( m_hPhysicsEnt->GetAbsOrigin() );
|
|
GetMotor()->UpdateYaw();
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case TASK_DOG_LAUNCH_ITEM:
|
|
if( IsActivityFinished() )
|
|
{
|
|
if ( m_hPhysicsEnt )
|
|
{
|
|
m_hPhysicsEnt->SetOwnerEntity( NULL );
|
|
}
|
|
|
|
TaskComplete();
|
|
}
|
|
break;
|
|
|
|
case TASK_DOG_WAIT_FOR_TARGET_TO_FACE:
|
|
{
|
|
if ( CanTargetSeeMe() )
|
|
TaskComplete();
|
|
}
|
|
break;
|
|
|
|
case TASK_WAIT_FOR_MOVEMENT:
|
|
{
|
|
if ( GetState() == NPC_STATE_SCRIPT || IsInAScript() )
|
|
{
|
|
BaseClass::RunTask( pTask );
|
|
return;
|
|
}
|
|
|
|
if ( m_hPhysicsEnt != NULL )
|
|
{
|
|
IPhysicsObject *pPhysObj = m_hPhysicsEnt->VPhysicsGetObject();
|
|
|
|
if ( !pPhysObj )
|
|
{
|
|
Warning( "npc_dog TASK_WAIT_FOR_MOVEMENT with NULL m_hPhysicsEnt->VPhysicsGetObject\n" );
|
|
}
|
|
|
|
if ( pPhysObj && pPhysObj->GetGameFlags() & FVPHYSICS_PLAYER_HELD )
|
|
TaskFail( "Player picked it up!" );
|
|
|
|
//If the object is moving then my old goal might not be valid
|
|
//cancel the schedule and make it restart again in a bit.
|
|
if ( pPhysObj && pPhysObj->IsAsleep() == false && GetNavigator()->IsGoalActive() == false )
|
|
{
|
|
Vector vecGoalPos;
|
|
Vector vecDir;
|
|
|
|
vecDir = GetLocalOrigin() - m_hPhysicsEnt->WorldSpaceCenter();
|
|
VectorNormalize(vecDir);
|
|
vecDir.z = 0;
|
|
|
|
vecGoalPos = m_hPhysicsEnt->WorldSpaceCenter() + (vecDir * DOG_PHYSOBJ_MOVE_TO_DIST );
|
|
|
|
GetNavigator()->ClearGoal();
|
|
|
|
float flDistance = (vecGoalPos - GetLocalOrigin()).Length();
|
|
|
|
//If I'm near my goal, then just walk to it.
|
|
Activity aActivity = ACT_RUN;
|
|
|
|
if ( ( vecGoalPos - GetLocalOrigin() ).Length() <= 128 )
|
|
aActivity = ACT_WALK;
|
|
|
|
GetNavigator()->SetGoal( AI_NavGoal_t( vecGoalPos, aActivity ), AIN_NO_PATH_TASK_FAIL );
|
|
|
|
if ( flDistance <= DOG_PHYSOBJ_MOVE_TO_DIST )
|
|
{
|
|
TaskComplete();
|
|
GetNavigator()->StopMoving();
|
|
}
|
|
}
|
|
}
|
|
|
|
BaseClass::RunTask( pTask );
|
|
}
|
|
break;
|
|
|
|
case TASK_DOG_WAIT_FOR_OBJECT:
|
|
{
|
|
if ( m_hPhysicsEnt != NULL )
|
|
{
|
|
if ( FVisible( m_hPhysicsEnt ) == false )
|
|
{
|
|
m_flTimeToCatch = 0.0f;
|
|
ClearBeams();
|
|
TaskFail( "Lost sight of the object!" );
|
|
m_hPhysicsEnt->SetOwnerEntity( NULL );
|
|
return;
|
|
}
|
|
|
|
m_hPhysicsEnt->SetOwnerEntity( this );
|
|
|
|
Vector vForward;
|
|
AngleVectors( GetAbsAngles(), &vForward );
|
|
|
|
|
|
Vector vGunPos;
|
|
GetAttachment( m_iPhysGunAttachment, vGunPos );
|
|
|
|
Vector vToObject = m_hPhysicsEnt->WorldSpaceCenter() - vGunPos;
|
|
float flDistance = vToObject.Length();
|
|
|
|
VectorNormalize( vToObject );
|
|
|
|
SetAim( m_hPhysicsEnt->WorldSpaceCenter() - GetAbsOrigin() );
|
|
|
|
CBasePlayer *pPlayer = AI_GetSinglePlayer();
|
|
|
|
float flDistanceToPlayer = flDistance;
|
|
|
|
if ( pPlayer )
|
|
{
|
|
flDistanceToPlayer = (pPlayer->GetAbsOrigin() - m_hPhysicsEnt->WorldSpaceCenter()).Length();
|
|
}
|
|
|
|
IPhysicsObject *pPhysObj = m_hPhysicsEnt->VPhysicsGetObject();
|
|
if ( !pPhysObj )
|
|
{
|
|
Warning( "npc_dog: TASK_DOG_WAIT_FOR_OBJECT with m_hPhysicsEnt->VPhysicsGetObject == NULL\n" );
|
|
}
|
|
|
|
if ( pPhysObj && !( pPhysObj->GetGameFlags() & FVPHYSICS_PLAYER_HELD ) && flDistanceToPlayer > ( flDistance * 2 ) )
|
|
{
|
|
if ( m_flTimeToPull <= gpGlobals->curtime )
|
|
{
|
|
Vector vCurrentVel;
|
|
float flCurrentVel;
|
|
AngularImpulse vCurrentAI;
|
|
|
|
pPhysObj->GetVelocity( &vCurrentVel, &vCurrentAI );
|
|
|
|
flCurrentVel = vCurrentVel.Length();
|
|
VectorNormalize( vCurrentVel );
|
|
|
|
if ( pPhysObj && flDistance <= DOG_PULL_DISTANCE )
|
|
{
|
|
Vector vDir = ( vGunPos - m_hPhysicsEnt->WorldSpaceCenter() );
|
|
|
|
VectorNormalize( vDir );
|
|
|
|
vCurrentVel = vCurrentVel * ( flCurrentVel * DOG_PULL_VELOCITY_MOD );
|
|
|
|
vCurrentAI = vCurrentAI * DOG_PULL_ANGULARIMP_MOD;
|
|
pPhysObj->SetVelocity( &vCurrentVel, &vCurrentAI );
|
|
|
|
vDir = vDir * flDistance * DOG_PULL_TO_GUN_VEL_MOD;
|
|
|
|
Vector vAngle( 0, 0, 0 );
|
|
pPhysObj->AddVelocity( &vDir, &vAngle );
|
|
|
|
CreateBeams();
|
|
}
|
|
|
|
float flDot = DotProduct( vCurrentVel, vForward );
|
|
|
|
if ( flDistance >= DOG_PULL_DISTANCE && flDistance <= ( DOG_PULL_DISTANCE * 2 ) && flDot > -0.3 )
|
|
{
|
|
if ( pPhysObj->IsAsleep() == false && !( pPhysObj->GetGameFlags() & FVPHYSICS_PLAYER_HELD ) )
|
|
{
|
|
Vector vecGoalPos;
|
|
Vector vecDir;
|
|
|
|
vecDir = GetLocalOrigin() - m_hPhysicsEnt->WorldSpaceCenter();
|
|
VectorNormalize(vecDir);
|
|
vecDir.z = 0;
|
|
|
|
vecGoalPos = m_hPhysicsEnt->WorldSpaceCenter() + (vecDir * DOG_PHYSOBJ_MOVE_TO_DIST );
|
|
|
|
GetNavigator()->ClearGoal();
|
|
|
|
//If I'm near my goal, then just walk to it.
|
|
Activity aActivity = ACT_RUN;
|
|
|
|
if ( ( vecGoalPos - GetLocalOrigin() ).Length() <= 128 )
|
|
aActivity = ACT_WALK;
|
|
|
|
GetNavigator()->SetGoal( AI_NavGoal_t( vecGoalPos, aActivity ), AIN_NO_PATH_TASK_FAIL );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
float flDirDot = DotProduct( vToObject, vForward );
|
|
|
|
if ( flDirDot < 0.2 )
|
|
{
|
|
GetMotor()->SetIdealYawToTarget( m_hPhysicsEnt->GetAbsOrigin() );
|
|
GetMotor()->UpdateYaw();
|
|
}
|
|
|
|
if ( m_flTimeToCatch < gpGlobals->curtime && m_bDoWaitforObjectBehavior == false )
|
|
{
|
|
m_hPhysicsEnt->SetOwnerEntity( NULL );
|
|
m_flTimeToCatch = 0.0f;
|
|
ClearBeams();
|
|
TaskFail( "Done waiting!" );
|
|
}
|
|
else if ( pPhysObj && ( flDistance <= DOG_CATCH_DISTANCE && !( pPhysObj->GetGameFlags() & FVPHYSICS_PLAYER_HELD ) ) )
|
|
{
|
|
AngularImpulse vZero( 0, 0, 0 );
|
|
pPhysObj->SetVelocity( &vec3_origin, &vZero );
|
|
|
|
GetNavigator()->StopMoving();
|
|
|
|
//Fire Output!
|
|
m_OnCatch.FireOutput( this, this );
|
|
m_bHasObject = true;
|
|
ClearBeams();
|
|
TaskComplete();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
GetNavigator()->StopMoving();
|
|
|
|
ClearBeams();
|
|
TaskFail("No Physics Object!");
|
|
}
|
|
|
|
}
|
|
break;
|
|
|
|
case TASK_DOG_CATCH_OBJECT:
|
|
if( IsActivityFinished() )
|
|
{
|
|
m_flTimeToCatch = 0.0f;
|
|
TaskComplete();
|
|
}
|
|
break;
|
|
default:
|
|
BaseClass::RunTask( pTask );
|
|
break;
|
|
}
|
|
}
|
|
|
|
void CNPC_Dog::SetupThrowTarget( void )
|
|
{
|
|
if ( m_hThrowTarget == NULL )
|
|
{
|
|
m_hThrowTarget = AI_GetSinglePlayer();
|
|
}
|
|
|
|
SetTarget( m_hThrowTarget );
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
//---------------------------------------------------------
|
|
void CNPC_Dog::StartTask( const Task_t *pTask )
|
|
{
|
|
switch( pTask->iTask )
|
|
{
|
|
|
|
case TASK_DOG_SETUP_THROW_TARGET:
|
|
{
|
|
SetupThrowTarget();
|
|
TaskComplete();
|
|
}
|
|
break;
|
|
case TASK_DOG_GET_PATH_TO_PHYSOBJ:
|
|
{
|
|
FindPhysicsObject( STRING( m_sObjectName ) );
|
|
|
|
if ( m_hPhysicsEnt == NULL )
|
|
{
|
|
FindPhysicsObject( NULL );
|
|
return;
|
|
}
|
|
|
|
IPhysicsObject *pPhysicsObject = m_hPhysicsEnt->VPhysicsGetObject();
|
|
|
|
Vector vecGoalPos;
|
|
Vector vecDir;
|
|
|
|
vecDir = GetLocalOrigin() - m_hPhysicsEnt->WorldSpaceCenter();
|
|
VectorNormalize(vecDir);
|
|
vecDir.z = 0;
|
|
|
|
if ( m_hPhysicsEnt->GetOwnerEntity() == NULL )
|
|
m_hPhysicsEnt->SetOwnerEntity( this );
|
|
|
|
if ( pPhysicsObject )
|
|
pPhysicsObject->RecheckCollisionFilter();
|
|
|
|
vecGoalPos = m_hPhysicsEnt->WorldSpaceCenter() + (vecDir * DOG_PHYSOBJ_MOVE_TO_DIST );
|
|
|
|
//If I'm near my goal, then just walk to it.
|
|
Activity aActivity = ACT_RUN;
|
|
|
|
if ( ( vecGoalPos - GetLocalOrigin() ).Length() <= 128 )
|
|
aActivity = ACT_WALK;
|
|
|
|
if ( GetNavigator()->SetGoal( AI_NavGoal_t( vecGoalPos, aActivity ), AIN_NO_PATH_TASK_FAIL ) == false )
|
|
{
|
|
if ( m_hUnreachableObjects.Find( m_hPhysicsEnt ) == -1 )
|
|
m_hUnreachableObjects.AddToTail( m_hPhysicsEnt );
|
|
|
|
FindPhysicsObject( NULL, m_hPhysicsEnt );
|
|
|
|
m_flTimeToCatch = gpGlobals->curtime + 0.1;
|
|
m_flNextRouteTime = gpGlobals->curtime + 0.3;
|
|
m_flNextSwat = gpGlobals->curtime + 0.1;
|
|
|
|
GetNavigator()->ClearGoal();
|
|
}
|
|
else
|
|
{
|
|
TaskComplete();
|
|
}
|
|
}
|
|
break;
|
|
|
|
case TASK_DOG_FACE_OBJECT:
|
|
{
|
|
if( m_hPhysicsEnt == NULL )
|
|
{
|
|
// Physics Object is gone! Probably was an explosive
|
|
// or something else broke it.
|
|
TaskFail("Physics ent NULL");
|
|
return;
|
|
}
|
|
|
|
Vector vecDir;
|
|
|
|
vecDir = m_hPhysicsEnt->WorldSpaceCenter() - GetLocalOrigin();
|
|
VectorNormalize(vecDir);
|
|
|
|
GetMotor()->SetIdealYaw( UTIL_VecToYaw( vecDir ) );
|
|
TaskComplete();
|
|
}
|
|
break;
|
|
|
|
case TASK_DOG_PICKUP_ITEM:
|
|
{
|
|
if( m_hPhysicsEnt == NULL )
|
|
{
|
|
// Physics Object is gone! Probably was an explosive
|
|
// or something else broke it.
|
|
TaskFail("Physics ent NULL");
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
SetIdealActivity( (Activity)ACT_DOG_PICKUP );
|
|
}
|
|
}
|
|
|
|
break;
|
|
|
|
case TASK_DOG_LAUNCH_ITEM:
|
|
{
|
|
if( m_hPhysicsEnt == NULL )
|
|
{
|
|
// Physics Object is gone! Probably was an explosive
|
|
// or something else broke it.
|
|
TaskFail("Physics ent NULL");
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
if ( m_hPhysicsEnt == NULL || m_bHasObject == false )
|
|
{
|
|
TaskFail( "Don't have the item!" );
|
|
return;
|
|
}
|
|
|
|
SetIdealActivity( (Activity)ACT_DOG_THROW );
|
|
}
|
|
}
|
|
|
|
break;
|
|
|
|
case TASK_DOG_WAIT_FOR_TARGET_TO_FACE:
|
|
{
|
|
if ( CanTargetSeeMe() )
|
|
TaskComplete();
|
|
}
|
|
break;
|
|
|
|
case TASK_DOG_WAIT_FOR_OBJECT:
|
|
{
|
|
SetIdealActivity( (Activity)ACT_DOG_WAITING );
|
|
}
|
|
break;
|
|
|
|
case TASK_DOG_CATCH_OBJECT:
|
|
{
|
|
SetIdealActivity( (Activity)ACT_DOG_CATCH );
|
|
}
|
|
break;
|
|
|
|
case TASK_DOG_DELAY_SWAT:
|
|
m_flNextSwat = gpGlobals->curtime + pTask->flTaskData;
|
|
|
|
if ( m_hThrowTarget == NULL )
|
|
m_hThrowTarget = AI_GetSinglePlayer();
|
|
|
|
TaskComplete();
|
|
break;
|
|
|
|
default:
|
|
BaseClass::StartTask( pTask );
|
|
}
|
|
}
|
|
|
|
void CNPC_Dog::InputTurnBoneFollowersOff( inputdata_t &inputdata )
|
|
{
|
|
if ( m_bBoneFollowersActive )
|
|
{
|
|
m_bBoneFollowersActive = false;
|
|
m_BoneFollowerManager.DestroyBoneFollowers();
|
|
}
|
|
|
|
}
|
|
|
|
void CNPC_Dog::InputTurnBoneFollowersOn( inputdata_t &inputdata )
|
|
{
|
|
if ( !m_bBoneFollowersActive )
|
|
{
|
|
m_bBoneFollowersActive = true;
|
|
m_BoneFollowerManager.InitBoneFollowers( this, ARRAYSIZE(pFollowerBoneNames), pFollowerBoneNames );
|
|
}
|
|
}
|
|
|
|
AI_BEGIN_CUSTOM_NPC( npc_dog, CNPC_Dog )
|
|
|
|
DECLARE_USES_SCHEDULE_PROVIDER( CAI_FollowBehavior )
|
|
|
|
DECLARE_ACTIVITY( ACT_DOG_THROW )
|
|
DECLARE_ACTIVITY( ACT_DOG_PICKUP )
|
|
DECLARE_ACTIVITY( ACT_DOG_WAITING )
|
|
DECLARE_ACTIVITY( ACT_DOG_CATCH )
|
|
|
|
DECLARE_CONDITION( COND_DOG_LOST_PHYSICS_ENTITY )
|
|
|
|
DECLARE_TASK( TASK_DOG_DELAY_SWAT )
|
|
DECLARE_TASK( TASK_DOG_GET_PATH_TO_PHYSOBJ )
|
|
DECLARE_TASK( TASK_DOG_LAUNCH_ITEM )
|
|
DECLARE_TASK( TASK_DOG_PICKUP_ITEM )
|
|
DECLARE_TASK( TASK_DOG_FACE_OBJECT )
|
|
DECLARE_TASK( TASK_DOG_WAIT_FOR_OBJECT )
|
|
DECLARE_TASK( TASK_DOG_CATCH_OBJECT )
|
|
|
|
DECLARE_TASK( TASK_DOG_WAIT_FOR_TARGET_TO_FACE )
|
|
DECLARE_TASK( TASK_DOG_SETUP_THROW_TARGET )
|
|
|
|
DECLARE_ANIMEVENT( AE_DOG_THROW )
|
|
DECLARE_ANIMEVENT( AE_DOG_PICKUP )
|
|
DECLARE_ANIMEVENT( AE_DOG_CATCH )
|
|
DECLARE_ANIMEVENT( AE_DOG_PICKUP_NOEFFECT )
|
|
|
|
|
|
DEFINE_SCHEDULE
|
|
(
|
|
SCHED_DOG_FIND_OBJECT,
|
|
|
|
" Tasks"
|
|
" TASK_DOG_DELAY_SWAT 3"
|
|
" TASK_DOG_GET_PATH_TO_PHYSOBJ 0"
|
|
" TASK_RUN_PATH 0"
|
|
" TASK_WAIT_FOR_MOVEMENT 0"
|
|
" TASK_DOG_FACE_OBJECT 0"
|
|
" TASK_FACE_IDEAL 0"
|
|
" TASK_DOG_PICKUP_ITEM 0"
|
|
" TASK_DOG_SETUP_THROW_TARGET 0"
|
|
" TASK_FACE_TARGET 0.5"
|
|
" TASK_DOG_WAIT_FOR_TARGET_TO_FACE 0"
|
|
" TASK_DOG_LAUNCH_ITEM 0"
|
|
""
|
|
" Interrupts"
|
|
" COND_DOG_LOST_PHYSICS_ENTITY"
|
|
)
|
|
|
|
DEFINE_SCHEDULE
|
|
(
|
|
SCHED_DOG_WAIT_THROW_OBJECT,
|
|
" Tasks"
|
|
" TASK_DOG_SETUP_THROW_TARGET 0"
|
|
" TASK_FACE_TARGET 0.5"
|
|
" TASK_DOG_WAIT_FOR_TARGET_TO_FACE 0"
|
|
" TASK_DOG_LAUNCH_ITEM 0"
|
|
""
|
|
" Interrupts"
|
|
" COND_DOG_LOST_PHYSICS_ENTITY"
|
|
)
|
|
|
|
DEFINE_SCHEDULE
|
|
(
|
|
SCHED_DOG_CATCH_OBJECT,
|
|
|
|
" Tasks"
|
|
" TASK_DOG_WAIT_FOR_OBJECT 0"
|
|
" TASK_DOG_CATCH_OBJECT 0"
|
|
" TASK_FACE_PLAYER 0.5"
|
|
" TASK_DOG_WAIT_FOR_TARGET_TO_FACE 0"
|
|
" TASK_DOG_LAUNCH_ITEM 0"
|
|
" TASK_SET_FAIL_SCHEDULE SCHEDULE:SCHED_IDLE_STAND"
|
|
""
|
|
" Interrupts"
|
|
" COND_DOG_LOST_PHYSICS_ENTITY"
|
|
)
|
|
|
|
AI_END_CUSTOM_NPC()
|