//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $NoKeywords: $
//=============================================================================//

#ifndef AI_BEHAVIOR_H
#define AI_BEHAVIOR_H

#include "ai_component.h"
#include "ai_basenpc.h"
#include "ai_default.h"
#include "AI_Criteria.h"
#include "networkvar.h"

#ifdef DEBUG
#pragma warning(push)
#include <typeinfo>
#pragma warning(pop)
#pragma warning(disable:4290)
#endif

#if defined( _WIN32 )
#pragma once
#endif

//-----------------------------------------------------------------------------
// CAI_Behavior...
//
// Purpose:	The core component that defines a behavior in an NPC by selecting
//			schedules and running tasks
//
//			Intended to be used as an organizational tool as well as a way
//			for various NPCs to share behaviors without sharing an inheritance
//			relationship, and without cramming those behaviors into the base
//			NPC class.
//-----------------------------------------------------------------------------


//-----------------------------------------------------------------------------
// Purpose: Base class defines interface to behaviors and provides bridging
//			methods
//-----------------------------------------------------------------------------

class IBehaviorBackBridge;

//-------------------------------------

abstract_class CAI_BehaviorBase : public CAI_Component
{
	DECLARE_CLASS( CAI_BehaviorBase, CAI_Component )
public:
	CAI_BehaviorBase(CAI_BaseNPC *pOuter = NULL)
	 : 	CAI_Component(pOuter),
	 	m_pBackBridge(NULL)
	{
	}

	virtual const char *GetName() = 0;

	virtual bool KeyValue( const char *szKeyName, const char *szValue ) 
	{
		return false;
	}
	
	bool IsRunning()								{ Assert( GetOuter() ); return ( GetOuter()->GetRunningBehavior() == this ); }
	virtual bool CanSelectSchedule()				{ return true; }
	virtual void BeginScheduleSelection() 			{}
	virtual void EndScheduleSelection() 			{}
	
	void SetBackBridge( IBehaviorBackBridge *pBackBridge )
	{
		Assert( m_pBackBridge == NULL || pBackBridge == NULL );
		m_pBackBridge = pBackBridge;
	}

	void BridgePrecache()									{ Precache();		}
	void BridgeSpawn()										{ Spawn();			}
	void BridgeUpdateOnRemove()								{ UpdateOnRemove();	}
	void BridgeEvent_Killed( const CTakeDamageInfo &info )	{ Event_Killed( info );	}
	void BridgeCleanupOnDeath( CBaseEntity *pCulprit, bool bFireDeathOutput )		{ CleanupOnDeath( pCulprit, bFireDeathOutput ); }

	void BridgeOnChangeHintGroup( string_t oldGroup, string_t newGroup ) { 	OnChangeHintGroup( oldGroup, newGroup ); }

	void BridgeGatherConditions()					{ GatherConditions(); }
	void BridgePrescheduleThink()					{ PrescheduleThink(); }
	void BridgeOnScheduleChange()					{ OnScheduleChange(); }
	void BridgeOnStartSchedule( int scheduleType );

	int  BridgeSelectSchedule();
	bool BridgeSelectFailSchedule( int failedSchedule, int failedTask, AI_TaskFailureCode_t taskFailCode, int *pResult );
	bool BridgeStartTask( const Task_t *pTask );
	bool BridgeRunTask( const Task_t *pTask);
	bool BridgeAimGun( void );
	int BridgeTranslateSchedule( int scheduleType );
	bool BridgeGetSchedule( int localScheduleID, CAI_Schedule **ppResult );
	bool BridgeTaskName(int taskID, const char **);
	Activity BridgeNPC_TranslateActivity( Activity activity );
	void BridgeBuildScheduleTestBits()		{ BuildScheduleTestBits(); }
	bool BridgeIsCurTaskContinuousMove( bool *pResult );
	void BridgeOnMovementFailed()					{ OnMovementFailed(); }
	void BridgeOnMovementComplete()					{ OnMovementComplete(); }
	float BridgeGetDefaultNavGoalTolerance();
	bool BridgeFValidateHintType( CAI_Hint *pHint, bool *pResult );
	bool BridgeIsValidEnemy( CBaseEntity *pEnemy );
	CBaseEntity *BridgeBestEnemy();
	bool BridgeIsValidCover( const Vector &vLocation, CAI_Hint const *pHint );
	bool BridgeIsValidShootPosition( const Vector &vLocation, CAI_Node *pNode, CAI_Hint const *pHint );
	float BridgeGetMaxTacticalLateralMovement( void );
	bool BridgeShouldIgnoreSound( CSound *pSound );
	void BridgeOnSeeEntity( CBaseEntity *pEntity );
	void BridgeOnFriendDamaged( CBaseCombatCharacter *pSquadmate, CBaseEntity *pAttacker );
	bool BridgeIsInterruptable( void );
	bool BridgeIsNavigationUrgent( void );
	bool BridgeShouldPlayerAvoid( void );
	int	 BridgeOnTakeDamage_Alive( const CTakeDamageInfo &info );
	float BridgeGetReasonableFacingDist( void );
	bool BridgeShouldAlwaysThink( bool *pResult );
	void BridgeOnChangeActiveWeapon( CBaseCombatWeapon *pOldWeapon, CBaseCombatWeapon *pNewWeapon );
	void BridgeOnRestore();
	virtual bool BridgeSpeakMapmakerInterruptConcept( string_t iszConcept );
	bool BridgeCanFlinch( void );
	bool BridgeIsCrouching( void );
	bool BridgeIsCrouchedActivity( Activity activity );
	bool BridgeQueryHearSound( CSound *pSound );
	bool BridgeCanRunAScriptedNPCInteraction( bool bForced );
	Activity BridgeGetFlinchActivity( bool bHeavyDamage, bool bGesture );
	bool BridgeOnCalcBaseMove( AILocalMoveGoal_t *pMoveGoal, float distClear, AIMoveResult_t *pResult );
	void BridgeModifyOrAppendCriteria( AI_CriteriaSet& criteriaSet );
	void BridgeTeleport( const Vector *newPosition, const QAngle *newAngles, const Vector *newVelocity );
	void BridgeHandleAnimEvent( animevent_t *pEvent );

	virtual void GatherConditions();
	virtual void GatherConditionsNotActive() { return; } // Override this and your behavior will call this in place of GatherConditions() when your behavior is NOT the active one.
	virtual void OnUpdateShotRegulator() {}

	virtual CAI_ClassScheduleIdSpace *GetClassScheduleIdSpace();

	virtual int  DrawDebugTextOverlays( int text_offset );

	virtual int	Save( ISave &save );
	virtual int	Restore( IRestore &restore );

	static void SaveBehaviors(ISave &save, CAI_BehaviorBase *pCurrentBehavior, CAI_BehaviorBase **ppBehavior, int nBehaviors );
	static int RestoreBehaviors(IRestore &restore, CAI_BehaviorBase **ppBehavior, int nBehaviors ); // returns index of "current" behavior, or -1

protected:

	int GetNpcState() { return GetOuter()->m_NPCState; }

	virtual void Precache()										{}
	virtual void Spawn()										{}
	virtual void UpdateOnRemove()								{}
	virtual void Event_Killed( const CTakeDamageInfo &info )	{}
	virtual void CleanupOnDeath( CBaseEntity *pCulprit, bool bFireDeathOutput ) {}
	
	virtual void PrescheduleThink();
	virtual void OnScheduleChange();
	virtual void OnStartSchedule( int scheduleType );

	virtual int SelectSchedule();
	virtual int	SelectFailSchedule( int failedSchedule, int failedTask, AI_TaskFailureCode_t taskFailCode );
	virtual void StartTask( const Task_t *pTask );
	virtual void RunTask( const Task_t *pTask );
	virtual void AimGun( void );
	virtual int TranslateSchedule( int scheduleType );
	virtual CAI_Schedule *GetSchedule(int schedule);
	virtual const char *GetSchedulingErrorName();
	virtual void BuildScheduleTestBits() {}
	bool IsCurSchedule( int schedId, bool fIdeal = true );


	CAI_Hint *		GetHintNode()							{ return GetOuter()->GetHintNode(); }
	const CAI_Hint *GetHintNode() const						{ return GetOuter()->GetHintNode(); }
	void			SetHintNode( CAI_Hint *pHintNode )		{ GetOuter()->SetHintNode( pHintNode ); }
	void			ClearHintNode( float reuseDelay = 0.0 )	{ GetOuter()->ClearHintNode( reuseDelay ); }

protected:
	// Used by derived classes to chain a task to a task that might not be the 
	// one they are currently handling:
	void	ChainStartTask( int task, float taskData = 0 );
	void	ChainRunTask( int task, float taskData = 0 );

protected:

	virtual Activity NPC_TranslateActivity( Activity activity );

	virtual bool IsCurTaskContinuousMove();
	virtual void OnMovementFailed() {};
	virtual void OnMovementComplete() {};
	virtual float GetDefaultNavGoalTolerance();
	virtual bool FValidateHintType( CAI_Hint *pHint );

	virtual	bool IsValidEnemy( CBaseEntity *pEnemy );
	virtual CBaseEntity *BestEnemy();
	virtual	bool IsValidCover( const Vector &vLocation, CAI_Hint const *pHint );
	virtual	bool IsValidShootPosition( const Vector &vLocation, CAI_Node *pNode, CAI_Hint const *pHint );
	virtual float GetMaxTacticalLateralMovement( void );
	virtual bool ShouldIgnoreSound( CSound *pSound );
	virtual void OnSeeEntity( CBaseEntity *pEntity );
	virtual void OnFriendDamaged( CBaseCombatCharacter *pSquadmate, CBaseEntity *pAttacker );
	virtual bool IsInterruptable( void );
	virtual bool IsNavigationUrgent( void );
	virtual int	 OnTakeDamage_Alive( const CTakeDamageInfo &info );
	virtual float GetReasonableFacingDist( void );
	virtual bool ShouldPlayerAvoid( void );
	virtual bool CanFlinch( void );
	virtual bool IsCrouching( void );
	virtual bool IsCrouchedActivity( Activity activity );
	virtual bool QueryHearSound( CSound *pSound );
	virtual bool CanRunAScriptedNPCInteraction( bool bForced );
	virtual Activity GetFlinchActivity( bool bHeavyDamage, bool bGesture );
	virtual bool OnCalcBaseMove( AILocalMoveGoal_t *pMoveGoal, float distClear, AIMoveResult_t *pResult );
	virtual void ModifyOrAppendCriteria( AI_CriteriaSet& criteriaSet );
	virtual void Teleport( const Vector *newPosition, const QAngle *newAngles, const Vector *newVelocity );
	virtual void HandleAnimEvent( animevent_t *pEvent );

	virtual bool ShouldAlwaysThink();

	virtual void OnChangeActiveWeapon( CBaseCombatWeapon *pOldWeapon, CBaseCombatWeapon *pNewWeapon ) {};
	virtual bool SpeakMapmakerInterruptConcept( string_t iszConcept ) { return false; };
	
	virtual void OnRestore() {};
	
	bool NotifyChangeBehaviorStatus( bool fCanFinishSchedule = false );

	bool HaveSequenceForActivity( Activity activity )		{ return GetOuter()->HaveSequenceForActivity( activity ); }
	
	//---------------------------------

	string_t			GetHintGroup()			{ return GetOuter()->GetHintGroup();	}
	void				ClearHintGroup()			{ GetOuter()->ClearHintGroup();			}
	void				SetHintGroup( string_t name )	{ GetOuter()->SetHintGroup( name );		}

	virtual void		OnChangeHintGroup( string_t oldGroup, string_t newGroup ) {}

	//
	// These allow derived classes to implement custom schedules
	//
	static CAI_GlobalScheduleNamespace *GetSchedulingSymbols()		{ return CAI_BaseNPC::GetSchedulingSymbols(); }
	static bool				LoadSchedules()							{ return true; }
	virtual bool			IsBehaviorSchedule( int scheduleType )	{ return false; }

	CAI_Navigator *			GetNavigator() 							{ return GetOuter()->GetNavigator(); 		}
	CAI_Motor *				GetMotor() 								{ return GetOuter()->GetMotor(); 			}
	CAI_TacticalServices *	GetTacticalServices()					{ return GetOuter()->GetTacticalServices();	}

	bool 				 m_fOverrode;
	IBehaviorBackBridge *m_pBackBridge;

	DECLARE_DATADESC();
};

//-----------------------------------------------------------------------------
// Purpose: Template provides provides back bridge to owning class and 
//			establishes namespace settings
//-----------------------------------------------------------------------------

template <class NPC_CLASS = CAI_BaseNPC, const int ID_SPACE_OFFSET = 100000>
class CAI_Behavior : public CAI_ComponentWithOuter<NPC_CLASS, CAI_BehaviorBase>
{
public:
	DECLARE_CLASS_NOFRIEND( CAI_Behavior, NPC_CLASS );
	
	enum
	{
		NEXT_TASK 			= ID_SPACE_OFFSET,
		NEXT_SCHEDULE 		= ID_SPACE_OFFSET,
		NEXT_CONDITION 		= ID_SPACE_OFFSET
	};

	void SetCondition( int condition )
	{
		if ( condition >= ID_SPACE_OFFSET && condition < ID_SPACE_OFFSET + 10000 ) // it's local to us
			condition = GetClassScheduleIdSpace()->ConditionLocalToGlobal( condition );
		this->GetOuter()->SetCondition( condition );
	}

	bool HasCondition( int condition )
	{
		if ( condition >= ID_SPACE_OFFSET && condition < ID_SPACE_OFFSET + 10000 ) // it's local to us
			condition = GetClassScheduleIdSpace()->ConditionLocalToGlobal( condition );
		return this->GetOuter()->HasCondition( condition );
	}

	bool HasInterruptCondition( int condition )
	{
		if ( condition >= ID_SPACE_OFFSET && condition < ID_SPACE_OFFSET + 10000 ) // it's local to us
			condition = GetClassScheduleIdSpace()->ConditionLocalToGlobal( condition );
		return this->GetOuter()->HasInterruptCondition( condition );
	}

	void ClearCondition( int condition )
	{
		if ( condition >= ID_SPACE_OFFSET && condition < ID_SPACE_OFFSET + 10000 ) // it's local to us
			condition = GetClassScheduleIdSpace()->ConditionLocalToGlobal( condition );
		this->GetOuter()->ClearCondition( condition );
	}

protected:
	CAI_Behavior(NPC_CLASS *pOuter = NULL)
	 : CAI_ComponentWithOuter<NPC_CLASS, CAI_BehaviorBase>(pOuter)
	{
	}

	static CAI_GlobalScheduleNamespace *GetSchedulingSymbols()
	{
		return NPC_CLASS::GetSchedulingSymbols();
	}
	virtual CAI_ClassScheduleIdSpace *GetClassScheduleIdSpace()
	{
		return this->GetOuter()->GetClassScheduleIdSpace();
	}

	static CAI_ClassScheduleIdSpace &AccessClassScheduleIdSpaceDirect()
	{
		return NPC_CLASS::AccessClassScheduleIdSpaceDirect();
	}

private:
	virtual bool IsBehaviorSchedule( int scheduleType ) { return ( scheduleType >= ID_SPACE_OFFSET && scheduleType < ID_SPACE_OFFSET + 10000 ); }
};


//-----------------------------------------------------------------------------
// Purpose: Some bridges a little more complicated to allow behavior to see 
//			what base class would do or control order in which it's donw
//-----------------------------------------------------------------------------

abstract_class IBehaviorBackBridge
{
public:
	virtual void 		 BackBridge_GatherConditions() = 0;
	virtual int 		 BackBridge_SelectSchedule() = 0;
	virtual int 		 BackBridge_TranslateSchedule( int scheduleType ) = 0;
	virtual Activity 	 BackBridge_NPC_TranslateActivity( Activity activity ) = 0;
	virtual bool		 BackBridge_IsValidEnemy(CBaseEntity *pEnemy) = 0;
	virtual CBaseEntity* BackBridge_BestEnemy(void) = 0;
	virtual bool		 BackBridge_IsValidCover( const Vector &vLocation, CAI_Hint const *pHint ) = 0;
	virtual bool		 BackBridge_IsValidShootPosition( const Vector &vLocation, CAI_Node *pNode, CAI_Hint const *pHint ) = 0;
	virtual float		 BackBridge_GetMaxTacticalLateralMovement( void ) = 0;
	virtual bool		 BackBridge_ShouldIgnoreSound( CSound *pSound ) = 0;
	virtual void		 BackBridge_OnSeeEntity( CBaseEntity *pEntity ) = 0;
	virtual void		 BackBridge_OnFriendDamaged( CBaseCombatCharacter *pSquadmate, CBaseEntity *pAttacker ) = 0;
	virtual bool		 BackBridge_IsInterruptable( void ) = 0;
	virtual bool		 BackBridge_IsNavigationUrgent( void ) = 0;
	virtual bool		 BackBridge_ShouldPlayerAvoid( void ) = 0;
	virtual int			 BackBridge_OnTakeDamage_Alive( const CTakeDamageInfo &info ) = 0;
	virtual float		 BackBridge_GetDefaultNavGoalTolerance() = 0;
	virtual float		 BackBridge_GetReasonableFacingDist( void ) = 0;
	virtual bool		 BackBridge_CanFlinch( void ) = 0;
	virtual bool		 BackBridge_IsCrouching( void ) = 0;
	virtual bool		 BackBridge_IsCrouchedActivity( Activity activity ) = 0;
	virtual bool		 BackBridge_QueryHearSound( CSound *pSound ) = 0;
	virtual bool		 BackBridge_CanRunAScriptedNPCInteraction( bool bForced ) = 0;
	virtual Activity	 BackBridge_GetFlinchActivity( bool bHeavyDamage, bool bGesture ) = 0;
	virtual bool		 BackBridge_OnCalcBaseMove( AILocalMoveGoal_t *pMoveGoal, float distClear, AIMoveResult_t *pResult ) = 0;
	virtual void		 BackBridge_ModifyOrAppendCriteria( AI_CriteriaSet& criteriaSet ) = 0;
	virtual void		 BackBridge_Teleport( const Vector *newPosition, const QAngle *newAngles, const Vector *newVelocity ) = 0;

	virtual void		 BackBridge_HandleAnimEvent( animevent_t *pEvent ) = 0;

//-------------------------------------

};

//-----------------------------------------------------------------------------
// Purpose: The common instantiation of the above template
//-----------------------------------------------------------------------------

typedef CAI_Behavior<> CAI_SimpleBehavior;

//-----------------------------------------------------------------------------
// Purpose: Base class for AIs that want to act as a host for CAI_Behaviors
//			NPCs aren't required to use this, but probably want to.
//-----------------------------------------------------------------------------

template <class BASE_NPC>
class CAI_BehaviorHost : public BASE_NPC,
						 private IBehaviorBackBridge
{
public:
	DECLARE_CLASS_NOFRIEND( CAI_BehaviorHost, BASE_NPC );

	CAI_BehaviorHost()
	  : m_pCurBehavior(NULL)
	{
#ifdef DEBUG
  		m_fDebugInCreateBehaviors = false;
#endif
	}

	void CleanupOnDeath( CBaseEntity *pCulprit = NULL, bool bFireDeathOutput = true );

	virtual int		Save( ISave &save );
	virtual int		Restore( IRestore &restore );
	virtual bool 	CreateComponents();

	// Automatically called during entity construction, derived class calls AddBehavior()
	virtual bool 	CreateBehaviors()	{ return true; }
	
	// forces movement and sets a new schedule
	virtual bool	ScheduledMoveToGoalEntity( int scheduleType, CBaseEntity *pGoalEntity, Activity movementActivity );
	virtual bool	ScheduledFollowPath( int scheduleType, CBaseEntity *pPathStart, Activity movementActivity );
	virtual void	ForceSelectedGo(CBaseEntity *pPlayer, const Vector &targetPos, const Vector &traceDir, bool bRun);
	virtual void	ForceSelectedGoRandom(void);

	// Bridges
	void			Precache();
	void			NPCInit();
	void			UpdateOnRemove();
	void			Event_Killed( const CTakeDamageInfo &info );
	void 			GatherConditions();
	void 			PrescheduleThink();
	int 			SelectSchedule();
	void			KeepRunningBehavior();
	int				SelectFailSchedule( int failedSchedule, int failedTask, AI_TaskFailureCode_t taskFailCode );
	void 			OnScheduleChange();
	void			OnStartSchedule( int scheduleType );
	int 			TranslateSchedule( int scheduleType );
	void 			StartTask( const Task_t *pTask );
	void 			RunTask( const Task_t *pTask );
	void			AimGun( void );
	CAI_Schedule *	GetSchedule(int localScheduleID);
	const char *	TaskName(int taskID);
	void			BuildScheduleTestBits();

	void			OnChangeHintGroup( string_t oldGroup, string_t newGroup );
	
	Activity 		NPC_TranslateActivity( Activity activity );

	bool			IsCurTaskContinuousMove();
	void			OnMovementFailed();
	void			OnMovementComplete();
	bool			FValidateHintType( CAI_Hint *pHint );
	float			GetDefaultNavGoalTolerance();

	bool			IsValidEnemy(CBaseEntity *pEnemy);
	CBaseEntity*	BestEnemy(void);
	bool			IsValidCover( const Vector &vLocation, CAI_Hint const *pHint );
	bool			IsValidShootPosition( const Vector &vLocation, CAI_Node *pNode, CAI_Hint const *pHint );
	float			GetMaxTacticalLateralMovement( void );
	bool			ShouldIgnoreSound( CSound *pSound );
	void			OnSeeEntity( CBaseEntity *pEntity );
	void			OnFriendDamaged( CBaseCombatCharacter *pSquadmate, CBaseEntity *pAttacker );
	bool			IsInterruptable( void );
	bool			IsNavigationUrgent( void );
	bool			ShouldPlayerAvoid( void );
	int				OnTakeDamage_Alive( const CTakeDamageInfo &info );
	float			GetReasonableFacingDist( void );
	bool			CanFlinch( void );
	bool			IsCrouching( void );
	bool			IsCrouchedActivity( Activity activity );
	bool			QueryHearSound( CSound *pSound );
	bool			CanRunAScriptedNPCInteraction( bool bForced );
	Activity		GetFlinchActivity( bool bHeavyDamage, bool bGesture );
	bool			OnCalcBaseMove( AILocalMoveGoal_t *pMoveGoal, float distClear, AIMoveResult_t *pResult );
	void			HandleAnimEvent( animevent_t *pEvent );
	
	bool			ShouldAlwaysThink();

	void			OnChangeActiveWeapon( CBaseCombatWeapon *pOldWeapon, CBaseCombatWeapon *pNewWeapon );
	virtual bool	SpeakMapmakerInterruptConcept( string_t iszConcept );

	void			OnRestore();

	void			ModifyOrAppendCriteria( AI_CriteriaSet& set );
	void			Teleport( const Vector *newPosition, const QAngle *newAngles, const Vector *newVelocity );

	//---------------------------------

	virtual bool	OnBehaviorChangeStatus( CAI_BehaviorBase *pBehavior, bool fCanFinishSchedule );
	virtual void	OnChangeRunningBehavior( CAI_BehaviorBase *pOldBehavior,  CAI_BehaviorBase *pNewBehavior );

protected:
	void			AddBehavior( CAI_BehaviorBase *pBehavior );
	
	bool			BehaviorSelectSchedule();
	virtual bool	ShouldBehaviorSelectSchedule( CAI_BehaviorBase *pBehavior ) { return true; }

	bool 			IsRunningBehavior() const;
	CAI_BehaviorBase *GetRunningBehavior();
	CAI_BehaviorBase *DeferSchedulingToBehavior( CAI_BehaviorBase *pNewBehavior );
	void			ChangeBehaviorTo( CAI_BehaviorBase *pNewBehavior );

	CAI_Schedule *	GetNewSchedule();
	CAI_Schedule *	GetFailSchedule();
private:
	void 			BackBridge_GatherConditions();
	int				BackBridge_SelectSchedule();
	int				BackBridge_TranslateSchedule( int scheduleType );
	Activity		BackBridge_NPC_TranslateActivity( Activity activity );
	bool			BackBridge_IsValidEnemy(CBaseEntity *pEnemy);
	CBaseEntity*	BackBridge_BestEnemy(void);
	bool			BackBridge_IsValidCover( const Vector &vLocation, CAI_Hint const *pHint );
	bool			BackBridge_IsValidShootPosition( const Vector &vLocation, CAI_Node *pNode, CAI_Hint const *pHint );
	float			BackBridge_GetMaxTacticalLateralMovement( void );
	bool			BackBridge_ShouldIgnoreSound( CSound *pSound );
	void			BackBridge_OnSeeEntity( CBaseEntity *pEntity );
	void			BackBridge_OnFriendDamaged( CBaseCombatCharacter *pSquadmate, CBaseEntity *pAttacker );
	bool			BackBridge_IsInterruptable( void );
	bool			BackBridge_IsNavigationUrgent( void );
	bool			BackBridge_ShouldPlayerAvoid( void );
	int				BackBridge_OnTakeDamage_Alive( const CTakeDamageInfo &info );
	float			BackBridge_GetDefaultNavGoalTolerance();
	float			BackBridge_GetReasonableFacingDist( void );
	bool			BackBridge_CanFlinch( void );
	bool			BackBridge_IsCrouching( void );
	bool			BackBridge_IsCrouchedActivity( Activity activity );
	bool			BackBridge_QueryHearSound( CSound *pSound );
	bool			BackBridge_CanRunAScriptedNPCInteraction( bool bForced );
	Activity		BackBridge_GetFlinchActivity( bool bHeavyDamage, bool bGesture );
	bool			BackBridge_OnCalcBaseMove( AILocalMoveGoal_t *pMoveGoal, float distClear, AIMoveResult_t *pResult );
	void			BackBridge_ModifyOrAppendCriteria( AI_CriteriaSet& criteriaSet );
	void			BackBridge_Teleport( const Vector *newPosition, const QAngle *newAngles, const Vector *newVelocity );

	void			BackBridge_HandleAnimEvent( animevent_t *pEvent );

	CAI_BehaviorBase **AccessBehaviors();
	int				NumBehaviors();

	CAI_BehaviorBase *			   m_pCurBehavior;
	CUtlVector<CAI_BehaviorBase *> m_Behaviors;

	bool			m_bCalledBehaviorSelectSchedule;
	
#ifdef DEBUG
	bool 			m_fDebugInCreateBehaviors;
#endif
	
};

//-----------------------------------------------------------------------------

// The first frame a behavior begins schedule selection, it won't have had it's GatherConditions()
// called. To fix this, BeginScheduleSelection() manually calls the new behavior's GatherConditions(),
// but sets this global so that the baseclass GatherConditions() isn't called as well.
extern bool g_bBehaviorHost_PreventBaseClassGatherConditions;

//-----------------------------------------------------------------------------

inline void CAI_BehaviorBase::BridgeOnStartSchedule( int scheduleType )
{
	int localId = AI_IdIsGlobal( scheduleType ) ? GetClassScheduleIdSpace()->ScheduleGlobalToLocal( scheduleType ) : scheduleType;
	OnStartSchedule( localId );
}

//-------------------------------------

inline int CAI_BehaviorBase::BridgeSelectSchedule()
{
	int result = SelectSchedule();

	if ( IsBehaviorSchedule( result ) )
		return GetClassScheduleIdSpace()->ScheduleLocalToGlobal( result );

	return result;
}

//-------------------------------------

inline bool CAI_BehaviorBase::BridgeSelectFailSchedule( int failedSchedule, int failedTask, AI_TaskFailureCode_t taskFailCode, int *pResult )
{
	m_fOverrode = true;
	int result = SelectFailSchedule( failedSchedule, failedTask, taskFailCode );
	if ( m_fOverrode )
	{
		if ( result != SCHED_NONE )
		{
			if ( IsBehaviorSchedule( result ) )
				*pResult = GetClassScheduleIdSpace()->ScheduleLocalToGlobal( result );
			else
				*pResult = result;
			return true;
		}
		Warning( "An AI behavior is in control but has no recommended schedule\n" );
	}
	return false;
}

//-------------------------------------

inline bool CAI_BehaviorBase::BridgeStartTask( const Task_t *pTask )
{
	m_fOverrode = true;
	StartTask( pTask );
	return m_fOverrode;
}

//-------------------------------------

inline bool CAI_BehaviorBase::BridgeRunTask( const Task_t *pTask)
{
	m_fOverrode = true;
	RunTask( pTask );
	return m_fOverrode;
}

//-------------------------------------

inline bool CAI_BehaviorBase::BridgeAimGun( void )
{
	m_fOverrode = true;
	AimGun();
	return m_fOverrode;
}

//-------------------------------------

inline void CAI_BehaviorBase::ChainStartTask( int task, float taskData )
{
	Task_t tempTask = { task, taskData }; 

	bool fPrevOverride = m_fOverrode;
	GetOuter()->StartTask( (const Task_t *)&tempTask );
	m_fOverrode = fPrevOverride;;
}

//-------------------------------------

inline void CAI_BehaviorBase::ChainRunTask( int task, float taskData )
{ 
	Task_t tempTask = { task, taskData }; 
	bool fPrevOverride = m_fOverrode;
	GetOuter()->RunTask( (const Task_t *)	&tempTask );
	m_fOverrode = fPrevOverride;;
}

//-------------------------------------

inline int CAI_BehaviorBase::BridgeTranslateSchedule( int scheduleType )
{
	int localId = AI_IdIsGlobal( scheduleType ) ? GetClassScheduleIdSpace()->ScheduleGlobalToLocal( scheduleType ) : scheduleType;
	int result = TranslateSchedule( localId );
	
	return result;
}

//-------------------------------------

inline bool CAI_BehaviorBase::BridgeGetSchedule( int localScheduleID, CAI_Schedule **ppResult )
{
	*ppResult = GetSchedule( localScheduleID );
	return (*ppResult != NULL );
}

//-------------------------------------

inline bool CAI_BehaviorBase::BridgeTaskName( int taskID, const char **ppResult )
{
	if ( AI_IdIsLocal( taskID ) )
	{
		*ppResult = GetSchedulingSymbols()->TaskIdToSymbol( GetClassScheduleIdSpace()->TaskLocalToGlobal( taskID ) );
		return (*ppResult != NULL );
	}
	return false;
}

//-------------------------------------

inline Activity CAI_BehaviorBase::BridgeNPC_TranslateActivity( Activity activity )
{
	return NPC_TranslateActivity( activity );
}

//-------------------------------------

inline bool CAI_BehaviorBase::BridgeIsCurTaskContinuousMove( bool *pResult )
{
	bool fPrevOverride = m_fOverrode;
	m_fOverrode = true;
	*pResult = IsCurTaskContinuousMove();
	bool result = m_fOverrode;
	m_fOverrode = fPrevOverride;
	return result;
}

//-------------------------------------

inline bool CAI_BehaviorBase::BridgeFValidateHintType( CAI_Hint *pHint, bool *pResult )
{
	bool fPrevOverride = m_fOverrode;
	m_fOverrode = true;
	*pResult = FValidateHintType( pHint );
	bool result = m_fOverrode;
	m_fOverrode = fPrevOverride;
	return result;
}

//-------------------------------------

inline bool CAI_BehaviorBase::BridgeIsValidEnemy( CBaseEntity *pEnemy )
{
	return IsValidEnemy( pEnemy );
}

//-------------------------------------

inline CBaseEntity *CAI_BehaviorBase::BridgeBestEnemy()
{
	return BestEnemy();
}

//-------------------------------------

inline bool CAI_BehaviorBase::BridgeIsValidCover( const Vector &vLocation, CAI_Hint const *pHint )
{
	return IsValidCover( vLocation, pHint );
}

//-------------------------------------

inline bool CAI_BehaviorBase::BridgeIsValidShootPosition( const Vector &vLocation, CAI_Node *pNode, CAI_Hint const *pHint )
{
	return IsValidShootPosition( vLocation, pNode, pHint );
}

//-------------------------------------

inline float CAI_BehaviorBase::BridgeGetMaxTacticalLateralMovement( void )
{
	return GetMaxTacticalLateralMovement();
}

//-------------------------------------

inline bool CAI_BehaviorBase::BridgeShouldIgnoreSound( CSound *pSound )
{
	return ShouldIgnoreSound( pSound );
}

//-------------------------------------

inline void CAI_BehaviorBase::BridgeOnSeeEntity( CBaseEntity *pEntity )
{
	OnSeeEntity( pEntity );
}

//-------------------------------------

inline void CAI_BehaviorBase::BridgeOnFriendDamaged( CBaseCombatCharacter *pSquadmate, CBaseEntity *pAttacker )
{
	OnFriendDamaged( pSquadmate, pAttacker );
}

//-------------------------------------

inline bool CAI_BehaviorBase::BridgeIsInterruptable( void )
{
	return IsInterruptable();
}

//-------------------------------------

inline bool CAI_BehaviorBase::BridgeIsNavigationUrgent( void )
{
	return IsNavigationUrgent();
}

//-------------------------------------

inline bool CAI_BehaviorBase::BridgeCanFlinch( void )
{
	return CanFlinch();
}

//-------------------------------------

inline bool CAI_BehaviorBase::BridgeIsCrouching( void )
{
	return IsCrouching();
}

//-------------------------------------

inline bool CAI_BehaviorBase::BridgeIsCrouchedActivity( Activity activity )
{
	return IsCrouchedActivity( activity );
}

inline bool CAI_BehaviorBase::BridgeQueryHearSound( CSound *pSound )
{
	return QueryHearSound( pSound );
}

//-------------------------------------

inline bool CAI_BehaviorBase::BridgeCanRunAScriptedNPCInteraction( bool bForced )
{
	return CanRunAScriptedNPCInteraction( bForced );
}

//-------------------------------------

inline bool CAI_BehaviorBase::BridgeShouldPlayerAvoid( void )
{
	return ShouldPlayerAvoid();
}

//-------------------------------------

inline int CAI_BehaviorBase::BridgeOnTakeDamage_Alive( const CTakeDamageInfo &info )
{
	return OnTakeDamage_Alive( info );
}

//-------------------------------------

inline float CAI_BehaviorBase::BridgeGetReasonableFacingDist( void )
{
	return GetReasonableFacingDist();
}

//-------------------------------------

inline bool CAI_BehaviorBase::BridgeShouldAlwaysThink( bool *pResult )
{
	bool fPrevOverride = m_fOverrode;
	m_fOverrode = true;
	*pResult = ShouldAlwaysThink();
	bool result = m_fOverrode;
	m_fOverrode = fPrevOverride;
	return result;
}

//-------------------------------------

inline void CAI_BehaviorBase::BridgeOnChangeActiveWeapon( CBaseCombatWeapon *pOldWeapon, CBaseCombatWeapon *pNewWeapon )
{
	OnChangeActiveWeapon( pOldWeapon, pNewWeapon );
}

//-------------------------------------

inline bool CAI_BehaviorBase::BridgeSpeakMapmakerInterruptConcept( string_t iszConcept )
{
	return SpeakMapmakerInterruptConcept( iszConcept );
}

//-------------------------------------

inline void CAI_BehaviorBase::BridgeOnRestore()
{
	OnRestore();
}

//-------------------------------------

inline float CAI_BehaviorBase::BridgeGetDefaultNavGoalTolerance()
{
	return GetDefaultNavGoalTolerance();
}

//-----------------------------------------------------------------------------

inline Activity CAI_BehaviorBase::BridgeGetFlinchActivity( bool bHeavyDamage, bool bGesture )
{
	return GetFlinchActivity( bHeavyDamage, bGesture );
}

//-----------------------------------------------------------------------------

inline bool CAI_BehaviorBase::BridgeOnCalcBaseMove( AILocalMoveGoal_t *pMoveGoal, float distClear, AIMoveResult_t *pResult )
{
	return OnCalcBaseMove( pMoveGoal, distClear, pResult );
}

//-----------------------------------------------------------------------------

inline void CAI_BehaviorBase::BridgeModifyOrAppendCriteria( AI_CriteriaSet& criteriaSet )
{
	ModifyOrAppendCriteria( criteriaSet );
}

//-----------------------------------------------------------------------------

inline void CAI_BehaviorBase::BridgeTeleport( const Vector *newPosition, const QAngle *newAngles, const Vector *newVelocity )
{
	Teleport( newPosition, newAngles, newVelocity );
}

//-----------------------------------------------------------------------------

inline void CAI_BehaviorBase::BridgeHandleAnimEvent( animevent_t *pEvent )
{
	HandleAnimEvent( pEvent );
}

//-----------------------------------------------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::CleanupOnDeath( CBaseEntity *pCulprit, bool bFireDeathOutput )
{
	DeferSchedulingToBehavior( NULL );
	for( int i = 0; i < m_Behaviors.Count(); i++ )
	{
		m_Behaviors[i]->BridgeCleanupOnDeath( pCulprit, bFireDeathOutput );
	}
	BaseClass::CleanupOnDeath( pCulprit, bFireDeathOutput );
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::GatherConditions()					
{ 
	// Iterate over behaviors and call GatherConditionsNotActive() on each behavior
	// not currently active.
	for( int i = 0; i < m_Behaviors.Count(); i++ )
	{
		if( m_Behaviors[i] != m_pCurBehavior )
		{
			m_Behaviors[i]->GatherConditionsNotActive();
		}
	}

	if ( m_pCurBehavior )
		m_pCurBehavior->BridgeGatherConditions(); 
	else
		BaseClass::GatherConditions();
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::BackBridge_GatherConditions()
{
	if ( g_bBehaviorHost_PreventBaseClassGatherConditions )
		return;

	BaseClass::GatherConditions();
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::OnScheduleChange()
{ 
	if ( m_pCurBehavior )
		m_pCurBehavior->BridgeOnScheduleChange(); 
	BaseClass::OnScheduleChange();
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::OnStartSchedule( int scheduleType )
{
	if ( m_pCurBehavior )
		m_pCurBehavior->BridgeOnStartSchedule( scheduleType ); 
	BaseClass::OnStartSchedule( scheduleType );
}

//-------------------------------------

template <class BASE_NPC>
inline int CAI_BehaviorHost<BASE_NPC>::BackBridge_SelectSchedule() 
{
	return BaseClass::SelectSchedule();
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::BehaviorSelectSchedule()
{
	for ( int i = 0; i < m_Behaviors.Count(); i++ )
	{
		if ( m_Behaviors[i]->CanSelectSchedule() && ShouldBehaviorSelectSchedule( m_Behaviors[i] ) )
		{
			DeferSchedulingToBehavior( m_Behaviors[i] );
			return true;
		}
	}
	
	DeferSchedulingToBehavior( NULL );
	return false;
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::IsRunningBehavior() const
{
	return ( m_pCurBehavior != NULL );
}

//-------------------------------------

template <class BASE_NPC>
inline CAI_BehaviorBase *CAI_BehaviorHost<BASE_NPC>::GetRunningBehavior()
{
	return m_pCurBehavior;
}

//-------------------------------------

template <class BASE_NPC>
inline CAI_Schedule *CAI_BehaviorHost<BASE_NPC>::GetNewSchedule()
{
	m_bCalledBehaviorSelectSchedule = false;
	CAI_Schedule *pResult = BaseClass::GetNewSchedule();
	if ( !m_bCalledBehaviorSelectSchedule && m_pCurBehavior )
		DeferSchedulingToBehavior( NULL );
	return pResult;
}

//-------------------------------------

template <class BASE_NPC>
inline CAI_Schedule *CAI_BehaviorHost<BASE_NPC>::GetFailSchedule()
{
	m_bCalledBehaviorSelectSchedule = false;
	CAI_Schedule *pResult = BaseClass::GetFailSchedule();
	if ( !m_bCalledBehaviorSelectSchedule && m_pCurBehavior )
		DeferSchedulingToBehavior( NULL );
	return pResult;
}

//------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::ChangeBehaviorTo( CAI_BehaviorBase *pNewBehavior )
{
	bool change = ( m_pCurBehavior != pNewBehavior );
	CAI_BehaviorBase *pOldBehavior = m_pCurBehavior;
	m_pCurBehavior = pNewBehavior;
	
	if ( change ) 
	{
		if ( m_pCurBehavior )
		{
			m_pCurBehavior->BeginScheduleSelection();

			g_bBehaviorHost_PreventBaseClassGatherConditions = true;
			m_pCurBehavior->GatherConditions();
			g_bBehaviorHost_PreventBaseClassGatherConditions = false;
		}

		if ( pOldBehavior )
		{
			pOldBehavior->EndScheduleSelection();
			this->VacateStrategySlot();
		}

		OnChangeRunningBehavior( pOldBehavior, pNewBehavior );
	}
}

//-------------------------------------

template <class BASE_NPC>
inline CAI_BehaviorBase *CAI_BehaviorHost<BASE_NPC>::DeferSchedulingToBehavior( CAI_BehaviorBase *pNewBehavior )
{
	CAI_BehaviorBase *pOldBehavior = m_pCurBehavior;
	ChangeBehaviorTo( pNewBehavior );
	return pOldBehavior;
}

//-------------------------------------

template <class BASE_NPC>
inline int CAI_BehaviorHost<BASE_NPC>::BackBridge_TranslateSchedule( int scheduleType ) 
{
	return BaseClass::TranslateSchedule( scheduleType );
}

//-------------------------------------

template <class BASE_NPC>
inline int CAI_BehaviorHost<BASE_NPC>::TranslateSchedule( int scheduleType ) 
{
	if ( m_pCurBehavior )
	{
		return m_pCurBehavior->BridgeTranslateSchedule( scheduleType );
	}
	return BaseClass::TranslateSchedule( scheduleType );
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::PrescheduleThink()
{
	BaseClass::PrescheduleThink();
	if ( m_pCurBehavior )
		m_pCurBehavior->BridgePrescheduleThink();
}	

//-------------------------------------

template <class BASE_NPC>
inline int CAI_BehaviorHost<BASE_NPC>::SelectSchedule()
{
	m_bCalledBehaviorSelectSchedule = true;
	if ( m_pCurBehavior )
	{
		return m_pCurBehavior->BridgeSelectSchedule();
	}

	return BaseClass::SelectSchedule();
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::KeepRunningBehavior()
{
	if ( m_pCurBehavior )
		m_bCalledBehaviorSelectSchedule = true;
}

//-------------------------------------

template <class BASE_NPC>
inline int CAI_BehaviorHost<BASE_NPC>::SelectFailSchedule( int failedSchedule, int failedTask, AI_TaskFailureCode_t taskFailCode )
{
	m_bCalledBehaviorSelectSchedule = true;
	int result = 0;
	if ( m_pCurBehavior && m_pCurBehavior->BridgeSelectFailSchedule( failedSchedule, failedTask, taskFailCode, &result ) )
		return result;
	return BaseClass::SelectFailSchedule( failedSchedule, failedTask, taskFailCode );
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::StartTask( const Task_t *pTask )
{
	if ( m_pCurBehavior && m_pCurBehavior->BridgeStartTask( pTask ) )
		return;
	BaseClass::StartTask( pTask );
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::RunTask( const Task_t *pTask )
{
	if ( m_pCurBehavior && m_pCurBehavior->BridgeRunTask( pTask ) )
		return;
	BaseClass::RunTask( pTask );
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::AimGun( void )
{
	if ( m_pCurBehavior && m_pCurBehavior->BridgeAimGun() )
		return;
	BaseClass::AimGun();
}

//-------------------------------------

template <class BASE_NPC>
inline CAI_Schedule *CAI_BehaviorHost<BASE_NPC>::GetSchedule(int localScheduleID)
{
	CAI_Schedule *pResult;
	if ( m_pCurBehavior && m_pCurBehavior->BridgeGetSchedule( localScheduleID, &pResult ) )
		return pResult;
	return BaseClass::GetSchedule( localScheduleID );
}

//-------------------------------------

template <class BASE_NPC>
inline const char *CAI_BehaviorHost<BASE_NPC>::TaskName(int taskID)
{
	const char *pszResult = NULL;
	if ( m_pCurBehavior && m_pCurBehavior->BridgeTaskName( taskID, &pszResult ) )
		return pszResult;
	return BaseClass::TaskName( taskID );
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::BuildScheduleTestBits()
{
	if ( m_pCurBehavior )
		m_pCurBehavior->BridgeBuildScheduleTestBits(); 
	BaseClass::BuildScheduleTestBits();
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::OnChangeHintGroup( string_t oldGroup, string_t newGroup )
{
	for( int i = 0; i < m_Behaviors.Count(); i++ )
	{
		m_Behaviors[i]->BridgeOnChangeHintGroup( oldGroup, newGroup );
	}
	BaseClass::OnChangeHintGroup( oldGroup, newGroup );
}

//-------------------------------------

template <class BASE_NPC>
inline Activity CAI_BehaviorHost<BASE_NPC>::BackBridge_NPC_TranslateActivity( Activity activity )
{
	return BaseClass::NPC_TranslateActivity( activity );
}

//-------------------------------------

template <class BASE_NPC>
inline Activity CAI_BehaviorHost<BASE_NPC>::NPC_TranslateActivity( Activity activity )
{
	if ( m_pCurBehavior )
	{
		return m_pCurBehavior->BridgeNPC_TranslateActivity( activity );
	}
	return BaseClass::NPC_TranslateActivity( activity );
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::IsCurTaskContinuousMove()
{
	bool result = false;
	if ( m_pCurBehavior && m_pCurBehavior->BridgeIsCurTaskContinuousMove( &result ) )
		return result;
	return BaseClass::IsCurTaskContinuousMove();
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::OnMovementFailed()
{ 
	if ( m_pCurBehavior )
		m_pCurBehavior->BridgeOnMovementFailed(); 
	BaseClass::OnMovementFailed();
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::OnMovementComplete()
{ 
	if ( m_pCurBehavior )
		m_pCurBehavior->BridgeOnMovementComplete(); 
	BaseClass::OnMovementComplete();
}

//-------------------------------------

template <class BASE_NPC>
inline float CAI_BehaviorHost<BASE_NPC>::GetDefaultNavGoalTolerance()
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeGetDefaultNavGoalTolerance();
	return BaseClass::GetDefaultNavGoalTolerance();
}

//-------------------------------------

template <class BASE_NPC>
inline float CAI_BehaviorHost<BASE_NPC>::BackBridge_GetDefaultNavGoalTolerance() 
{
	return BaseClass::GetDefaultNavGoalTolerance();
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::FValidateHintType( CAI_Hint *pHint )
{
	bool result = false;
	if ( m_pCurBehavior && m_pCurBehavior->BridgeFValidateHintType( pHint, &result ) )
		return result;
	return BaseClass::FValidateHintType( pHint );
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::BackBridge_IsValidEnemy(CBaseEntity *pEnemy)
{
	return BaseClass::IsValidEnemy( pEnemy );
}

//-------------------------------------

template <class BASE_NPC>
inline CBaseEntity *CAI_BehaviorHost<BASE_NPC>::BackBridge_BestEnemy(void)
{
	return BaseClass::BestEnemy();
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::BackBridge_IsValidCover( const Vector &vLocation, CAI_Hint const *pHint )
{
	return BaseClass::IsValidCover( vLocation, pHint );
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::BackBridge_IsValidShootPosition( const Vector &vLocation, CAI_Node *pNode, CAI_Hint const *pHint )
{
	return BaseClass::IsValidShootPosition( vLocation, pNode, pHint );
}

//-------------------------------------

template <class BASE_NPC>
inline float CAI_BehaviorHost<BASE_NPC>::BackBridge_GetMaxTacticalLateralMovement( void )
{
	return BaseClass::GetMaxTacticalLateralMovement();
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::BackBridge_ShouldIgnoreSound( CSound *pSound )
{
	return BaseClass::ShouldIgnoreSound( pSound );
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::BackBridge_OnSeeEntity( CBaseEntity *pEntity )
{
	BaseClass::OnSeeEntity( pEntity );
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::BackBridge_OnFriendDamaged( CBaseCombatCharacter *pSquadmate, CBaseEntity *pAttacker )
{
	BaseClass::OnFriendDamaged( pSquadmate, pAttacker );
}


//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::BackBridge_IsInterruptable( void )
{
	return BaseClass::IsInterruptable();
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::BackBridge_IsNavigationUrgent( void )
{
	return BaseClass::IsNavigationUrgent();
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::BackBridge_CanFlinch( void )
{
	return BaseClass::CanFlinch();
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::BackBridge_IsCrouching( void )
{
	return BaseClass::IsCrouching();
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::BackBridge_IsCrouchedActivity( Activity activity )
{
	return BaseClass::IsCrouchedActivity( activity );
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::BackBridge_QueryHearSound( CSound *pSound )
{
	return BaseClass::QueryHearSound( pSound );
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::BackBridge_CanRunAScriptedNPCInteraction( bool bForced )
{
	return BaseClass::CanRunAScriptedNPCInteraction( bForced );
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::BackBridge_ShouldPlayerAvoid( void )
{
	return BaseClass::ShouldPlayerAvoid();
}

//-------------------------------------

template <class BASE_NPC>
inline int CAI_BehaviorHost<BASE_NPC>::BackBridge_OnTakeDamage_Alive( const CTakeDamageInfo &info )
{
	return BaseClass::OnTakeDamage_Alive( info );
}

//-------------------------------------

template <class BASE_NPC>
inline float CAI_BehaviorHost<BASE_NPC>::BackBridge_GetReasonableFacingDist( void )
{
	return BaseClass::GetReasonableFacingDist();
}

//-------------------------------------

template <class BASE_NPC>
inline Activity CAI_BehaviorHost<BASE_NPC>::BackBridge_GetFlinchActivity( bool bHeavyDamage, bool bGesture )
{
	return BaseClass::GetFlinchActivity( bHeavyDamage, bGesture );
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::BackBridge_OnCalcBaseMove( AILocalMoveGoal_t *pMoveGoal, float distClear, AIMoveResult_t *pResult )
{
	return BaseClass::OnCalcBaseMove( pMoveGoal, distClear, pResult );
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::BackBridge_ModifyOrAppendCriteria( AI_CriteriaSet &criteriaSet )
{
	BaseClass::ModifyOrAppendCriteria( criteriaSet );
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::BackBridge_Teleport( const Vector *newPosition, const QAngle *newAngles, const Vector *newVelocity )
{
	BaseClass::Teleport( newPosition, newAngles, newVelocity );
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::BackBridge_HandleAnimEvent( animevent_t *pEvent )
{
	BaseClass::HandleAnimEvent( pEvent );
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::IsValidEnemy( CBaseEntity *pEnemy )
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeIsValidEnemy( pEnemy );
	
	return BaseClass::IsValidEnemy( pEnemy );
}

//-------------------------------------

template <class BASE_NPC>
inline CBaseEntity *CAI_BehaviorHost<BASE_NPC>::BestEnemy()
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeBestEnemy();
	
	return BaseClass::BestEnemy();
}
	
//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::ShouldAlwaysThink()
{
	bool result = false;
	if ( m_pCurBehavior && m_pCurBehavior->BridgeShouldAlwaysThink( &result ) )
		return result;
	return BaseClass::ShouldAlwaysThink();
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::OnChangeActiveWeapon( CBaseCombatWeapon *pOldWeapon, CBaseCombatWeapon *pNewWeapon )
{
	for( int i = 0; i < m_Behaviors.Count(); i++ )
	{
		m_Behaviors[i]->BridgeOnChangeActiveWeapon( pOldWeapon, pNewWeapon );
	}
	BaseClass::OnChangeActiveWeapon( pOldWeapon, pNewWeapon );
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::SpeakMapmakerInterruptConcept( string_t iszConcept )
{
	for( int i = 0; i < m_Behaviors.Count(); i++ )
	{
		if ( m_Behaviors[i]->BridgeSpeakMapmakerInterruptConcept( iszConcept ) )
			return true;
	}

	return false;
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::OnRestore()
{
	for( int i = 0; i < m_Behaviors.Count(); i++ )
	{
		m_Behaviors[i]->BridgeOnRestore();
	}
	BaseClass::OnRestore();
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::IsValidCover( const Vector &vLocation, CAI_Hint const *pHint )
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeIsValidCover( vLocation, pHint );
	
	return BaseClass::IsValidCover( vLocation, pHint );
}
	
//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::IsValidShootPosition( const Vector &vLocation, CAI_Node *pNode, CAI_Hint const *pHint )
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeIsValidShootPosition( vLocation, pNode, pHint );
	
	return BaseClass::IsValidShootPosition( vLocation, pNode, pHint );
}

//-------------------------------------

template <class BASE_NPC>
inline float CAI_BehaviorHost<BASE_NPC>::GetMaxTacticalLateralMovement( void )
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeGetMaxTacticalLateralMovement();

	return BaseClass::GetMaxTacticalLateralMovement();
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::ShouldIgnoreSound( CSound *pSound )
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeShouldIgnoreSound( pSound );
	
	return BaseClass::ShouldIgnoreSound( pSound );
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::OnSeeEntity( CBaseEntity *pEntity )
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeOnSeeEntity( pEntity );

	BaseClass::OnSeeEntity( pEntity );
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::OnFriendDamaged( CBaseCombatCharacter *pSquadmate, CBaseEntity *pAttacker )
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeOnFriendDamaged( pSquadmate, pAttacker );

	BaseClass::OnFriendDamaged( pSquadmate, pAttacker );
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::IsInterruptable( void )
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeIsInterruptable();
	
	return BaseClass::IsInterruptable();
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::IsNavigationUrgent( void )
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeIsNavigationUrgent();

	return BaseClass::IsNavigationUrgent();
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::CanFlinch( void )
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeCanFlinch();

	return BaseClass::CanFlinch();
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::IsCrouching( void )
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeIsCrouching();

	return BaseClass::IsCrouching();
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::IsCrouchedActivity( Activity activity )
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeIsCrouchedActivity( activity );

	return BaseClass::IsCrouchedActivity( activity );
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::QueryHearSound( CSound *pSound )
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeQueryHearSound( pSound );

	return BaseClass::QueryHearSound( pSound );
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::CanRunAScriptedNPCInteraction( bool bForced )
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeCanRunAScriptedNPCInteraction( bForced );

	return BaseClass::CanRunAScriptedNPCInteraction( bForced );
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::ShouldPlayerAvoid( void )
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeShouldPlayerAvoid();
	
	return BaseClass::ShouldPlayerAvoid();
}

//-------------------------------------

template <class BASE_NPC>
inline int CAI_BehaviorHost<BASE_NPC>::OnTakeDamage_Alive( const CTakeDamageInfo &info )
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeOnTakeDamage_Alive( info );
	
	return BaseClass::OnTakeDamage_Alive( info );
}

//-------------------------------------

template <class BASE_NPC>
inline float CAI_BehaviorHost<BASE_NPC>::GetReasonableFacingDist( void )	
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeGetReasonableFacingDist();
	
	return BaseClass::GetReasonableFacingDist();
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::Precache()
{
	BaseClass::Precache();
	for( int i = 0; i < m_Behaviors.Count(); i++ )
	{
		m_Behaviors[i]->BridgePrecache();
	}
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::ScheduledMoveToGoalEntity( int scheduleType, CBaseEntity *pGoalEntity, Activity movementActivity )
{
	// If a behavior is active, we need to stop running it
	ChangeBehaviorTo( NULL );

	return BaseClass::ScheduledMoveToGoalEntity( scheduleType, pGoalEntity, movementActivity );
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::ScheduledFollowPath( int scheduleType, CBaseEntity *pPathStart, Activity movementActivity )
{
	// If a behavior is active, we need to stop running it
	ChangeBehaviorTo( NULL );

	return BaseClass::ScheduledFollowPath( scheduleType, pPathStart, movementActivity );
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::ForceSelectedGo(CBaseEntity *pPlayer, const Vector &targetPos, const Vector &traceDir, bool bRun)
{
	// If a behavior is active, we need to stop running it
	ChangeBehaviorTo( NULL );

	BaseClass::ForceSelectedGo(pPlayer, targetPos, traceDir, bRun);
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::ForceSelectedGoRandom(void)
{
	// If a behavior is active, we need to stop running it
	ChangeBehaviorTo( NULL );

	BaseClass::ForceSelectedGoRandom();
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::NPCInit()
{
	BaseClass::NPCInit();
	for( int i = 0; i < m_Behaviors.Count(); i++ )
	{
		m_Behaviors[i]->BridgeSpawn();
	}
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::UpdateOnRemove()
{
	for( int i = 0; i < m_Behaviors.Count(); i++ )
	{
		m_Behaviors[i]->BridgeUpdateOnRemove();
	}
	BaseClass::UpdateOnRemove();
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::Event_Killed( const CTakeDamageInfo &info )
{
	for( int i = 0; i < m_Behaviors.Count(); i++ )
	{
		m_Behaviors[i]->BridgeEvent_Killed( info );
	}
	BaseClass::Event_Killed( info );
}

//-------------------------------------

template <class BASE_NPC>
inline Activity CAI_BehaviorHost<BASE_NPC>::GetFlinchActivity( bool bHeavyDamage, bool bGesture )
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeGetFlinchActivity( bHeavyDamage, bGesture );

	return BaseClass::GetFlinchActivity( bHeavyDamage, bGesture );
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::OnCalcBaseMove( AILocalMoveGoal_t *pMoveGoal, float distClear, AIMoveResult_t *pResult )
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeOnCalcBaseMove( pMoveGoal, distClear, pResult );

	return BaseClass::OnCalcBaseMove( pMoveGoal, distClear, pResult );
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::ModifyOrAppendCriteria( AI_CriteriaSet &criteriaSet )
{
	BaseClass::ModifyOrAppendCriteria( criteriaSet );

	if ( m_pCurBehavior )
	{
		// Append active behavior name
		criteriaSet.AppendCriteria( "active_behavior", GetRunningBehavior()->GetName() );
		
		m_pCurBehavior->BridgeModifyOrAppendCriteria( criteriaSet );
		return;
	}
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::Teleport( const Vector *newPosition, const QAngle *newAngles, const Vector *newVelocity )
{
	if ( m_pCurBehavior )
	{
		m_pCurBehavior->BridgeTeleport( newPosition, newAngles, newVelocity );
		return;
	}

	BaseClass::Teleport( newPosition, newAngles, newVelocity );
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::HandleAnimEvent( animevent_t *pEvent )
{
	if ( m_pCurBehavior )
		return m_pCurBehavior->BridgeHandleAnimEvent( pEvent );

	return BaseClass::HandleAnimEvent( pEvent );
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::OnBehaviorChangeStatus(  CAI_BehaviorBase *pBehavior, bool fCanFinishSchedule )
{
	if ( pBehavior == GetRunningBehavior() && !pBehavior->CanSelectSchedule() && !fCanFinishSchedule )
	{
		DeferSchedulingToBehavior( NULL );
		return true;
	}
	return false;
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::OnChangeRunningBehavior( CAI_BehaviorBase *pOldBehavior,  CAI_BehaviorBase *pNewBehavior )
{
}

//-------------------------------------

template <class BASE_NPC>
inline void CAI_BehaviorHost<BASE_NPC>::AddBehavior( CAI_BehaviorBase *pBehavior )
{
#ifdef DEBUG
	Assert( m_Behaviors.Find( pBehavior ) == m_Behaviors.InvalidIndex() );
	Assert( m_fDebugInCreateBehaviors );
	for ( int i = 0; i < m_Behaviors.Count(); i++)
	{
		Assert( typeid(*m_Behaviors[i]) != typeid(*pBehavior) );
	}
#endif
	m_Behaviors.AddToTail( pBehavior );
	pBehavior->SetOuter( this );
	pBehavior->SetBackBridge( this );
}

//-------------------------------------

template <class BASE_NPC>
inline CAI_BehaviorBase **CAI_BehaviorHost<BASE_NPC>::AccessBehaviors()
{
	if (m_Behaviors.Count())
		return m_Behaviors.Base();
	return NULL;
}

//-------------------------------------

template <class BASE_NPC>
inline int CAI_BehaviorHost<BASE_NPC>::NumBehaviors()
{
	return m_Behaviors.Count();
}

//-------------------------------------

template <class BASE_NPC>
inline int CAI_BehaviorHost<BASE_NPC>::Save( ISave &save )
{
	int result = BaseClass::Save( save );
	if ( result )
		CAI_BehaviorBase::SaveBehaviors( save, m_pCurBehavior, AccessBehaviors(), NumBehaviors() );
	return result;
}

//-------------------------------------

template <class BASE_NPC>
inline int CAI_BehaviorHost<BASE_NPC>::Restore( IRestore &restore )
{
	int result = BaseClass::Restore( restore );
	if ( result )
	{
		int iCurrent = CAI_BehaviorBase::RestoreBehaviors( restore, AccessBehaviors(), NumBehaviors() );
		if ( iCurrent != -1 )
			m_pCurBehavior = AccessBehaviors()[iCurrent];
		else
			m_pCurBehavior = NULL;
	}
	return result;
}

//-------------------------------------

template <class BASE_NPC>
inline bool CAI_BehaviorHost<BASE_NPC>::CreateComponents()
{
	if ( BaseClass::CreateComponents() )
	{
#ifdef DEBUG
		m_fDebugInCreateBehaviors = true;
#endif
		bool result = CreateBehaviors();
#ifdef DEBUG
		m_fDebugInCreateBehaviors = false;
#endif
		return result;
	}
	return false;
}

//-----------------------------------------------------------------------------

#endif // AI_BEHAVIOR_H