288 lines
5.9 KiB
C++

//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
//=============================================================================//
#include "cbase.h"
#include "utlrbtree.h"
#include "saverestore_utlvector.h"
#include "ai_goalentity.h"
// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"
//-----------------------------------------------------------------------------
//
// CAI_GoalEntity implementation
//
BEGIN_DATADESC( CAI_GoalEntity )
DEFINE_KEYFIELD( m_iszActor, FIELD_STRING, "Actor" ),
DEFINE_KEYFIELD( m_iszGoal, FIELD_STRING, "Goal" ),
DEFINE_KEYFIELD( m_fStartActive, FIELD_BOOLEAN, "StartActive" ),
DEFINE_KEYFIELD( m_iszConceptModifiers, FIELD_STRING, "BaseConceptModifiers" ),
DEFINE_KEYFIELD( m_SearchType, FIELD_INTEGER, "SearchType" ),
DEFINE_UTLVECTOR( m_actors, FIELD_EHANDLE ),
DEFINE_FIELD( m_hGoalEntity, FIELD_EHANDLE ),
DEFINE_FIELD( m_flags, FIELD_INTEGER ),
DEFINE_THINKFUNC( DelayedRefresh ),
// Inputs
DEFINE_INPUTFUNC( FIELD_VOID, "Activate", InputActivate ),
DEFINE_INPUTFUNC( FIELD_VOID, "UpdateActors", InputUpdateActors ),
DEFINE_INPUTFUNC( FIELD_VOID, "Deactivate", InputDeactivate ),
END_DATADESC()
#ifdef MAPBASE_VSCRIPT
BEGIN_ENT_SCRIPTDESC( CAI_GoalEntity, CBaseEntity, "The base class for goal entities used to control NPC behavior." )
DEFINE_SCRIPTFUNC( IsActive, "Check if the goal entity is active." )
DEFINE_SCRIPTFUNC( NumActors, "Get the number of actors using this goal entity." )
END_SCRIPTDESC();
#endif
//-------------------------------------
void CAI_GoalEntity::Spawn()
{
SetThink( &CAI_GoalEntity::DelayedRefresh );
SetNextThink( gpGlobals->curtime + 0.1f );
}
//-------------------------------------
void CAI_GoalEntity::OnRestore()
{
BaseClass::OnRestore();
ExitDormant();
if ( ( m_flags & ACTIVE ) )
gEntList.AddListenerEntity( this );
}
//-------------------------------------
void CAI_GoalEntity::DelayedRefresh()
{
inputdata_t ignored;
if ( m_fStartActive )
{
Assert( !(m_flags & ACTIVE) );
InputActivate( ignored );
m_fStartActive = false;
}
else
InputUpdateActors( ignored );
SetThink( NULL );
}
//-------------------------------------
void CAI_GoalEntity::PruneActors()
{
for ( int i = m_actors.Count() - 1; i >= 0; i-- )
{
if ( m_actors[i] == NULL || m_actors[i]->IsMarkedForDeletion() || m_actors[i]->GetState() == NPC_STATE_DEAD )
m_actors.FastRemove( i );
}
}
//-------------------------------------
void CAI_GoalEntity::ResolveNames()
{
m_actors.SetCount( 0 );
CBaseEntity *pEntity = NULL;
for (;;)
{
switch ( m_SearchType )
{
case ST_ENTNAME:
{
pEntity = gEntList.FindEntityByName( pEntity, m_iszActor );
break;
}
case ST_CLASSNAME:
{
pEntity = gEntList.FindEntityByClassname( pEntity, STRING( m_iszActor ) );
break;
}
}
if ( !pEntity )
break;
CAI_BaseNPC *pActor = pEntity->MyNPCPointer();
if ( pActor && pActor->GetState() != NPC_STATE_DEAD )
{
AIHANDLE temp;
temp = pActor;
m_actors.AddToTail( temp );
}
}
m_hGoalEntity = gEntList.FindEntityByName( NULL, m_iszGoal );
}
//-------------------------------------
void CAI_GoalEntity::InputActivate( inputdata_t &inputdata )
{
if ( !( m_flags & ACTIVE ) )
{
gEntList.AddListenerEntity( this );
UpdateActors();
m_flags |= ACTIVE;
for ( int i = 0; i < m_actors.Count(); i++ )
{
EnableGoal( m_actors[i] );
}
}
}
//-------------------------------------
void CAI_GoalEntity::InputUpdateActors( inputdata_t &inputdata )
{
int i;
CUtlRBTree<CAI_BaseNPC *> prevActors;
CUtlRBTree<CAI_BaseNPC *>::IndexType_t index;
SetDefLessFunc( prevActors );
PruneActors();
for ( i = 0; i < m_actors.Count(); i++ )
{
prevActors.Insert( m_actors[i] );
}
ResolveNames();
for ( i = 0; i < m_actors.Count(); i++ )
{
index = prevActors.Find( m_actors[i] );
if ( index == prevActors.InvalidIndex() )
{
if ( m_flags & ACTIVE )
EnableGoal( m_actors[i] );
}
else
prevActors.Remove( m_actors[i] );
}
for ( index = prevActors.FirstInorder(); index != prevActors.InvalidIndex(); index = prevActors.NextInorder( index ) )
{
if ( m_flags & ACTIVE )
DisableGoal( prevActors[ index ] );
}
}
//-------------------------------------
void CAI_GoalEntity::InputDeactivate( inputdata_t &inputdata )
{
if ( m_flags & ACTIVE )
{
gEntList.RemoveListenerEntity( this );
UpdateActors();
m_flags &= ~ACTIVE;
for ( int i = 0; i < m_actors.Count(); i++ )
{
DisableGoal( m_actors[i] );
}
}
}
//-------------------------------------
void CAI_GoalEntity::EnterDormant( void )
{
if ( m_flags & ACTIVE )
{
m_flags |= DORMANT;
for ( int i = 0; i < m_actors.Count(); i++ )
{
DisableGoal( m_actors[i] );
}
}
}
//-------------------------------------
void CAI_GoalEntity::ExitDormant( void )
{
if ( m_flags & DORMANT )
{
m_flags &= ~DORMANT;
inputdata_t ignored;
InputUpdateActors( ignored );
}
}
//-------------------------------------
void CAI_GoalEntity::UpdateOnRemove()
{
if ( m_flags & ACTIVE )
{
inputdata_t inputdata;
InputDeactivate( inputdata );
}
BaseClass::UpdateOnRemove();
}
//-------------------------------------
void CAI_GoalEntity::OnEntityCreated( CBaseEntity *pEntity )
{
Assert( m_flags & ACTIVE );
if ( pEntity->MyNPCPointer() )
{
SetThink( &CAI_GoalEntity::DelayedRefresh );
SetNextThink( gpGlobals->curtime + 0.1f );
}
}
//-------------------------------------
void CAI_GoalEntity::OnEntityDeleted( CBaseEntity *pEntity )
{
Assert( pEntity != this );
}
//-----------------------------------------------------------------------------
int CAI_GoalEntity::DrawDebugTextOverlays()
{
char tempstr[512];
int offset = BaseClass::DrawDebugTextOverlays();
Q_snprintf( tempstr, sizeof(tempstr), "Active: %s", IsActive() ? "yes" : "no" );
EntityText( offset, tempstr, 0 );
offset++;
return offset;
}