source-engine/game/server/ai_scriptconditions.cpp

847 lines
21 KiB
C++
Raw Permalink Normal View History

2020-04-22 12:56:21 -04:00
//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $NoKeywords: $
//=============================================================================//
#include "cbase.h"
#include "IEffects.h"
#include "collisionutils.h"
#include "ai_basenpc.h"
#include "ai_scriptconditions.h"
#include "saverestore_utlvector.h"
// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"
#define SF_ACTOR_AS_ACTIVATOR ( 1 << 0 )
ConVar debugscriptconditions( "ai_debugscriptconditions", "0" );
#define ScrCondDbgMsg( msg ) \
do \
{ \
if ( debugscriptconditions.GetBool() ) \
{ \
DevMsg msg; \
} \
} \
while (0)
//=============================================================================
//
// CAI_ScriptConditions
//
//=============================================================================
LINK_ENTITY_TO_CLASS(ai_script_conditions, CAI_ScriptConditions);
BEGIN_DATADESC( CAI_ScriptConditions )
DEFINE_THINKFUNC( EvaluationThink ),
DEFINE_OUTPUT( m_OnConditionsSatisfied, "OnConditionsSatisfied" ),
DEFINE_OUTPUT( m_OnConditionsTimeout, "OnConditionsTimeout" ),
DEFINE_OUTPUT( m_NoValidActors, "NoValidActors" ),
//---------------------------------
DEFINE_INPUTFUNC( FIELD_VOID, "Enable", InputEnable ),
DEFINE_INPUTFUNC( FIELD_VOID, "Disable", InputDisable ),
//---------------------------------
// Inputs
DEFINE_KEYFIELD(m_fDisabled, FIELD_BOOLEAN, "StartDisabled" ),
DEFINE_FIELD( m_hTarget, FIELD_EHANDLE ),
DEFINE_KEYFIELD(m_Actor, FIELD_STRING, "Actor" ),
DEFINE_KEYFIELD(m_flRequiredTime, FIELD_FLOAT, "RequiredTime" ),
#ifndef HL2_EPISODIC
DEFINE_FIELD( m_hActor, FIELD_EHANDLE ),
DEFINE_EMBEDDED(m_Timer ),
DEFINE_EMBEDDED(m_Timeout ),
#endif
DEFINE_KEYFIELD(m_fMinState, FIELD_INTEGER, "MinimumState" ),
DEFINE_KEYFIELD(m_fMaxState, FIELD_INTEGER, "MaximumState" ),
DEFINE_KEYFIELD(m_fScriptStatus, FIELD_INTEGER, "ScriptStatus" ),
DEFINE_KEYFIELD(m_fActorSeePlayer, FIELD_INTEGER, "ActorSeePlayer" ),
DEFINE_KEYFIELD(m_flPlayerActorProximity, FIELD_FLOAT, "PlayerActorProximity" ),
DEFINE_EMBEDDED(m_PlayerActorProxTester),
DEFINE_KEYFIELD(m_flPlayerActorFOV, FIELD_FLOAT, "PlayerActorFOV" ),
DEFINE_KEYFIELD(m_bPlayerActorFOVTrueCone, FIELD_BOOLEAN, "PlayerActorFOVTrueCone" ),
DEFINE_KEYFIELD(m_fPlayerActorLOS, FIELD_INTEGER, "PlayerActorLOS" ),
DEFINE_KEYFIELD(m_fActorSeeTarget, FIELD_INTEGER, "ActorSeeTarget" ),
DEFINE_KEYFIELD(m_flActorTargetProximity, FIELD_FLOAT, "ActorTargetProximity" ),
DEFINE_EMBEDDED(m_ActorTargetProxTester),
DEFINE_KEYFIELD(m_flPlayerTargetProximity, FIELD_FLOAT, "PlayerTargetProximity" ),
DEFINE_EMBEDDED(m_PlayerTargetProxTester),
DEFINE_KEYFIELD(m_flPlayerTargetFOV, FIELD_FLOAT, "PlayerTargetFOV" ),
DEFINE_KEYFIELD(m_bPlayerTargetFOVTrueCone, FIELD_BOOLEAN, "PlayerTargetFOVTrueCone" ),
DEFINE_KEYFIELD(m_fPlayerTargetLOS, FIELD_INTEGER, "PlayerTargetLOS" ),
DEFINE_KEYFIELD(m_fPlayerBlockingActor, FIELD_INTEGER, "PlayerBlockingActor" ),
DEFINE_KEYFIELD(m_flMinTimeout, FIELD_FLOAT, "MinTimeout" ),
DEFINE_KEYFIELD(m_flMaxTimeout, FIELD_FLOAT, "MaxTimeout" ),
DEFINE_KEYFIELD(m_fActorInPVS, FIELD_INTEGER, "ActorInPVS" ),
DEFINE_KEYFIELD(m_fActorInVehicle, FIELD_INTEGER, "ActorInVehicle" ),
DEFINE_KEYFIELD(m_fPlayerInVehicle, FIELD_INTEGER, "PlayerInVehicle" ),
DEFINE_UTLVECTOR( m_ElementList, FIELD_EMBEDDED ),
DEFINE_FIELD( m_bLeaveAsleep, FIELD_BOOLEAN ),
END_DATADESC()
BEGIN_SIMPLE_DATADESC( CAI_ProxTester )
DEFINE_FIELD( m_distSq, FIELD_FLOAT ),
DEFINE_FIELD( m_fInside, FIELD_BOOLEAN ),
END_DATADESC()
BEGIN_SIMPLE_DATADESC( CAI_ScriptConditionsElement )
DEFINE_FIELD( m_hActor, FIELD_EHANDLE ),
DEFINE_EMBEDDED(m_Timer ),
DEFINE_EMBEDDED(m_Timeout ),
END_DATADESC()
//-----------------------------------------------------------------------------
#define EVALUATOR( name ) { &CAI_ScriptConditions::Eval##name, #name }
CAI_ScriptConditions::EvaluatorInfo_t CAI_ScriptConditions::gm_Evaluators[] =
{
EVALUATOR( ActorSeePlayer ),
EVALUATOR( State ),
EVALUATOR( PlayerActorProximity ),
EVALUATOR( PlayerTargetProximity ),
EVALUATOR( ActorTargetProximity ),
EVALUATOR( PlayerBlockingActor ),
EVALUATOR( PlayerActorLook ),
EVALUATOR( PlayerTargetLook ),
EVALUATOR( ActorSeeTarget),
EVALUATOR( PlayerActorLOS ),
EVALUATOR( PlayerTargetLOS ),
#ifdef HL2_EPISODIC
EVALUATOR( ActorInPVS ),
EVALUATOR( PlayerInVehicle ),
EVALUATOR( ActorInVehicle ),
#endif
};
void CAI_ScriptConditions::OnRestore( void )
{
BaseClass::OnRestore();
#ifndef HL2_EPISODIC
//Old HL2 save game! Fix up to new system.
if ( m_hActor )
{
CAI_ScriptConditionsElement conditionactor;
conditionactor.SetActor( m_hActor );
conditionactor.SetTimeOut( m_Timeout );
conditionactor.SetTimer( m_Timer );
m_ElementList.AddToTail( conditionactor );
m_hActor = NULL;
}
if ( m_ElementList.Count() == 0 && m_Actor == NULL_STRING && m_fDisabled == false )
{
AddNewElement( NULL );
}
#endif
}
//-----------------------------------------------------------------------------
bool CAI_ScriptConditions::EvalState( const EvalArgs_t &args )
{
if ( !args.pActor )
return true;
CAI_BaseNPC *pNpc = args.pActor->MyNPCPointer();
// !!!LATER - fix this code, we shouldn't need the table anymore
// now that we've placed the NPC state defs in a logical order (sjb)
static int stateVals[] =
{
-1, // NPC_STATE_NONE
0, // NPC_STATE_IDLE
1, // NPC_STATE_ALERT
2, // NPC_STATE_COMBAT
-1, // NPC_STATE_SCRIPT
-1, // NPC_STATE_PLAYDEAD
-1, // NPC_STATE_PRONE
-1, // NPC_STATE_DEAD
};
int valState = stateVals[pNpc->m_NPCState];
if ( valState < 0 )
{
if ( pNpc->m_NPCState == NPC_STATE_SCRIPT && m_fScriptStatus >= TRS_TRUE )
return true;
return false;
}
const int valLow = stateVals[m_fMinState];
const int valHigh = stateVals[m_fMaxState];
if ( valLow > valHigh )
{
DevMsg( "Script condition warning: Invalid setting for Maximum/Minimum state\n" );
Disable();
return false;
}
return ( valState >= valLow && valState <= valHigh );
}
//-----------------------------------------------------------------------------
bool CAI_ScriptConditions::EvalActorSeePlayer( const EvalArgs_t &args )
{
if( m_fActorSeePlayer == TRS_NONE )
{
// Don't care, so don't do any work.
return true;
}
if ( !args.pActor )
return true;
bool fCanSeePlayer = args.pActor->MyNPCPointer()->HasCondition( COND_SEE_PLAYER );
return ( (int)m_fActorSeePlayer == (int)fCanSeePlayer );
}
//-----------------------------------------------------------------------------
bool CAI_ScriptConditions::EvalActorSeeTarget( const EvalArgs_t &args )
{
if( m_fActorSeeTarget == TRS_NONE )
{
// Don't care, so don't do any work.
return true;
}
if ( args.pTarget )
{
if ( !args.pActor )
return true;
CAI_BaseNPC *pNPCActor = args.pActor->MyNPCPointer();
#ifdef HL2_EPISODIC
// This is the code we want to have written for HL2, but HL2 shipped without the QuerySeeEntity() call. This #ifdef really wants to be
// something like #ifndef HL2_RETAIL, since this change does want to be in any products that are built henceforth. (sjb)
bool fSee = pNPCActor->FInViewCone( args.pTarget ) && pNPCActor->FVisible( args.pTarget ) && pNPCActor->QuerySeeEntity( args.pTarget );
#else
bool fSee = pNPCActor->FInViewCone( args.pTarget ) && pNPCActor->FVisible( args.pTarget );
#endif//HL2_EPISODIC
if( fSee )
{
if( m_fActorSeeTarget == TRS_TRUE )
{
return true;
}
return false;
}
else
{
if( m_fActorSeeTarget == TRS_FALSE )
{
return true;
}
return false;
}
}
return true;
}
//-----------------------------------------------------------------------------
bool CAI_ScriptConditions::EvalPlayerActorProximity( const EvalArgs_t &args )
{
return ( !args.pActor || m_PlayerActorProxTester.Check( args.pPlayer, args.pActor ) );
}
//-----------------------------------------------------------------------------
bool CAI_ScriptConditions::EvalPlayerTargetProximity( const EvalArgs_t &args )
{
return ( !args.pTarget ||
m_PlayerTargetProxTester.Check( args.pPlayer, args.pTarget ) );
}
//-----------------------------------------------------------------------------
bool CAI_ScriptConditions::EvalActorTargetProximity( const EvalArgs_t &args )
{
return ( !args.pTarget || !args.pActor ||
m_ActorTargetProxTester.Check( args.pActor, args.pTarget ) );
}
//-----------------------------------------------------------------------------
bool CAI_ScriptConditions::EvalPlayerActorLook( const EvalArgs_t &args )
{
return ( !args.pActor ||
IsInFOV( args.pPlayer, args.pActor, m_flPlayerActorFOV, m_bPlayerActorFOVTrueCone ) );
}
//-----------------------------------------------------------------------------
bool CAI_ScriptConditions::EvalPlayerTargetLook( const EvalArgs_t &args )
{
return ( !args.pTarget || IsInFOV( args.pPlayer, args.pTarget, m_flPlayerTargetFOV, m_bPlayerTargetFOVTrueCone ) );
}
//-----------------------------------------------------------------------------
bool CAI_ScriptConditions::EvalPlayerActorLOS( const EvalArgs_t &args )
{
if( m_fPlayerActorLOS == TRS_NONE )
{
// Don't execute expensive code if we don't care.
return true;
}
return ( !args.pActor || PlayerHasLineOfSight( args.pPlayer, args.pActor, m_fPlayerActorLOS == TRS_FALSE ) );
}
//-----------------------------------------------------------------------------
bool CAI_ScriptConditions::EvalPlayerTargetLOS( const EvalArgs_t &args )
{
if( m_fPlayerTargetLOS == TRS_NONE )
{
// Don't execute expensive code if we don't care.
return true;
}
return ( !args.pTarget || PlayerHasLineOfSight( args.pPlayer, args.pTarget, m_fPlayerTargetLOS == TRS_FALSE ) );
}
bool CAI_ScriptConditions::EvalActorInPVS( const EvalArgs_t &args )
{
if( m_fActorInPVS == TRS_NONE )
{
// Don't execute expensive code if we don't care.
return true;
}
return ( !args.pActor || ActorInPlayersPVS( args.pActor, m_fActorInPVS == TRS_FALSE ) );
}
//-----------------------------------------------------------------------------
bool CAI_ScriptConditions::EvalPlayerBlockingActor( const EvalArgs_t &args )
{
if ( m_fPlayerBlockingActor == TRS_NONE )
return true;
#if 0
CAI_BaseNPC *pNpc = args.pActor->MyNPCPointer();
const float testDist = 30.0;
Vector origin = args.pActor->WorldSpaceCenter();
Vector delta = UTIL_YawToVector( args.pActor->GetAngles().y ) * testDist;
Vector vecAbsMins, vecAbsMaxs;
args.pActor->CollisionProp()->WorldSpaceAABB( &vecAbsMins, &vecAbsMaxs );
bool intersect = IsBoxIntersectingRay( vecAbsMins, vecAbsMaxs, origin, delta );
#endif
if ( m_fPlayerBlockingActor == TRS_FALSE )
return true;
return false; // for now, never say player is blocking
}
//-----------------------------------------------------------------------------
bool CAI_ScriptConditions::EvalPlayerInVehicle( const EvalArgs_t &args )
{
// We don't care
if ( m_fPlayerInVehicle == TRS_NONE )
return true;
// Need a player to test
if ( args.pPlayer == NULL )
return false;
// Desired states must match
return ( !!args.pPlayer->IsInAVehicle() == m_fPlayerInVehicle );
}
//-----------------------------------------------------------------------------
bool CAI_ScriptConditions::EvalActorInVehicle( const EvalArgs_t &args )
{
// We don't care
if ( m_fActorInVehicle == TRS_NONE )
return true;
if ( !args.pActor )
return true;
// Must be able to be in a vehicle at all
CBaseCombatCharacter *pBCC = args.pActor->MyCombatCharacterPointer();
if ( pBCC == NULL )
return false;
// Desired states must match
return ( !!pBCC->IsInAVehicle() == m_fActorInVehicle );
}
//-----------------------------------------------------------------------------
void CAI_ScriptConditions::Spawn()
{
Assert( ( m_fMinState == NPC_STATE_IDLE || m_fMinState == NPC_STATE_COMBAT || m_fMinState == NPC_STATE_ALERT ) &&
( m_fMaxState == NPC_STATE_IDLE || m_fMaxState == NPC_STATE_COMBAT || m_fMaxState == NPC_STATE_ALERT ) );
m_PlayerActorProxTester.Init( m_flPlayerActorProximity );
m_PlayerTargetProxTester.Init( m_flPlayerTargetProximity );
m_ActorTargetProxTester.Init( m_flActorTargetProximity );
m_bLeaveAsleep = m_fDisabled;
}
//-----------------------------------------------------------------------------
void CAI_ScriptConditions::Activate()
{
BaseClass::Activate();
// When we spawn, m_fDisabled is initial state as given by worldcraft.
// following that, we keep it updated and it reflects current state.
if( !m_fDisabled )
Enable();
#ifdef HL2_EPISODIC
gEntList.AddListenerEntity( this );
#endif
}
//-----------------------------------------------------------------------------
void CAI_ScriptConditions::UpdateOnRemove( void )
{
gEntList.RemoveListenerEntity( this );
BaseClass::UpdateOnRemove();
m_ElementList.Purge();
}
//-----------------------------------------------------------------------------
void CAI_ScriptConditions::EvaluationThink()
{
if ( m_fDisabled == true )
return;
int iActorsDone = 0;
#ifdef HL2_DLL
if( AI_GetSinglePlayer()->GetFlags() & FL_NOTARGET )
{
ScrCondDbgMsg( ("%s WARNING: Player is NOTARGET. This will affect all LOS conditiosn involving the player!\n", GetDebugName()) );
}
#endif
for ( int i = 0; i < m_ElementList.Count(); )
{
CAI_ScriptConditionsElement *pConditionElement = &m_ElementList[i];
if ( pConditionElement == NULL )
{
i++;
continue;
}
CBaseEntity *pActor = pConditionElement->GetActor();
CBaseEntity *pActivator = this;
#ifdef HL2_EPISODIC
if ( pActor && HasSpawnFlags( SF_ACTOR_AS_ACTIVATOR ) )
{
pActivator = pActor;
}
#endif
AssertMsg( !m_fDisabled, ("Violated invariant between CAI_ScriptConditions disabled state and think func setting") );
if ( m_Actor != NULL_STRING && !pActor )
{
if ( m_ElementList.Count() == 1 )
{
DevMsg( "Warning: Active AI script conditions associated with an non-existant or destroyed NPC\n" );
m_NoValidActors.FireOutput( this, this, 0 );
}
iActorsDone++;
m_ElementList.Remove( i );
continue;
}
i++;
if( m_flMinTimeout > 0 && pConditionElement->GetTimeOut()->Expired() )
{
ScrCondDbgMsg( ( "%s firing output OnConditionsTimeout (%f seconds)\n", STRING( GetEntityName() ), pConditionElement->GetTimeOut()->GetInterval() ) );
iActorsDone++;
m_OnConditionsTimeout.FireOutput( pActivator, this );
continue;
}
bool result = true;
const int nEvaluators = sizeof( gm_Evaluators ) / sizeof( gm_Evaluators[0] );
EvalArgs_t args =
{
pActor,
GetPlayer(),
m_hTarget.Get()
};
for ( int i = 0; i < nEvaluators; ++i )
{
if ( !(this->*gm_Evaluators[i].pfnEvaluator)( args ) )
{
pConditionElement->GetTimer()->Reset();
result = false;
ScrCondDbgMsg( ( "%s failed on: %s\n", GetDebugName(), gm_Evaluators[ i ].pszName ) );
break;
}
}
if ( result )
{
ScrCondDbgMsg( ( "%s waiting... %f\n", GetDebugName(), pConditionElement->GetTimer()->GetRemaining() ) );
}
if ( result && pConditionElement->GetTimer()->Expired() )
{
ScrCondDbgMsg( ( "%s firing output OnConditionsSatisfied\n", GetDebugName() ) );
// Default behavior for now, provide worldcraft option later.
iActorsDone++;
m_OnConditionsSatisfied.FireOutput( pActivator, this );
}
}
//All done!
if ( iActorsDone == m_ElementList.Count() )
{
Disable();
m_ElementList.Purge();
}
SetThinkTime();
}
//-----------------------------------------------------------------------------
int CAI_ScriptConditions::AddNewElement( CBaseEntity *pActor )
{
CAI_ScriptConditionsElement conditionelement;
conditionelement.SetActor( pActor );
if( m_flMaxTimeout > 0 )
{
conditionelement.GetTimeOut()->Set( random->RandomFloat( m_flMinTimeout, m_flMaxTimeout ), false );
}
else
{
conditionelement.GetTimeOut()->Set( m_flMinTimeout, false );
}
conditionelement.GetTimer()->Set( m_flRequiredTime );
if ( m_flRequiredTime > 0 )
{
conditionelement.GetTimer()->Reset();
}
return m_ElementList.AddToTail( conditionelement );
}
//-----------------------------------------------------------------------------
void CAI_ScriptConditions::Enable( void )
{
m_hTarget = gEntList.FindEntityByName( NULL, m_target );
CBaseEntity *pActor = gEntList.FindEntityByName( NULL, m_Actor );
if ( m_ElementList.Count() == 0 )
{
if ( m_Actor != NULL_STRING && pActor == NULL )
{
DevMsg( "Warning: Spawning AI script conditions (%s) associated with an non-existant NPC\n", GetDebugName() );
m_NoValidActors.FireOutput( this, this, 0 );
Disable();
return;
}
if ( pActor && pActor->MyNPCPointer() == NULL )
{
Warning( "Script condition warning: warning actor is not an NPC\n" );
Disable();
return;
}
}
while( pActor != NULL )
{
if( !ActorInList(pActor) )
{
AddNewElement( pActor );
}
pActor = gEntList.FindEntityByName( pActor, m_Actor );
}
//If we are hitting this it means we are using a Target->Player condition
if ( m_Actor == NULL_STRING )
{
if( !ActorInList(pActor) )
{
AddNewElement( NULL );
}
}
m_fDisabled = false;
SetThink( &CAI_ScriptConditions::EvaluationThink );
SetThinkTime();
}
//-----------------------------------------------------------------------------
void CAI_ScriptConditions::Disable( void )
{
SetThink( NULL );
m_fDisabled = true;
}
//-----------------------------------------------------------------------------
void CAI_ScriptConditions::InputEnable( inputdata_t &inputdata )
{
m_bLeaveAsleep = false;
Enable();
}
//-----------------------------------------------------------------------------
void CAI_ScriptConditions::InputDisable( inputdata_t &inputdata )
{
m_bLeaveAsleep = true;
Disable();
}
//-----------------------------------------------------------------------------
bool CAI_ScriptConditions::IsInFOV( CBaseEntity *pViewer, CBaseEntity *pViewed, float fov, bool bTrueCone )
{
CBaseCombatCharacter *pCombatantViewer = (pViewer) ? pViewer->MyCombatCharacterPointer() : NULL;
if ( fov < 360 && pCombatantViewer /*&& pViewed*/ )
{
Vector vLookDir;
Vector vActorDir;
// Halve the fov. As expressed here, fov is the full size of the viewcone.
float flFovDotResult;
flFovDotResult = cos( DEG2RAD( fov / 2 ) );
float fDotPr = 1;
if( bTrueCone )
{
// 3D Check
vLookDir = pCombatantViewer->EyeDirection3D( );
vActorDir = pViewed->EyePosition() - pViewer->EyePosition();
vActorDir.NormalizeInPlace();
fDotPr = vLookDir.Dot(vActorDir);
}
else
{
// 2D Check
vLookDir = pCombatantViewer->EyeDirection2D( );
vActorDir = pViewed->EyePosition() - pViewer->EyePosition();
vActorDir.z = 0.0;
vActorDir.AsVector2D().NormalizeInPlace();
fDotPr = vLookDir.AsVector2D().Dot(vActorDir.AsVector2D());
}
if ( fDotPr < flFovDotResult )
{
if( fov < 0 )
{
// Designer has requested that the player
// NOT be looking at this place.
return true;
}
return false;
}
}
if( fov < 0 )
{
return false;
}
return true;
}
//-----------------------------------------------------------------------------
bool CAI_ScriptConditions::PlayerHasLineOfSight( CBaseEntity *pViewer, CBaseEntity *pViewed, bool fNot )
{
CBaseCombatCharacter *pCombatantViewer = pViewer->MyCombatCharacterPointer();
if( pCombatantViewer )
{
// We always trace towards the player, so we handle players-in-vehicles
if ( pViewed->FVisible( pCombatantViewer ) )
{
// Line of sight exists.
if( fNot )
{
return false;
}
else
{
return true;
}
}
else
{
// No line of sight.
if( fNot )
{
return true;
}
else
{
return false;
}
}
}
return true;
}
//-----------------------------------------------------------------------------
bool CAI_ScriptConditions::ActorInPlayersPVS( CBaseEntity *pActor, bool bNot )
{
if ( pActor == NULL )
return true;
bool bInPVS = !!UTIL_FindClientInPVS( pActor->edict());
if ( bInPVS )
{
if( bNot )
{
return false;
}
else
{
return true;
}
}
else
{
if( bNot )
{
return true;
}
else
{
return false;
}
}
}
//-----------------------------------------------------------------------------
bool CAI_ScriptConditions::ActorInList( CBaseEntity *pActor )
{
for ( int i = 0; i < m_ElementList.Count(); i++ )
{
if ( m_ElementList[i].GetActor() == pActor )
return true;
}
return false;
}
//-----------------------------------------------------------------------------
void CAI_ScriptConditions::OnEntitySpawned( CBaseEntity *pEntity )
{
if( m_fDisabled && m_bLeaveAsleep )
{
// Don't add elements if we're not currently running and don't want to automatically wake up.
// Any spawning NPC's we miss during this time will be found and added when manually Enabled().
return;
}
if ( pEntity->MyNPCPointer() == NULL )
return;
if ( pEntity->NameMatches( m_Actor ) )
{
if ( ActorInList( pEntity ) == false )
{
AddNewElement( pEntity );
if ( m_fDisabled == true && m_bLeaveAsleep == false )
{
Enable();
}
}
}
}
//=============================================================================