|
|
//========= Copyright � 1996-2005, Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $NoKeywords: $
//=============================================================================//
// Author: Michael S. Booth ([email protected]), 2003
#include "cbase.h"
#include "cs_bot.h"
#include "usermessages.h"
// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"
extern int gmsgBotVoice;
//--------------------------------------------------------------------------------------------------------------
/**
* Returns true if the radio message is an order to do something * NOTE: "Report in" is not considered a "command" because it doesnt ask the bot to go somewhere, or change its mind */ bool CCSBot::IsRadioCommand( RadioType event ) const { if (event == RADIO_AFFIRMATIVE || event == RADIO_NEGATIVE || event == RADIO_ENEMY_SPOTTED || event == RADIO_SECTOR_CLEAR || event == RADIO_REPORTING_IN || event == RADIO_REPORT_IN_TEAM || event == RADIO_ENEMY_DOWN || event == RADIO_INVALID ) return false;
return true; }
//--------------------------------------------------------------------------------------------------------------
/**
* Respond to radio commands from HUMAN players */ void CCSBot::RespondToRadioCommands( void ) { // bots use the chatter system to respond to each other
if (m_radioSubject != NULL && m_radioSubject->IsPlayer()) { CCSPlayer *player = m_radioSubject; if (player->IsBot()) { m_lastRadioCommand = RADIO_INVALID; return; } } if (m_lastRadioCommand == RADIO_INVALID) return;
// a human player has issued a radio command
GetChatter()->ResetRadioSilenceDuration();
// if we are doing something important, ignore the radio
// unless it is a "report in" request - we can do that while we continue to do other things
/// @todo Create "uninterruptable" flag
if (m_lastRadioCommand != RADIO_REPORT_IN_TEAM) { if (IsBusy()) { // consume command
m_lastRadioCommand = RADIO_INVALID; return; } }
// wait for reaction time before responding
// delay needs to be long enough for the radio message we're responding to to finish
float respondTime = 1.0f + 2.0f * GetProfile()->GetReactionTime(); if (IsRogue()) respondTime += 2.0f;
if (gpGlobals->curtime - m_lastRadioRecievedTimestamp < respondTime) return;
// rogues won't follow commands, unless already following the player
if (!IsFollowing() && IsRogue()) { if (IsRadioCommand( m_lastRadioCommand )) { GetChatter()->Negative(); }
// consume command
m_lastRadioCommand = RADIO_INVALID; return; }
CCSPlayer *player = m_radioSubject; if (player == NULL) return;
// respond to command
bool canDo = false; const float inhibitAutoFollowDuration = 60.0f; switch( m_lastRadioCommand ) { case RADIO_REPORT_IN_TEAM: { GetChatter()->ReportingIn(); break; }
case RADIO_FOLLOW_ME: case RADIO_COVER_ME: case RADIO_STICK_TOGETHER_TEAM: case RADIO_REGROUP_TEAM: { if (!IsFollowing()) { Follow( player ); player->AllowAutoFollow(); canDo = true; } break; }
case RADIO_ENEMY_SPOTTED: case RADIO_NEED_BACKUP: case RADIO_TAKING_FIRE: if (!IsFollowing()) { Follow( player ); GetChatter()->Say( "OnMyWay" ); player->AllowAutoFollow(); canDo = false; } break;
case RADIO_TEAM_FALL_BACK: { if (TryToRetreat()) canDo = true; break; }
case RADIO_HOLD_THIS_POSITION: { // find the leader's area
SetTask( HOLD_POSITION ); StopFollowing(); player->InhibitAutoFollow( inhibitAutoFollowDuration ); Hide( TheNavMesh->GetNearestNavArea( m_radioPosition ) ); canDo = true; break; }
case RADIO_GO_GO_GO: case RADIO_STORM_THE_FRONT: StopFollowing(); Hunt(); canDo = true; player->InhibitAutoFollow( inhibitAutoFollowDuration ); break;
case RADIO_GET_OUT_OF_THERE: if (TheCSBots()->IsBombPlanted()) { EscapeFromBomb(); player->InhibitAutoFollow( inhibitAutoFollowDuration ); canDo = true; } break;
case RADIO_SECTOR_CLEAR: { // if this is a defusal scenario, and the bomb is planted,
// and a human player cleared a bombsite, check it off our list too
if (TheCSBots()->GetScenario() == CCSBotManager::SCENARIO_DEFUSE_BOMB) { if (GetTeamNumber() == TEAM_CT && TheCSBots()->IsBombPlanted()) { const CCSBotManager::Zone *zone = TheCSBots()->GetClosestZone( player );
if (zone) { GetGameState()->ClearBombsite( zone->m_index );
// if we are huting for the planted bomb, re-select bombsite
if (GetTask() == FIND_TICKING_BOMB) Idle();
canDo = true; } } } break; }
default: // ignore all other radio commands for now
return; }
if (canDo) { // affirmative
GetChatter()->Affirmative();
// if we agreed to follow a new command, put away our grenade
if (IsRadioCommand( m_lastRadioCommand ) && IsUsingGrenade()) { EquipBestWeapon(); } }
// consume command
m_lastRadioCommand = RADIO_INVALID; }
//--------------------------------------------------------------------------------------------------------------
/**
* Decide if we should move to help the player, return true if we will */ bool CCSBot::RespondToHelpRequest( CCSPlayer *them, Place place, float maxRange ) { if (IsRogue()) return false;
// if we're busy, ignore
if (IsBusy()) return false;
Vector themOrigin = GetCentroid( them );
// if we are too far away, ignore
if (maxRange > 0.0f) { // compute actual travel distance
PathCost cost(this); float travelDistance = NavAreaTravelDistance( m_lastKnownArea, TheNavMesh->GetNearestNavArea( themOrigin ), cost ); if (travelDistance < 0.0f) return false;
if (travelDistance > maxRange) return false; }
if (place == UNDEFINED_PLACE) { // if we have no "place" identifier, go directly to them
// if we are already there, ignore
float rangeSq = (them->GetAbsOrigin() - GetAbsOrigin()).LengthSqr(); const float close = 750.0f * 750.0f; if (rangeSq < close) return true;
MoveTo( themOrigin, FASTEST_ROUTE ); } else { // if we are already there, ignore
if (GetPlace() == place) return true;
// go to where help is needed
Vector pos; if ( GetRandomSpotAtPlace( place, &pos ) ) { MoveTo( pos, FASTEST_ROUTE ); } else { MoveTo( themOrigin, FASTEST_ROUTE ); } }
// acknowledge
GetChatter()->Say( "OnMyWay" );
return true; }
//--------------------------------------------------------------------------------------------------------------
/**
* Send a radio message */ void CCSBot::SendRadioMessage( RadioType event ) { // make sure this is a radio event
if (event <= RADIO_START_1 || event >= RADIO_END) return;
PrintIfWatched( "%3.1f: SendRadioMessage( %s )\n", gpGlobals->curtime, RadioEventName[ event ] );
// note the time the message was sent
TheCSBots()->SetRadioMessageTimestamp( event, GetTeamNumber() );
m_lastRadioSentTimestamp = gpGlobals->curtime;
char slot[2]; slot[1] = '\000';
if (event > RADIO_START_1 && event < RADIO_START_2) { HandleMenu_Radio1( event ); } else if (event > RADIO_START_2 && event < RADIO_START_3) { HandleMenu_Radio2( event ); } else { HandleMenu_Radio3( event ); } }
//--------------------------------------------------------------------------------------------------------------
/**
* Send voice chatter. Also sends the entindex and duration for voice feedback. */ void CCSBot::SpeakAudio( const char *voiceFilename, float duration, int pitch ) { if( !IsAlive() ) return;
if ( IsObserver() ) return;
CRecipientFilter filter; ConstructRadioFilter( filter );
CCSUsrMsg_RawAudio msg;
msg.set_pitch( pitch ); msg.set_entidx( entindex() ); msg.set_duration( duration ); msg.set_voice_filename( voiceFilename );
SendUserMessage ( filter, CS_UM_RawAudio, msg );
GetChatter()->ResetRadioSilenceDuration();
m_voiceEndTimestamp = gpGlobals->curtime + duration; }
//--------------------------------------------------------------------------------------------------------------
/**
* Send voice chatter through the response rules system. */ bool CCSBot::SpeakAudioResponseRules( const char *pConcept, AI_CriteriaSet *criteria, float duration ) { if( !IsAlive() ) return false;
if ( IsObserver() ) return false;
CRecipientFilter filter; ConstructRadioFilter( filter );
AI_CriteriaSet local; if ( !criteria ) criteria = &local;
AIConcept_t concept( pConcept ); if ( Speak( concept, criteria, NULL, 0, &filter ) ) { GetChatter()->ResetRadioSilenceDuration(); m_voiceEndTimestamp = gpGlobals->curtime + duration; return true; }
return false; }
|