You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
541 lines
18 KiB
541 lines
18 KiB
// NextBotEventResponderInterface.h
|
|
// Interface for propagating and responding to events
|
|
// Author: Michael Booth, May 2006
|
|
// Copyright (c) 2006 Turtle Rock Studios, Inc. - All Rights Reserved
|
|
|
|
#ifndef _NEXT_BOT_EVENT_RESPONDER_INTERFACE_H_
|
|
#define _NEXT_BOT_EVENT_RESPONDER_INTERFACE_H_
|
|
|
|
class Path;
|
|
class CTakeDamageInfo;
|
|
class CBaseEntity;
|
|
class CDOTABaseAbility;
|
|
|
|
struct CSoundParameters;
|
|
struct animevent_t;
|
|
|
|
#include "ai_speech.h"
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------------------------------
|
|
enum MoveToFailureType
|
|
{
|
|
FAIL_NO_PATH_EXISTS,
|
|
FAIL_STUCK,
|
|
FAIL_FELL_OFF,
|
|
};
|
|
|
|
//--------------------------------------------------------------------------------------------------------------------------
|
|
/**
|
|
* Events propagated to/between components.
|
|
* To add an event, add its signature here and implement its propagation
|
|
* to derived classes via FirstContainedResponder() and NextContainedResponder().
|
|
* NOTE: Also add a translator to the Action class in NextBotBehavior.h.
|
|
*/
|
|
class INextBotEventResponder
|
|
{
|
|
public:
|
|
DECLARE_CLASS_NOBASE( INextBotEventResponder );
|
|
|
|
virtual ~INextBotEventResponder() { }
|
|
|
|
// these methods are used by derived classes to define how events propagate
|
|
virtual INextBotEventResponder *FirstContainedResponder( void ) const { return NULL; }
|
|
virtual INextBotEventResponder *NextContainedResponder( INextBotEventResponder *current ) const { return NULL; }
|
|
|
|
//
|
|
// Events. All events must be 'extended' by calling the derived class explicitly to ensure propagation.
|
|
// Each event must implement its propagation in this interface class.
|
|
//
|
|
virtual void OnLeaveGround( CBaseEntity *ground ); // invoked when bot leaves ground for any reason
|
|
virtual void OnLandOnGround( CBaseEntity *ground ); // invoked when bot lands on the ground after being in the air
|
|
|
|
virtual void OnContact( CBaseEntity *other, CGameTrace *result = NULL ); // invoked when bot touches 'other'
|
|
|
|
virtual void OnMoveToSuccess( const Path *path ); // invoked when a bot reaches the end of the given Path
|
|
virtual void OnMoveToFailure( const Path *path, MoveToFailureType reason ); // invoked when a bot fails to reach the end of the given Path
|
|
virtual void OnStuck( void ); // invoked when bot becomes stuck while trying to move
|
|
virtual void OnUnStuck( void ); // invoked when a previously stuck bot becomes un-stuck and can again move
|
|
|
|
virtual void OnPostureChanged( void ); // when bot has assumed new posture (query IBody for posture)
|
|
|
|
virtual void OnAnimationActivityComplete( int activity ); // when animation activity has finished playing
|
|
virtual void OnAnimationActivityInterrupted( int activity );// when animation activity was replaced by another animation
|
|
virtual void OnAnimationEvent( animevent_t *event ); // when a QC-file animation event is triggered by the current animation sequence
|
|
|
|
virtual void OnIgnite( void ); // when bot starts to burn
|
|
virtual void OnInjured( const CTakeDamageInfo &info ); // when bot is damaged by something
|
|
virtual void OnKilled( const CTakeDamageInfo &info ); // when the bot's health reaches zero
|
|
virtual void OnOtherKilled( CBaseCombatCharacter *victim, const CTakeDamageInfo &info ); // when someone else dies
|
|
|
|
virtual void OnSight( CBaseEntity *subject ); // when subject initially enters bot's visual awareness
|
|
virtual void OnLostSight( CBaseEntity *subject ); // when subject leaves enters bot's visual awareness
|
|
|
|
virtual void OnSound( CBaseEntity *source, const Vector &pos, KeyValues *keys ); // when an entity emits a sound. "pos" is world coordinates of sound. "keys" are from sound's GameData
|
|
virtual void OnSpokeConcept( CBaseCombatCharacter *who, AIConcept_t concept, AI_Response *response ); // when an Actor speaks a concept
|
|
virtual void OnWeaponFired( CBaseCombatCharacter *whoFired, CBaseCombatWeapon *weapon ); // when someone fires a weapon
|
|
|
|
virtual void OnNavAreaChanged( CNavArea *newArea, CNavArea *oldArea ); // when bot enters a new navigation area
|
|
|
|
virtual void OnModelChanged( void ); // when the entity's model has been changed
|
|
|
|
virtual void OnPickUp( CBaseEntity *item, CBaseCombatCharacter *giver ); // when something is added to our inventory
|
|
virtual void OnDrop( CBaseEntity *item ); // when something is removed from our inventory
|
|
|
|
virtual void OnCommandAttack( CBaseEntity *victim ); // attack the given entity
|
|
virtual void OnCommandApproach( const Vector &pos, float range = 0.0f ); // move to within range of the given position
|
|
virtual void OnCommandApproach( CBaseEntity *goal ); // follow the given leader
|
|
virtual void OnCommandRetreat( CBaseEntity *threat, float range = 0.0f ); // retreat from the threat at least range units away (0 == infinite)
|
|
virtual void OnCommandPause( float duration = 0.0f ); // pause for the given duration (0 == forever)
|
|
virtual void OnCommandResume( void ); // resume after a pause
|
|
|
|
virtual void OnCommandString( const char *command ); // for debugging: respond to an arbitrary string representing a generalized command
|
|
|
|
virtual void OnShoved( CBaseEntity *pusher ); // 'pusher' has shoved me
|
|
virtual void OnBlinded( CBaseEntity *blinder ); // 'blinder' has blinded me with a flash of light
|
|
|
|
virtual void OnTerritoryContested( int territoryID ); // territory has been invaded and is changing ownership
|
|
virtual void OnTerritoryCaptured( int territoryID ); // we have captured enemy territory
|
|
virtual void OnTerritoryLost( int territoryID ); // we have lost territory to the enemy
|
|
|
|
virtual void OnWin( void );
|
|
virtual void OnLose( void );
|
|
|
|
#ifdef DOTA_SERVER_DLL
|
|
virtual void OnCommandMoveTo( const Vector &pos );
|
|
virtual void OnCommandMoveToAggressive( const Vector &pos );
|
|
virtual void OnCommandAttack( CBaseEntity *victim, bool bDeny );
|
|
virtual void OnCastAbilityNoTarget( CDOTABaseAbility *ability );
|
|
virtual void OnCastAbilityOnPosition( CDOTABaseAbility *ability, const Vector &pos );
|
|
virtual void OnCastAbilityOnTarget( CDOTABaseAbility *ability, CBaseEntity *target );
|
|
virtual void OnDropItem( const Vector &pos, CBaseEntity *item );
|
|
virtual void OnPickupItem( CBaseEntity *item );
|
|
virtual void OnPickupRune( CBaseEntity *item );
|
|
virtual void OnStop();
|
|
virtual void OnFriendThreatened( CBaseEntity *friendly, CBaseEntity *threat );
|
|
virtual void OnCancelAttack( CBaseEntity *pTarget );
|
|
virtual void OnDominated();
|
|
virtual void OnWarped( Vector vStartPos );
|
|
#endif
|
|
};
|
|
|
|
|
|
inline void INextBotEventResponder::OnLeaveGround( CBaseEntity *ground )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnLeaveGround( ground );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnLandOnGround( CBaseEntity *ground )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnLandOnGround( ground );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnContact( CBaseEntity *other, CGameTrace *result )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnContact( other, result );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnMoveToSuccess( const Path *path )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnMoveToSuccess( path );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnMoveToFailure( const Path *path, MoveToFailureType reason )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnMoveToFailure( path, reason );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnStuck( void )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnStuck();
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnUnStuck( void )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnUnStuck();
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnPostureChanged( void )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnPostureChanged();
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnAnimationActivityComplete( int activity )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnAnimationActivityComplete( activity );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnAnimationActivityInterrupted( int activity )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnAnimationActivityInterrupted( activity );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnAnimationEvent( animevent_t *event )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnAnimationEvent( event );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnIgnite( void )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnIgnite();
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnInjured( const CTakeDamageInfo &info )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnInjured( info );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnKilled( const CTakeDamageInfo &info )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnKilled( info );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnOtherKilled( CBaseCombatCharacter *victim, const CTakeDamageInfo &info )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnOtherKilled( victim, info );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnSight( CBaseEntity *subject )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnSight( subject );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnLostSight( CBaseEntity *subject )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnLostSight( subject );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnSound( CBaseEntity *source, const Vector &pos, KeyValues *keys )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnSound( source, pos, keys );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnSpokeConcept( CBaseCombatCharacter *who, AIConcept_t concept, AI_Response *response )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnSpokeConcept( who, concept, response );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnWeaponFired( CBaseCombatCharacter *whoFired, CBaseCombatWeapon *weapon )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnWeaponFired( whoFired, weapon );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnNavAreaChanged( CNavArea *newArea, CNavArea *oldArea )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnNavAreaChanged( newArea, oldArea );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnModelChanged( void )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnModelChanged();
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnPickUp( CBaseEntity *item, CBaseCombatCharacter *giver )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnPickUp( item, giver );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnDrop( CBaseEntity *item )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnDrop( item );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnShoved( CBaseEntity *pusher )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnShoved( pusher );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnBlinded( CBaseEntity *blinder )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnBlinded( blinder );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnCommandAttack( CBaseEntity *victim )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnCommandAttack( victim );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnCommandApproach( const Vector &pos, float range )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnCommandApproach( pos, range );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnCommandApproach( CBaseEntity *goal )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnCommandApproach( goal );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnCommandRetreat( CBaseEntity *threat, float range )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnCommandRetreat( threat, range );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnCommandPause( float duration )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnCommandPause( duration );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnCommandResume( void )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnCommandResume();
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnCommandString( const char *command )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnCommandString( command );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnTerritoryContested( int territoryID )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnTerritoryContested( territoryID );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnTerritoryCaptured( int territoryID )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnTerritoryCaptured( territoryID );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnTerritoryLost( int territoryID )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnTerritoryLost( territoryID );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnWin( void )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnWin();
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnLose( void )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnLose();
|
|
}
|
|
}
|
|
|
|
#ifdef DOTA_SERVER_DLL
|
|
inline void INextBotEventResponder::OnCommandMoveTo( const Vector &pos )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnCommandMoveTo( pos );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnCommandMoveToAggressive( const Vector &pos )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnCommandMoveToAggressive( pos );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnCommandAttack( CBaseEntity *victim, bool bDeny )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnCommandAttack( victim, bDeny );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnCastAbilityNoTarget( CDOTABaseAbility *ability )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnCastAbilityNoTarget( ability );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnCastAbilityOnPosition( CDOTABaseAbility *ability, const Vector &pos )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnCastAbilityOnPosition( ability, pos );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnCastAbilityOnTarget( CDOTABaseAbility *ability, CBaseEntity *target )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnCastAbilityOnTarget( ability, target );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnDropItem( const Vector &pos, CBaseEntity *item )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnDropItem( pos, item );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnPickupItem( CBaseEntity *item )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnPickupItem( item );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnPickupRune( CBaseEntity *item )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnPickupRune( item );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnStop()
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnStop();
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnFriendThreatened( CBaseEntity *friendly, CBaseEntity *threat )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnFriendThreatened( friendly, threat );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnCancelAttack( CBaseEntity *pTarget )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnCancelAttack( pTarget );
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnDominated()
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnDominated();
|
|
}
|
|
}
|
|
|
|
inline void INextBotEventResponder::OnWarped( Vector vStartPos )
|
|
{
|
|
for ( INextBotEventResponder *sub = FirstContainedResponder(); sub; sub = NextContainedResponder( sub ) )
|
|
{
|
|
sub->OnWarped( vStartPos );
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#endif // _NEXT_BOT_EVENT_RESPONDER_INTERFACE_H_
|