jkxr/Projects/Android/jni/OpenJK/code/game/AI_Stormtrooper.cpp
Simon 4597b03873 Initial Commit
Opens in Android Studio but haven't even tried to build it yet (it won't.. I know that much!)
2022-09-18 16:37:21 +01:00

2723 lines
78 KiB
C++

/*
===========================================================================
Copyright (C) 2000 - 2013, Raven Software, Inc.
Copyright (C) 2001 - 2013, Activision, Inc.
Copyright (C) 2013 - 2015, OpenJK contributors
This file is part of the OpenJK source code.
OpenJK is free software; you can redistribute it and/or modify it
under the terms of the GNU General Public License version 2 as
published by the Free Software Foundation.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, see <http://www.gnu.org/licenses/>.
===========================================================================
*/
#include "b_local.h"
#include "g_nav.h"
#include "anims.h"
#include "g_navigator.h"
#include "../cgame/cg_local.h"
#include "g_functions.h"
extern void CG_DrawAlert( vec3_t origin, float rating );
extern void G_AddVoiceEvent( gentity_t *self, int event, int speakDebounceTime );
extern void AI_GroupUpdateSquadstates( AIGroupInfo_t *group, gentity_t *member, int newSquadState );
extern qboolean AI_GroupContainsEntNum( AIGroupInfo_t *group, int entNum );
extern void AI_GroupUpdateEnemyLastSeen( AIGroupInfo_t *group, vec3_t spot );
extern void AI_GroupUpdateClearShotTime( AIGroupInfo_t *group );
extern void NPC_TempLookTarget( gentity_t *self, int lookEntNum, int minLookTime, int maxLookTime );
extern qboolean G_ExpandPointToBBox( vec3_t point, const vec3_t mins, const vec3_t maxs, int ignore, int clipmask );
extern void ChangeWeapon( gentity_t *ent, int newWeapon );
extern void NPC_CheckGetNewWeapon( void );
extern qboolean Q3_TaskIDPending( gentity_t *ent, taskID_t taskType );
extern int GetTime ( int lastTime );
extern void NPC_AimAdjust( int change );
extern qboolean FlyingCreature( gentity_t *ent );
extern void NPC_EvasionSaber( void );
extern qboolean RT_Flying( gentity_t *self );
extern cvar_t *d_asynchronousGroupAI;
#define MAX_VIEW_DIST 1024
#define MAX_VIEW_SPEED 250
#define MAX_LIGHT_INTENSITY 255
#define MIN_LIGHT_THRESHOLD 0.1
#define ST_MIN_LIGHT_THRESHOLD 30
#define ST_MAX_LIGHT_THRESHOLD 180
#define DISTANCE_THRESHOLD 0.075f
#define MIN_TURN_AROUND_DIST_SQ (10000) //(100 squared) don't stop running backwards if your goal is less than 100 away
#define SABER_AVOID_DIST 128.0f//256.0f
#define SABER_AVOID_DIST_SQ (SABER_AVOID_DIST*SABER_AVOID_DIST)
#define DISTANCE_SCALE 0.35f //These first three get your base detection rating, ideally add up to 1
#define FOV_SCALE 0.40f //
#define LIGHT_SCALE 0.25f //
#define SPEED_SCALE 0.25f //These next two are bonuses
#define TURNING_SCALE 0.25f //
#define REALIZE_THRESHOLD 0.6f
#define CAUTIOUS_THRESHOLD ( REALIZE_THRESHOLD * 0.75 )
qboolean NPC_CheckPlayerTeamStealth( void );
static qboolean enemyLOS;
static qboolean enemyCS;
static qboolean enemyInFOV;
static qboolean hitAlly;
static qboolean faceEnemy;
static qboolean doMove;
static qboolean shoot;
static float enemyDist;
static vec3_t impactPos;
int groupSpeechDebounceTime[TEAM_NUM_TEAMS];//used to stop several group AI from speaking all at once
void NPC_Saboteur_Precache( void )
{
G_SoundIndex( "sound/chars/shadowtrooper/cloak.wav" );
G_SoundIndex( "sound/chars/shadowtrooper/decloak.wav" );
}
void Saboteur_Decloak( gentity_t *self, int uncloakTime )
{
if ( self && self->client )
{
if ( self->client->ps.powerups[PW_CLOAKED] && TIMER_Done(self, "decloakwait"))
{//Uncloak
self->client->ps.powerups[PW_CLOAKED] = 0;
self->client->ps.powerups[PW_UNCLOAKING] = level.time + 2000;
//FIXME: temp sound
G_SoundOnEnt( self, CHAN_ITEM, "sound/chars/shadowtrooper/decloak.wav" );
TIMER_Set( self, "nocloak", uncloakTime );
// Can't Recloak
//self->NPC->aiFlags &= ~NPCAI_SHIELDS;
}
}
}
void Saboteur_Cloak( gentity_t *self )
{
if ( self && self->client && self->NPC )
{//FIXME: need to have this timer set once first?
if ( TIMER_Done( self, "nocloak" ) )
{//not sitting around waiting to cloak again
if ( !(self->NPC->aiFlags&NPCAI_SHIELDS) )
{//not allowed to cloak, actually
Saboteur_Decloak( self );
}
else if ( !self->client->ps.powerups[PW_CLOAKED] )
{//cloak
self->client->ps.powerups[PW_CLOAKED] = Q3_INFINITE;
self->client->ps.powerups[PW_UNCLOAKING] = level.time + 2000;
//FIXME: debounce attacks?
//FIXME: temp sound
G_SoundOnEnt( self, CHAN_ITEM, "sound/chars/shadowtrooper/cloak.wav" );
}
}
}
}
//Local state enums
enum
{
LSTATE_NONE = 0,
LSTATE_UNDERFIRE,
LSTATE_INVESTIGATE,
};
void ST_AggressionAdjust( gentity_t *self, int change )
{
int upper_threshold, lower_threshold;
self->NPC->stats.aggression += change;
//FIXME: base this on initial NPC stats
if ( self->client->playerTeam == TEAM_PLAYER )
{//good guys are less aggressive
upper_threshold = 7;
lower_threshold = 1;
}
else
{//bad guys are more aggressive
upper_threshold = 10;
lower_threshold = 3;
}
if ( self->NPC->stats.aggression > upper_threshold )
{
self->NPC->stats.aggression = upper_threshold;
}
else if ( self->NPC->stats.aggression < lower_threshold )
{
self->NPC->stats.aggression = lower_threshold;
}
}
void ST_ClearTimers( gentity_t *ent )
{
TIMER_Set( ent, "chatter", 0 );
TIMER_Set( ent, "duck", 0 );
TIMER_Set( ent, "stand", 0 );
TIMER_Set( ent, "shuffleTime", 0 );
TIMER_Set( ent, "sleepTime", 0 );
TIMER_Set( ent, "enemyLastVisible", 0 );
TIMER_Set( ent, "roamTime", 0 );
TIMER_Set( ent, "hideTime", 0 );
TIMER_Set( ent, "attackDelay", 0 ); //FIXME: Slant for difficulty levels
TIMER_Set( ent, "stick", 0 );
TIMER_Set( ent, "scoutTime", 0 );
TIMER_Set( ent, "flee", 0 );
TIMER_Set( ent, "interrogating", 0 );
TIMER_Set( ent, "verifyCP", 0 );
TIMER_Set( ent, "strafeRight", 0 );
TIMER_Set( ent, "strafeLeft", 0 );
}
enum
{
SPEECH_CHASE,
SPEECH_CONFUSED,
SPEECH_COVER,
SPEECH_DETECTED,
SPEECH_GIVEUP,
SPEECH_LOOK,
SPEECH_LOST,
SPEECH_OUTFLANK,
SPEECH_ESCAPING,
SPEECH_SIGHT,
SPEECH_SOUND,
SPEECH_SUSPICIOUS,
SPEECH_YELL,
SPEECH_PUSHED
};
static void ST_Speech( gentity_t *self, int speechType, float failChance )
{
if ( Q_flrand(0.0f, 1.0f) < failChance )
{
return;
}
if ( failChance >= 0 )
{//a negative failChance makes it always talk
if ( self->NPC->group )
{//group AI speech debounce timer
if ( self->NPC->group->speechDebounceTime > level.time )
{
return;
}
/*
else if ( !self->NPC->group->enemy )
{
if ( groupSpeechDebounceTime[self->client->playerTeam] > level.time )
{
return;
}
}
*/
}
else if ( !TIMER_Done( self, "chatter" ) )
{//personal timer
return;
}
else if ( groupSpeechDebounceTime[self->client->playerTeam] > level.time )
{//for those not in group AI
//FIXME: let certain speech types interrupt others? Let closer NPCs interrupt farther away ones?
return;
}
}
if ( self->NPC->group )
{//So they don't all speak at once...
//FIXME: if they're not yet mad, they have no group, so distracting a group of them makes them all speak!
self->NPC->group->speechDebounceTime = level.time + Q_irand( 2000, 4000 );
}
else
{
TIMER_Set( self, "chatter", Q_irand( 2000, 4000 ) );
}
groupSpeechDebounceTime[self->client->playerTeam] = level.time + Q_irand( 2000, 4000 );
if ( self->NPC->blockedSpeechDebounceTime > level.time )
{
return;
}
switch( speechType )
{
case SPEECH_CHASE:
G_AddVoiceEvent( self, Q_irand(EV_CHASE1, EV_CHASE3), 2000 );
break;
case SPEECH_CONFUSED:
G_AddVoiceEvent( self, Q_irand(EV_CONFUSE1, EV_CONFUSE3), 2000 );
break;
case SPEECH_COVER:
G_AddVoiceEvent( self, Q_irand(EV_COVER1, EV_COVER5), 2000 );
break;
case SPEECH_DETECTED:
G_AddVoiceEvent( self, Q_irand(EV_DETECTED1, EV_DETECTED5), 2000 );
break;
case SPEECH_GIVEUP:
G_AddVoiceEvent( self, Q_irand(EV_GIVEUP1, EV_GIVEUP4), 2000 );
break;
case SPEECH_LOOK:
G_AddVoiceEvent( self, Q_irand(EV_LOOK1, EV_LOOK2), 2000 );
break;
case SPEECH_LOST:
G_AddVoiceEvent( self, EV_LOST1, 2000 );
break;
case SPEECH_OUTFLANK:
G_AddVoiceEvent( self, Q_irand(EV_OUTFLANK1, EV_OUTFLANK2), 2000 );
break;
case SPEECH_ESCAPING:
G_AddVoiceEvent( self, Q_irand(EV_ESCAPING1, EV_ESCAPING3), 2000 );
break;
case SPEECH_SIGHT:
G_AddVoiceEvent( self, Q_irand(EV_SIGHT1, EV_SIGHT3), 2000 );
break;
case SPEECH_SOUND:
G_AddVoiceEvent( self, Q_irand(EV_SOUND1, EV_SOUND3), 2000 );
break;
case SPEECH_SUSPICIOUS:
G_AddVoiceEvent( self, Q_irand(EV_SUSPICIOUS1, EV_SUSPICIOUS5), 2000 );
break;
case SPEECH_YELL:
G_AddVoiceEvent( self, Q_irand( EV_ANGER1, EV_ANGER3 ), 2000 );
break;
case SPEECH_PUSHED:
G_AddVoiceEvent( self, Q_irand( EV_PUSHED1, EV_PUSHED3 ), 2000 );
break;
default:
break;
}
self->NPC->blockedSpeechDebounceTime = level.time + 2000;
}
void ST_MarkToCover( gentity_t *self )
{
if ( !self || !self->NPC )
{
return;
}
self->NPC->localState = LSTATE_UNDERFIRE;
TIMER_Set( self, "attackDelay", Q_irand( 500, 2500 ) );
ST_AggressionAdjust( self, -3 );
if ( self->NPC->group && self->NPC->group->numGroup > 1 )
{
ST_Speech( self, SPEECH_COVER, 0 );//FIXME: flee sound?
}
}
void ST_StartFlee( gentity_t *self, gentity_t *enemy, vec3_t dangerPoint, int dangerLevel, int minTime, int maxTime )
{
if ( !self || !self->NPC )
{
return;
}
G_StartFlee( self, enemy, dangerPoint, dangerLevel, minTime, maxTime );
if ( self->NPC->group && self->NPC->group->numGroup > 1 )
{
ST_Speech( self, SPEECH_COVER, 0 );//FIXME: flee sound?
}
}
/*
-------------------------
NPC_ST_Pain
-------------------------
*/
void NPC_ST_Pain( gentity_t *self, gentity_t *inflictor, gentity_t *other, const vec3_t point, int damage, int mod,int hitLoc )
{
self->NPC->localState = LSTATE_UNDERFIRE;
TIMER_Set( self, "duck", -1 );
TIMER_Set( self, "hideTime", -1 );
TIMER_Set( self, "stand", 2000 );
NPC_Pain( self, inflictor, other, point, damage, mod, hitLoc );
if ( !damage && self->health > 0 )
{//FIXME: better way to know I was pushed
G_AddVoiceEvent( self, Q_irand(EV_PUSHED1, EV_PUSHED3), 2000 );
}
}
/*
-------------------------
ST_HoldPosition
-------------------------
*/
static void ST_HoldPosition( void )
{
if ( NPCInfo->squadState == SQUAD_RETREAT )
{
TIMER_Set( NPC, "flee", -level.time );
}
TIMER_Set( NPC, "verifyCP", Q_irand( 1000, 3000 ) );//don't look for another one for a few seconds
NPC_FreeCombatPoint( NPCInfo->combatPoint, qtrue );
//NPCInfo->combatPoint = -1;//???
if ( !Q3_TaskIDPending( NPC, TID_MOVE_NAV ) )
{//don't have a script waiting for me to get to my point, okay to stop trying and stand
AI_GroupUpdateSquadstates( NPCInfo->group, NPC, SQUAD_STAND_AND_SHOOT );
NPCInfo->goalEntity = NULL;
}
}
void NPC_ST_SayMovementSpeech( void )
{
if ( !NPCInfo->movementSpeech )
{
return;
}
if ( NPCInfo->group &&
NPCInfo->group->commander &&
NPCInfo->group->commander->client &&
NPCInfo->group->commander->client->NPC_class == CLASS_IMPERIAL &&
!Q_irand( 0, 3 ) )
{//imperial (commander) gives the order
ST_Speech( NPCInfo->group->commander, NPCInfo->movementSpeech, NPCInfo->movementSpeechChance );
}
else
{//really don't want to say this unless we can actually get there...
ST_Speech( NPC, NPCInfo->movementSpeech, NPCInfo->movementSpeechChance );
}
NPCInfo->movementSpeech = 0;
NPCInfo->movementSpeechChance = 0.0f;
}
void NPC_ST_StoreMovementSpeech( int speech, float chance )
{
NPCInfo->movementSpeech = speech;
NPCInfo->movementSpeechChance = chance;
}
/*
-------------------------
ST_Move
-------------------------
*/
void ST_TransferMoveGoal( gentity_t *self, gentity_t *other );
static qboolean ST_Move( void )
{
NPCInfo->combatMove = qtrue;//always doMove straight toward our goal
qboolean moved = NPC_MoveToGoal( qtrue );
if (moved==qfalse)
{
ST_HoldPosition();
}
NPC_ST_SayMovementSpeech();
return moved;
}
/*
-------------------------
NPC_ST_SleepShuffle
-------------------------
*/
static void NPC_ST_SleepShuffle( void )
{
//Play an awake script if we have one
if ( G_ActivateBehavior( NPC, BSET_AWAKE) )
{
return;
}
//Automate some movement and noise
if ( TIMER_Done( NPC, "shuffleTime" ) )
{
//TODO: Play sleeping shuffle animation
//int soundIndex = Q_irand( 0, 1 );
/*
switch ( soundIndex )
{
case 0:
G_Sound( NPC, G_SoundIndex("sound/chars/imperialsleeper1/scav4/hunh.mp3") );
break;
case 1:
G_Sound( NPC, G_SoundIndex("sound/chars/imperialsleeper3/scav4/tryingtosleep.wav") );
break;
}
*/
TIMER_Set( NPC, "shuffleTime", 4000 );
TIMER_Set( NPC, "sleepTime", 2000 );
return;
}
//They made another noise while we were stirring, see if we can see them
if ( TIMER_Done( NPC, "sleepTime" ) )
{
NPC_CheckPlayerTeamStealth();
TIMER_Set( NPC, "sleepTime", 2000 );
}
}
/*
-------------------------
NPC_ST_Sleep
-------------------------
*/
void NPC_BSST_Sleep( void )
{
int alertEvent = NPC_CheckAlertEvents( qfalse, qtrue );//only check sounds since we're alseep!
//There is an event we heard
if ( alertEvent >= 0 )
{
//See if it was enough to wake us up
if ( level.alertEvents[alertEvent].level == AEL_DISCOVERED && (NPCInfo->scriptFlags&SCF_LOOK_FOR_ENEMIES) )
{
if ( &g_entities[0] && g_entities[0].health > 0 )
{
G_SetEnemy( NPC, &g_entities[0] );
return;
}
}
//Otherwise just stir a bit
NPC_ST_SleepShuffle();
return;
}
}
/*
-------------------------
NPC_CheckEnemyStealth
-------------------------
*/
qboolean NPC_CheckEnemyStealth( gentity_t *target )
{
float target_dist, minDist = 40;//any closer than 40 and we definitely notice
//In case we aquired one some other way
if ( NPC->enemy != NULL )
return qtrue;
//Ignore notarget
if ( target->flags & FL_NOTARGET )
return qfalse;
if ( target->health <= 0 )
{
return qfalse;
}
if ( target->client->ps.weapon == WP_SABER && target->client->ps.SaberActive() && !target->client->ps.saberInFlight )
{//if target has saber in hand and activated, we wake up even sooner even if not facing him
minDist = 100;
}
target_dist = DistanceSquared( target->currentOrigin, NPC->currentOrigin );
//If the target is this close, then wake up regardless
if ( !(target->client->ps.pm_flags&PMF_DUCKED)//not ducking
&& (NPCInfo->scriptFlags&SCF_LOOK_FOR_ENEMIES)//looking for enemies
&& target_dist < (minDist*minDist) )//closer than minDist
{
G_SetEnemy( NPC, target );
NPCInfo->enemyLastSeenTime = level.time;
TIMER_Set( NPC, "attackDelay", Q_irand( 500, 2500 ) );
return qtrue;
}
float maxViewDist = MAX_VIEW_DIST;
// if ( NPCInfo->stats.visrange > maxViewDist )
{//FIXME: should we always just set maxViewDist to this?
maxViewDist = NPCInfo->stats.visrange;
}
if ( target_dist > (maxViewDist*maxViewDist) )
{//out of possible visRange
return qfalse;
}
//Check FOV first
if ( InFOV( target, NPC, NPCInfo->stats.hfov, NPCInfo->stats.vfov ) == qfalse )
return qfalse;
qboolean clearLOS = ( target->client->ps.leanofs ) ? NPC_ClearLOS( target->client->renderInfo.eyePoint ) : NPC_ClearLOS( target );
//Now check for clear line of vision
if ( clearLOS )
{
if ( target->client->NPC_class == CLASS_ATST )
{//can't miss 'em!
G_SetEnemy( NPC, target );
TIMER_Set( NPC, "attackDelay", Q_irand( 500, 2500 ) );
return qtrue;
}
vec3_t targ_org = {target->currentOrigin[0],target->currentOrigin[1],target->currentOrigin[2]+target->maxs[2]-4};
float hAngle_perc = NPC_GetHFOVPercentage( targ_org, NPC->client->renderInfo.eyePoint, NPC->client->renderInfo.eyeAngles, NPCInfo->stats.hfov );
float vAngle_perc = NPC_GetVFOVPercentage( targ_org, NPC->client->renderInfo.eyePoint, NPC->client->renderInfo.eyeAngles, NPCInfo->stats.vfov );
//Scale them vertically some, and horizontally pretty harshly
vAngle_perc *= vAngle_perc;//( vAngle_perc * vAngle_perc );
hAngle_perc *= ( hAngle_perc * hAngle_perc );
//Cap our vertical vision severely
//if ( vAngle_perc <= 0.3f ) // was 0.5f
// return qfalse;
//Assess the player's current status
target_dist = Distance( target->currentOrigin, NPC->currentOrigin );
float target_speed = VectorLength( target->client->ps.velocity );
int target_crouching = ( target->client->usercmd.upmove < 0 );
float dist_rating = ( target_dist / maxViewDist );
float speed_rating = ( target_speed / MAX_VIEW_SPEED );
float turning_rating = AngleDelta( target->client->ps.viewangles[PITCH], target->lastAngles[PITCH] )/180.0f + AngleDelta( target->client->ps.viewangles[YAW], target->lastAngles[YAW] )/180.0f;
float light_level = ( target->lightLevel / MAX_LIGHT_INTENSITY );
float FOV_perc = 1.0f - ( hAngle_perc + vAngle_perc ) * 0.5f; //FIXME: Dunno about the average...
float vis_rating = 0.0f;
//Too dark
if ( light_level < MIN_LIGHT_THRESHOLD )
return qfalse;
//Too close?
if ( dist_rating < DISTANCE_THRESHOLD )
{
G_SetEnemy( NPC, target );
TIMER_Set( NPC, "attackDelay", Q_irand( 500, 2500 ) );
return qtrue;
}
//Out of range
if ( dist_rating > 1.0f )
return qfalse;
//Cap our speed checks
if ( speed_rating > 1.0f )
speed_rating = 1.0f;
//Calculate the distance, fov and light influences
//...Visibilty linearly wanes over distance
float dist_influence = DISTANCE_SCALE * ( ( 1.0f - dist_rating ) );
//...As the percentage out of the FOV increases, straight perception suffers on an exponential scale
float fov_influence = FOV_SCALE * ( 1.0f - FOV_perc );
//...Lack of light hides, abundance of light exposes
float light_influence = ( light_level - 0.5f ) * LIGHT_SCALE;
//Calculate our base rating
float target_rating = dist_influence + fov_influence + light_influence;
//Now award any final bonuses to this number
int contents = gi.pointcontents( targ_org, target->s.number );
if ( contents&CONTENTS_WATER )
{
int myContents = gi.pointcontents( NPC->client->renderInfo.eyePoint, NPC->s.number );
if ( !(myContents&CONTENTS_WATER) )
{//I'm not in water
if ( NPC->client->NPC_class == CLASS_SWAMPTROOPER )
{//these guys can see in in/through water pretty well
vis_rating = 0.10f;//10% bonus
}
else
{
vis_rating = 0.35f;//35% bonus
}
}
else
{//else, if we're both in water
if ( NPC->client->NPC_class == CLASS_SWAMPTROOPER )
{//I can see him just fine
}
else
{
vis_rating = 0.15f;//15% bonus
}
}
}
else
{//not in water
if ( contents&CONTENTS_FOG )
{
vis_rating = 0.15f;//15% bonus
}
}
target_rating *= (1.0f - vis_rating);
//...Motion draws the eye quickly
target_rating += speed_rating * SPEED_SCALE;
target_rating += turning_rating * TURNING_SCALE;
//FIXME: check to see if they're animating, too? But can we do something as simple as frame != oldframe?
//...Smaller targets are harder to indentify
if ( target_crouching )
{
target_rating *= 0.9f; //10% bonus
}
//If he's violated the threshold, then realize him
//float difficulty_scale = 1.0f + (2.0f-g_spskill->value);//if playing on easy, 20% harder to be seen...?
float realize, cautious;
if ( NPC->client->NPC_class == CLASS_SWAMPTROOPER )
{//swamptroopers can see much better
realize = (float)CAUTIOUS_THRESHOLD/**difficulty_scale*/;
cautious = (float)CAUTIOUS_THRESHOLD * 0.75f/**difficulty_scale*/;
}
else
{
realize = (float)REALIZE_THRESHOLD/**difficulty_scale*/;
cautious = (float)CAUTIOUS_THRESHOLD * 0.75f/**difficulty_scale*/;
}
if ( target_rating > realize && (NPCInfo->scriptFlags&SCF_LOOK_FOR_ENEMIES) )
{
G_SetEnemy( NPC, target );
NPCInfo->enemyLastSeenTime = level.time;
TIMER_Set( NPC, "attackDelay", Q_irand( 500, 2500 ) );
return qtrue;
}
//If he's above the caution threshold, then realize him in a few seconds unless he moves to cover
if ( target_rating > cautious && !(NPCInfo->scriptFlags&SCF_IGNORE_ALERTS) )
{//FIXME: ambushing guys should never talk
if ( TIMER_Done( NPC, "enemyLastVisible" ) )
{//If we haven't already, start the counter
int lookTime = Q_irand( 4500, 8500 );
//NPCInfo->timeEnemyLastVisible = level.time + 2000;
TIMER_Set( NPC, "enemyLastVisible", lookTime );
//TODO: Play a sound along the lines of, "Huh? What was that?"
ST_Speech( NPC, SPEECH_SIGHT, 0 );
NPC_TempLookTarget( NPC, target->s.number, lookTime, lookTime );
//FIXME: set desired yaw and pitch towards this guy?
}
else if ( TIMER_Get( NPC, "enemyLastVisible" ) <= level.time + 500 && (NPCInfo->scriptFlags&SCF_LOOK_FOR_ENEMIES) ) //FIXME: Is this reliable?
{
if ( NPCInfo->rank < RANK_LT && !Q_irand( 0, 2 ) )
{
int interrogateTime = Q_irand( 2000, 4000 );
ST_Speech( NPC, SPEECH_SUSPICIOUS, 0 );
TIMER_Set( NPC, "interrogating", interrogateTime );
G_SetEnemy( NPC, target );
NPCInfo->enemyLastSeenTime = level.time;
TIMER_Set( NPC, "attackDelay", interrogateTime );
TIMER_Set( NPC, "stand", interrogateTime );
}
else
{
G_SetEnemy( NPC, target );
NPCInfo->enemyLastSeenTime = level.time;
//FIXME: ambush guys (like those popping out of water) shouldn't delay...
TIMER_Set( NPC, "attackDelay", Q_irand( 500, 2500 ) );
TIMER_Set( NPC, "stand", Q_irand( 500, 2500 ) );
}
return qtrue;
}
return qfalse;
}
}
return qfalse;
}
qboolean NPC_CheckPlayerTeamStealth( void )
{
/*
//NOTENOTE: For now, all stealh checks go against the player, since
// he is the main focus. Squad members and rivals do not
// fall into this category and will be ignored.
NPC_CheckEnemyStealth( &g_entities[0] ); //Change this pointer to assess other entities
*/
gentity_t *enemy;
for ( int i = 0; i < ENTITYNUM_WORLD; i++ )
{
if(!PInUse(i))
continue;
enemy = &g_entities[i];
if ( enemy
&& enemy->client
&& NPC_ValidEnemy( enemy ) )
{
if ( NPC_CheckEnemyStealth( enemy ) ) //Change this pointer to assess other entities
{
return qtrue;
}
}
}
return qfalse;
}
qboolean NPC_CheckEnemiesInSpotlight( void )
{
gentity_t *entityList[MAX_GENTITIES];
gentity_t *enemy, *suspect = NULL;
int i, numListedEntities;
vec3_t mins, maxs;
for ( i = 0 ; i < 3 ; i++ )
{
mins[i] = NPC->client->renderInfo.eyePoint[i] - NPC->speed;
maxs[i] = NPC->client->renderInfo.eyePoint[i] + NPC->speed;
}
numListedEntities = gi.EntitiesInBox( mins, maxs, entityList, MAX_GENTITIES );
for ( i = 0; i < numListedEntities; i++ )
{
if(!PInUse(i))
continue;
enemy = entityList[i];
if ( enemy && enemy->client && NPC_ValidEnemy( enemy ) && enemy->client->playerTeam == NPC->client->enemyTeam )
{//valid ent & client, valid enemy, on the target team
//check to see if they're in my FOV
if ( InFOV( enemy->currentOrigin, NPC->client->renderInfo.eyePoint, NPC->client->renderInfo.eyeAngles, NPCInfo->stats.hfov, NPCInfo->stats.vfov ) )
{//in my cone
//check to see that they're close enough
if ( DistanceSquared( NPC->client->renderInfo.eyePoint, enemy->currentOrigin )-256/*fudge factor: 16 squared*/ <= NPC->speed*NPC->speed )
{//within range
//check to see if we have a clear trace to them
if ( G_ClearLOS( NPC, enemy ) )
{//clear LOS
//make sure their light level is at least my beam's brightness
//FIXME: HOW?
//enemy->lightLevel / MAX_LIGHT_INTENSITY
//good enough, take him!
//FIXME: pick closest one?
//FIXME: have the graduated noticing like other NPCs? (based on distance, FOV dot, etc...)
G_SetEnemy( NPC, enemy );
TIMER_Set( NPC, "attackDelay", Q_irand( 500, 2500 ) );
return qtrue;
}
}
}
if ( InFOV( enemy->currentOrigin, NPC->client->renderInfo.eyePoint, NPC->client->renderInfo.eyeAngles, 90, NPCInfo->stats.vfov*3 ) )
{//one to look at if we don't get an enemy
if ( G_ClearLOS( NPC, enemy ) )
{//clear LOS
if ( suspect == NULL || DistanceSquared( NPC->client->renderInfo.eyePoint, enemy->currentOrigin ) < DistanceSquared( NPC->client->renderInfo.eyePoint, suspect->currentOrigin ) )
{//remember him
suspect = enemy;
}
}
}
}
}
if ( suspect && Q_flrand( 0, NPCInfo->stats.visrange*NPCInfo->stats.visrange ) > DistanceSquared( NPC->client->renderInfo.eyePoint, suspect->currentOrigin ) )
{//hey! who's that?
if ( TIMER_Done( NPC, "enemyLastVisible" ) )
{//If we haven't already, start the counter
int lookTime = Q_irand( 4500, 8500 );
//NPCInfo->timeEnemyLastVisible = level.time + 2000;
TIMER_Set( NPC, "enemyLastVisible", lookTime );
//TODO: Play a sound along the lines of, "Huh? What was that?"
ST_Speech( NPC, SPEECH_SIGHT, 0 );
//set desired yaw and pitch towards this guy?
//FIXME: this is permanent, they will never look away... *sigh*
NPC_FacePosition( suspect->currentOrigin, qtrue );
//FIXME: they still need some sort of eye/head tag/bone that can turn?
//NPC_TempLookTarget( NPC, suspect->s.number, lookTime, lookTime );
}
else if ( TIMER_Get( NPC, "enemyLastVisible" ) <= level.time + 500
&& (NPCInfo->scriptFlags&SCF_LOOK_FOR_ENEMIES) ) //FIXME: Is this reliable?
{
if ( !Q_irand( 0, 2 ) )
{
int interrogateTime = Q_irand( 2000, 4000 );
ST_Speech( NPC, SPEECH_SUSPICIOUS, 0 );
TIMER_Set( NPC, "interrogating", interrogateTime );
//G_SetEnemy( NPC, target );
//NPCInfo->enemyLastSeenTime = level.time;
//TIMER_Set( NPC, "attackDelay", interrogateTime );
//TIMER_Set( NPC, "stand", interrogateTime );
//set desired yaw and pitch towards this guy?
//FIXME: this is permanent, they will never look away... *sigh*
NPC_FacePosition( suspect->currentOrigin, qtrue );
//FIXME: they still need some sort of eye/head tag/bone that can turn?
//NPC_TempLookTarget( NPC, suspect->s.number, interrogateTime, interrogateTime );
}
}
}
return qfalse;
}
/*
-------------------------
NPC_ST_InvestigateEvent
-------------------------
*/
#define MAX_CHECK_THRESHOLD 1
static qboolean NPC_ST_InvestigateEvent( int eventID, bool extraSuspicious )
{
//If they've given themselves away, just take them as an enemy
if ( NPCInfo->confusionTime < level.time )
{
if ( level.alertEvents[eventID].level == AEL_DISCOVERED && (NPCInfo->scriptFlags&SCF_LOOK_FOR_ENEMIES) )
{
//NPCInfo->lastAlertID = level.alertEvents[eventID].ID;
if ( !level.alertEvents[eventID].owner ||
!level.alertEvents[eventID].owner->client ||
level.alertEvents[eventID].owner->health <= 0 ||
level.alertEvents[eventID].owner->client->playerTeam != NPC->client->enemyTeam )
{//not an enemy
return qfalse;
}
//FIXME: what if can't actually see enemy, don't know where he is... should we make them just become very alert and start looking for him? Or just let combat AI handle this... (act as if you lost him)
//ST_Speech( NPC, SPEECH_CHARGE, 0 );
G_SetEnemy( NPC, level.alertEvents[eventID].owner );
NPCInfo->enemyLastSeenTime = level.time;
TIMER_Set( NPC, "attackDelay", Q_irand( 500, 2500 ) );
if ( level.alertEvents[eventID].type == AET_SOUND )
{//heard him, didn't see him, stick for a bit
TIMER_Set( NPC, "roamTime", Q_irand( 500, 2500 ) );
}
return qtrue;
}
}
//don't look at the same alert twice
/*
if ( level.alertEvents[eventID].ID == NPCInfo->lastAlertID )
{
return qfalse;
}
NPCInfo->lastAlertID = level.alertEvents[eventID].ID;
*/
//Must be ready to take another sound event
/*
if ( NPCInfo->investigateSoundDebounceTime > level.time )
{
return qfalse;
}
*/
if ( level.alertEvents[eventID].type == AET_SIGHT )
{//sight alert, check the light level
if ( level.alertEvents[eventID].light < Q_irand( ST_MIN_LIGHT_THRESHOLD, ST_MAX_LIGHT_THRESHOLD ) )
{//below my threshhold of potentially seeing
return qfalse;
}
}
//Save the position for movement (if necessary)
VectorCopy( level.alertEvents[eventID].position, NPCInfo->investigateGoal );
//First awareness of it
NPCInfo->investigateCount += ( extraSuspicious ) ? 2 : 1;
//Clamp the value
if ( NPCInfo->investigateCount > 4 )
NPCInfo->investigateCount = 4;
//See if we should walk over and investigate
if ( level.alertEvents[eventID].level > AEL_MINOR && NPCInfo->investigateCount > 1 && (NPCInfo->scriptFlags&SCF_CHASE_ENEMIES) )
{
//make it so they can walk right to this point and look at it rather than having to use combatPoints
if ( G_ExpandPointToBBox( NPCInfo->investigateGoal, NPC->mins, NPC->maxs, NPC->s.number, ((NPC->clipmask&~CONTENTS_BODY)|CONTENTS_BOTCLIP) ) )
{//we were able to doMove the investigateGoal to a point in which our bbox would fit
//drop the goal to the ground so we can get at it
vec3_t end;
trace_t trace;
VectorCopy( NPCInfo->investigateGoal, end );
end[2] -= 512;//FIXME: not always right? What if it's even higher, somehow?
gi.trace( &trace, NPCInfo->investigateGoal, NPC->mins, NPC->maxs, end, ENTITYNUM_NONE, ((NPC->clipmask&~CONTENTS_BODY)|CONTENTS_BOTCLIP), (EG2_Collision)0, 0 );
if ( trace.fraction >= 1.0f )
{//too high to even bother
//FIXME: look at them???
}
else
{
VectorCopy( trace.endpos, NPCInfo->investigateGoal );
NPC_SetMoveGoal( NPC, NPCInfo->investigateGoal, 16, qtrue );
NPCInfo->localState = LSTATE_INVESTIGATE;
}
}
else
{
int id = NPC_FindCombatPoint( NPCInfo->investigateGoal, NPCInfo->investigateGoal, NPCInfo->investigateGoal, CP_INVESTIGATE|CP_HAS_ROUTE, 0 );
if ( id != -1 )
{
NPC_SetMoveGoal( NPC, level.combatPoints[id].origin, 16, qtrue, id );
NPCInfo->localState = LSTATE_INVESTIGATE;
}
}
//Say something
//FIXME: only if have others in group... these should be responses?
if ( NPCInfo->investigateDebounceTime+NPCInfo->pauseTime > level.time )
{//was already investigating
if ( NPCInfo->group &&
NPCInfo->group->commander &&
NPCInfo->group->commander->client &&
NPCInfo->group->commander->client->NPC_class == CLASS_IMPERIAL &&
!Q_irand( 0, 3 ) )
{
ST_Speech( NPCInfo->group->commander, SPEECH_LOOK, 0 );//FIXME: "I'll go check it out" type sounds
}
else
{
ST_Speech( NPC, SPEECH_LOOK, 0 );//FIXME: "I'll go check it out" type sounds
}
}
else
{
if ( level.alertEvents[eventID].type == AET_SIGHT )
{
ST_Speech( NPC, SPEECH_SIGHT, 0 );
}
else if ( level.alertEvents[eventID].type == AET_SOUND )
{
ST_Speech( NPC, SPEECH_SOUND, 0 );
}
}
//Setup the debounce info
NPCInfo->investigateDebounceTime = NPCInfo->investigateCount * 5000;
NPCInfo->investigateSoundDebounceTime = level.time + 2000;
NPCInfo->pauseTime = level.time;
}
else
{//just look?
//Say something
if ( level.alertEvents[eventID].type == AET_SIGHT )
{
ST_Speech( NPC, SPEECH_SIGHT, 0 );
}
else if ( level.alertEvents[eventID].type == AET_SOUND )
{
ST_Speech( NPC, SPEECH_SOUND, 0 );
}
//Setup the debounce info
NPCInfo->investigateDebounceTime = NPCInfo->investigateCount * 1000;
NPCInfo->investigateSoundDebounceTime = level.time + 1000;
NPCInfo->pauseTime = level.time;
VectorCopy( level.alertEvents[eventID].position, NPCInfo->investigateGoal );
if ( NPC->client->NPC_class == CLASS_ROCKETTROOPER
&& !RT_Flying( NPC ) )
{
//if ( !Q_irand( 0, 2 ) )
{//look around
NPC_SetAnim( NPC, SETANIM_BOTH, BOTH_GUARD_LOOKAROUND1, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD );
}
}
}
if ( level.alertEvents[eventID].level >= AEL_DANGER )
{
NPCInfo->investigateDebounceTime = Q_irand( 500, 2500 );
}
//Start investigating
NPCInfo->tempBehavior = BS_INVESTIGATE;
return qtrue;
}
/*
-------------------------
ST_OffsetLook
-------------------------
*/
static void ST_OffsetLook( float offset, vec3_t out )
{
vec3_t angles, forward, temp;
GetAnglesForDirection( NPC->currentOrigin, NPCInfo->investigateGoal, angles );
angles[YAW] += offset;
AngleVectors( angles, forward, NULL, NULL );
VectorMA( NPC->currentOrigin, 64, forward, out );
CalcEntitySpot( NPC, SPOT_HEAD, temp );
out[2] = temp[2];
}
/*
-------------------------
ST_LookAround
-------------------------
*/
static void ST_LookAround( void )
{
vec3_t lookPos;
float perc = (float) ( level.time - NPCInfo->pauseTime ) / (float) NPCInfo->investigateDebounceTime;
//Keep looking at the spot
if ( perc < 0.25 )
{
VectorCopy( NPCInfo->investigateGoal, lookPos );
}
else if ( perc < 0.5f ) //Look up but straight ahead
{
ST_OffsetLook( 0.0f, lookPos );
}
else if ( perc < 0.75f ) //Look right
{
ST_OffsetLook( 45.0f, lookPos );
}
else //Look left
{
ST_OffsetLook( -45.0f, lookPos );
}
NPC_FacePosition( lookPos );
}
/*
-------------------------
NPC_BSST_Investigate
-------------------------
*/
void NPC_BSST_Investigate( void )
{
//get group- mainly for group speech debouncing, but may use for group scouting/investigating AI, too
AI_GetGroup( NPC );
if( NPCInfo->scriptFlags & SCF_FIRE_WEAPON )
{
WeaponThink( qtrue );
}
if ( NPCInfo->confusionTime < level.time )
{
if ( NPCInfo->scriptFlags&SCF_LOOK_FOR_ENEMIES )
{
//Look for an enemy
if ( NPC_CheckPlayerTeamStealth() )
{
//NPCInfo->behaviorState = BS_HUNT_AND_KILL;//should be auto now
ST_Speech( NPC, SPEECH_DETECTED, 0 );
NPCInfo->tempBehavior = BS_DEFAULT;
NPC_UpdateAngles( qtrue, qtrue );
return;
}
}
}
if ( !(NPCInfo->scriptFlags&SCF_IGNORE_ALERTS) )
{
int alertEvent = NPC_CheckAlertEvents( qtrue, qtrue, NPCInfo->lastAlertID );
//There is an event to look at
if ( alertEvent >= 0 )
{
if ( NPCInfo->confusionTime < level.time )
{
if ( NPC_CheckForDanger( alertEvent ) )
{//running like hell
ST_Speech( NPC, SPEECH_COVER, 0 );//FIXME: flee sound?
return;
}
}
//if ( level.alertEvents[alertEvent].ID != NPCInfo->lastAlertID )
{
NPC_ST_InvestigateEvent( alertEvent, qtrue );
}
}
}
//If we're done looking, then just return to what we were doing
if ( ( NPCInfo->investigateDebounceTime + NPCInfo->pauseTime ) < level.time )
{
NPCInfo->tempBehavior = BS_DEFAULT;
NPCInfo->goalEntity = UpdateGoal();
NPC_UpdateAngles( qtrue, qtrue );
//Say something
ST_Speech( NPC, SPEECH_GIVEUP, 0 );
return;
}
//FIXME: else, look for new alerts
//See if we're searching for the noise's origin
if ( NPCInfo->localState == LSTATE_INVESTIGATE && (NPCInfo->goalEntity!=NULL) )
{
//See if we're there
if ( !STEER::Reached(NPC, NPCInfo->goalEntity, 32, FlyingCreature(NPC) != qfalse) )
{
ucmd.buttons |= BUTTON_WALKING;
//Try and doMove there
if ( NPC_MoveToGoal( qtrue ) )
{
//Bump our times
NPCInfo->investigateDebounceTime = NPCInfo->investigateCount * 5000;
NPCInfo->pauseTime = level.time;
NPC_UpdateAngles( qtrue, qtrue );
return;
}
}
//Otherwise we're done or have given up
//Say something
//ST_Speech( NPC, SPEECH_LOOK, 0.33f );
NPCInfo->localState = LSTATE_NONE;
}
//Look around
ST_LookAround();
}
/*
-------------------------
NPC_BSST_Patrol
-------------------------
*/
void NPC_BSST_Patrol( void )
{//FIXME: pick up on bodies of dead buddies?
//Not a scriptflag, but...
if ( NPC->client->NPC_class == CLASS_ROCKETTROOPER && (NPC->client->ps.eFlags&EF_SPOTLIGHT) )
{//using spotlight search mode
vec3_t eyeFwd, end, mins={-2,-2,-2}, maxs={2,2,2};
trace_t trace;
AngleVectors( NPC->client->renderInfo.eyeAngles, eyeFwd, NULL, NULL );
VectorMA( NPC->client->renderInfo.eyePoint, NPCInfo->stats.visrange, eyeFwd, end );
//get server-side trace impact point
gi.trace( &trace, NPC->client->renderInfo.eyePoint, mins, maxs, end, NPC->s.number, MASK_OPAQUE|CONTENTS_BODY|CONTENTS_CORPSE, (EG2_Collision)0, 0 );
NPC->speed = (trace.fraction*NPCInfo->stats.visrange);
if ( NPCInfo->scriptFlags&SCF_LOOK_FOR_ENEMIES )
{
//FIXME: do a FOV cone check, then a trace
if ( trace.entityNum < ENTITYNUM_WORLD )
{//hit something
//try cheap check first
gentity_t *enemy = &g_entities[trace.entityNum];
if ( enemy && enemy->client && NPC_ValidEnemy( enemy ) && enemy->client->playerTeam == NPC->client->enemyTeam )
{
G_SetEnemy( NPC, enemy );
TIMER_Set( NPC, "attackDelay", Q_irand( 500, 2500 ) );
//NPCInfo->behaviorState = BS_HUNT_AND_KILL;//should be auto now
//NPC_AngerSound();
NPC_UpdateAngles( qtrue, qtrue );
return;
}
}
//FIXME: maybe do a quick check of ents within the spotlight's radius?
//hmmm, look around
if ( NPC_CheckEnemiesInSpotlight() )
{
//NPCInfo->behaviorState = BS_HUNT_AND_KILL;//should be auto now
//NPC_AngerSound();
NPC_UpdateAngles( qtrue, qtrue );
return;
}
}
}
else
{
//get group- mainly for group speech debouncing, but may use for group scouting/investigating AI, too
AI_GetGroup( NPC );
if ( NPCInfo->confusionTime < level.time )
{
//Look for any enemies
if ( NPCInfo->scriptFlags&SCF_LOOK_FOR_ENEMIES )
{
if ( NPC_CheckPlayerTeamStealth() )
{
//NPCInfo->behaviorState = BS_HUNT_AND_KILL;//should be auto now
//NPC_AngerSound();
NPC_UpdateAngles( qtrue, qtrue );
return;
}
}
}
}
if ( !(NPCInfo->scriptFlags&SCF_IGNORE_ALERTS) )
{
int alertEvent = NPC_CheckAlertEvents( qtrue, qtrue );
//There is an event to look at
if ( alertEvent >= 0 )
{
if ( NPC_CheckForDanger( alertEvent ) )
{//going to run?
ST_Speech( NPC, SPEECH_COVER, 0 );
return;
}
else if (NPC->client->NPC_class==CLASS_BOBAFETT)
{
//NPCInfo->lastAlertID = level.alertEvents[eventID].ID;
if ( !level.alertEvents[alertEvent].owner ||
!level.alertEvents[alertEvent].owner->client ||
level.alertEvents[alertEvent].owner->health <= 0 ||
level.alertEvents[alertEvent].owner->client->playerTeam != NPC->client->enemyTeam )
{//not an enemy
return;
}
//FIXME: what if can't actually see enemy, don't know where he is... should we make them just become very alert and start looking for him? Or just let combat AI handle this... (act as if you lost him)
//ST_Speech( NPC, SPEECH_CHARGE, 0 );
G_SetEnemy( NPC, level.alertEvents[alertEvent].owner );
NPCInfo->enemyLastSeenTime = level.time;
TIMER_Set( NPC, "attackDelay", Q_irand( 500, 2500 ) );
return;
}
else if ( NPC_ST_InvestigateEvent( alertEvent, qfalse ) )
{//actually going to investigate it
NPC_UpdateAngles( qtrue, qtrue );
return;
}
}
}
//If we have somewhere to go, then do that
if ( UpdateGoal() )
{
ucmd.buttons |= BUTTON_WALKING;
//ST_Move( NPCInfo->goalEntity );
NPC_MoveToGoal( qtrue );
}
else// if ( !(NPCInfo->scriptFlags&SCF_IGNORE_ALERTS) )
{
if ( NPC->client->NPC_class != CLASS_IMPERIAL && NPC->client->NPC_class != CLASS_IMPWORKER )
{//imperials do not look around
if ( TIMER_Done( NPC, "enemyLastVisible" ) )
{//nothing suspicious, look around
if ( !Q_irand( 0, 30 ) )
{
NPCInfo->desiredYaw = NPC->s.angles[1] + Q_irand( -90, 90 );
}
if ( !Q_irand( 0, 30 ) )
{
NPCInfo->desiredPitch = Q_irand( -20, 20 );
}
}
}
}
NPC_UpdateAngles( qtrue, qtrue );
//TEMP hack for Imperial stand anim
if ( NPC->client->NPC_class == CLASS_IMPERIAL
|| NPC->client->NPC_class == CLASS_IMPWORKER )
{//hack
if ( NPC->client->ps.weapon != WP_CONCUSSION )
{//not Rax
if ( ucmd.forwardmove || ucmd.rightmove || ucmd.upmove )
{//moving
if( (!NPC->client->ps.torsoAnimTimer) || (NPC->client->ps.torsoAnim == BOTH_STAND4) )
{
if ( (ucmd.buttons&BUTTON_WALKING) && !(NPCInfo->scriptFlags&SCF_RUNNING) )
{//not running, only set upper anim
// No longer overrides scripted anims
NPC_SetAnim( NPC, SETANIM_TORSO, BOTH_STAND4, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD );
NPC->client->ps.torsoAnimTimer = 200;
}
}
}
else
{//standing still, set both torso and legs anim
// No longer overrides scripted anims
if( ( !NPC->client->ps.torsoAnimTimer || (NPC->client->ps.torsoAnim == BOTH_STAND4) ) &&
( !NPC->client->ps.legsAnimTimer || (NPC->client->ps.legsAnim == BOTH_STAND4) ) )
{
NPC_SetAnim( NPC, SETANIM_BOTH, BOTH_STAND4, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD );
NPC->client->ps.torsoAnimTimer = NPC->client->ps.legsAnimTimer = 200;
}
}
//FIXME: this is a disgusting hack that is supposed to make the Imperials start with their weapon holstered- need a better way
if ( NPC->client->ps.weapon != WP_NONE )
{
ChangeWeapon( NPC, WP_NONE );
NPC->client->ps.weapon = WP_NONE;
NPC->client->ps.weaponstate = WEAPON_READY;
G_RemoveWeaponModels( NPC );
}
}
}
}
/*
-------------------------
NPC_BSST_Idle
-------------------------
*/
/*
void NPC_BSST_Idle( void )
{
int alertEvent = NPC_CheckAlertEvents( qtrue, qtrue );
//There is an event to look at
if ( alertEvent >= 0 )
{
NPC_ST_InvestigateEvent( alertEvent, qfalse );
NPC_UpdateAngles( qtrue, qtrue );
return;
}
TIMER_Set( NPC, "roamTime", 2000 + Q_irand( 1000, 2000 ) );
NPC_UpdateAngles( qtrue, qtrue );
}
*/
/*
-------------------------
ST_CheckMoveState
-------------------------
*/
static void ST_CheckMoveState( void )
{
if ( Q3_TaskIDPending( NPC, TID_MOVE_NAV ) )
{//moving toward a goal that a script is waiting on, so don't stop for anything!
doMove = qtrue;
}
else if ( NPC->client->NPC_class == CLASS_ROCKETTROOPER
&& NPC->client->ps.groundEntityNum == ENTITYNUM_NONE )
{//no squad stuff
return;
}
// else if ( NPC->NPC->scriptFlags&SCF_NO_GROUPS )
{
doMove = qtrue;
}
//See if we're a scout
//See if we're moving towards a goal, not the enemy
if ( ( NPCInfo->goalEntity != NPC->enemy ) && ( NPCInfo->goalEntity != NULL ) )
{
//Did we make it?
if ( STEER::Reached(NPC, NPCInfo->goalEntity, 16, !!FlyingCreature(NPC)) ||
(enemyLOS && (NPCInfo->aiFlags&NPCAI_STOP_AT_LOS) && !Q3_TaskIDPending(NPC, TID_MOVE_NAV))
)
{//either hit our navgoal or our navgoal was not a crucial (scripted) one (maybe a combat point) and we're scouting and found our enemy
int newSquadState = SQUAD_STAND_AND_SHOOT;
//we got where we wanted to go, set timers based on why we were running
switch ( NPCInfo->squadState )
{
case SQUAD_RETREAT://was running away
//done fleeing, obviously
TIMER_Set( NPC, "duck", (NPC->max_health - NPC->health) * 100 );
TIMER_Set( NPC, "hideTime", Q_irand( 3000, 7000 ) );
TIMER_Set( NPC, "flee", -level.time );
newSquadState = SQUAD_COVER;
break;
case SQUAD_TRANSITION://was heading for a combat point
TIMER_Set( NPC, "hideTime", Q_irand( 2000, 4000 ) );
break;
case SQUAD_SCOUT://was running after player
break;
default:
break;
}
AI_GroupUpdateSquadstates( NPCInfo->group, NPC, newSquadState );
NPC_ReachedGoal();
//don't attack right away
TIMER_Set( NPC, "attackDelay", Q_irand( 250, 500 ) ); //FIXME: Slant for difficulty levels
//don't do something else just yet
// THIS IS THE ONE TRUE PLACE WHERE ROAM TIME IS SET
TIMER_Set( NPC, "roamTime", Q_irand( 8000, 15000 ) );//Q_irand( 1000, 4000 ) );
if (Q_irand(0, 3)==0)
{
TIMER_Set( NPC, "duck", Q_irand(5000, 10000) ); // just reached our goal, chance of ducking now
}
return;
}
//keep going, hold of roamTimer until we get there
TIMER_Set( NPC, "roamTime", Q_irand( 8000, 9000 ) );
}
}
void ST_ResolveBlockedShot( int hit )
{
int stuckTime;
//figure out how long we intend to stand here, max
if ( TIMER_Get( NPC, "roamTime" ) > TIMER_Get( NPC, "stick" ) )
{
stuckTime = TIMER_Get( NPC, "roamTime" )-level.time;
}
else
{
stuckTime = TIMER_Get( NPC, "stick" )-level.time;
}
if ( TIMER_Done( NPC, "duck" ) )
{//we're not ducking
if ( AI_GroupContainsEntNum( NPCInfo->group, hit ) )
{
gentity_t *member = &g_entities[hit];
if ( TIMER_Done( member, "duck" ) )
{//they aren't ducking
if ( TIMER_Done( member, "stand" ) )
{//they're not being forced to stand
//tell them to duck at least as long as I'm not moving
TIMER_Set( member, "duck", stuckTime ); // tell my friend to duck so I can shoot over his head
return;
}
}
}
}
else
{//maybe we should stand
if ( TIMER_Done( NPC, "stand" ) )
{//stand for as long as we'll be here
TIMER_Set( NPC, "stand", stuckTime );
return;
}
}
//Hmm, can't resolve this by telling them to duck or telling me to stand
//We need to doMove!
TIMER_Set( NPC, "roamTime", -1 );
TIMER_Set( NPC, "stick", -1 );
TIMER_Set( NPC, "duck", -1 );
TIMER_Set( NPC, "attakDelay", Q_irand( 1000, 3000 ) );
}
/*
-------------------------
ST_CheckFireState
-------------------------
*/
static void ST_CheckFireState( void )
{
if ( enemyCS )
{//if have a clear shot, always try
return;
}
if ( NPCInfo->squadState == SQUAD_RETREAT || NPCInfo->squadState == SQUAD_TRANSITION || NPCInfo->squadState == SQUAD_SCOUT )
{//runners never try to fire at the last pos
return;
}
if ( !VectorCompare( NPC->client->ps.velocity, vec3_origin ) )
{//if moving at all, don't do this
return;
}
//See if we should continue to fire on their last position
//!TIMER_Done( NPC, "stick" ) ||
if ( !hitAlly //we're not going to hit an ally
&& enemyInFOV //enemy is in our FOV //FIXME: or we don't have a clear LOS?
&& NPCInfo->enemyLastSeenTime > 0 //we've seen the enemy
&& NPCInfo->group //have a group
&& (NPCInfo->group->numState[SQUAD_RETREAT]>0||NPCInfo->group->numState[SQUAD_TRANSITION]>0||NPCInfo->group->numState[SQUAD_SCOUT]>0) )//laying down covering fire
{
if ( level.time - NPCInfo->enemyLastSeenTime < 10000 &&//we have seem the enemy in the last 10 seconds
(!NPCInfo->group || level.time - NPCInfo->group->lastSeenEnemyTime < 10000 ))//we are not in a group or the group has seen the enemy in the last 10 seconds
{
if ( !Q_irand( 0, 10 ) )
{
//Fire on the last known position
vec3_t muzzle, dir, angles;
qboolean tooClose = qfalse;
qboolean tooFar = qfalse;
CalcEntitySpot( NPC, SPOT_HEAD, muzzle );
if ( VectorCompare( impactPos, vec3_origin ) )
{//never checked ShotEntity this frame, so must do a trace...
trace_t tr;
//vec3_t mins = {-2,-2,-2}, maxs = {2,2,2};
vec3_t forward, end;
AngleVectors( NPC->client->ps.viewangles, forward, NULL, NULL );
VectorMA( muzzle, 8192, forward, end );
gi.trace( &tr, muzzle, vec3_origin, vec3_origin, end, NPC->s.number, MASK_SHOT, (EG2_Collision)0, 0 );
VectorCopy( tr.endpos, impactPos );
}
//see if impact would be too close to me
float distThreshold = 16384/*128*128*/;//default
switch ( NPC->s.weapon )
{
case WP_ROCKET_LAUNCHER:
case WP_FLECHETTE:
case WP_THERMAL:
case WP_TRIP_MINE:
case WP_DET_PACK:
distThreshold = 65536/*256*256*/;
break;
case WP_REPEATER:
if ( NPCInfo->scriptFlags&SCF_ALT_FIRE )
{
distThreshold = 65536/*256*256*/;
}
break;
case WP_CONCUSSION:
if ( !(NPCInfo->scriptFlags&SCF_ALT_FIRE) )
{
distThreshold = 65536/*256*256*/;
}
break;
default:
break;
}
float dist = DistanceSquared( impactPos, muzzle );
if ( dist < distThreshold )
{//impact would be too close to me
tooClose = qtrue;
}
else if ( level.time - NPCInfo->enemyLastSeenTime > 5000 ||
(NPCInfo->group && level.time - NPCInfo->group->lastSeenEnemyTime > 5000 ))
{//we've haven't seen them in the last 5 seconds
//see if it's too far from where he is
distThreshold = 65536/*256*256*/;//default
switch ( NPC->s.weapon )
{
case WP_ROCKET_LAUNCHER:
case WP_FLECHETTE:
case WP_THERMAL:
case WP_TRIP_MINE:
case WP_DET_PACK:
distThreshold = 262144/*512*512*/;
break;
case WP_REPEATER:
if ( NPCInfo->scriptFlags&SCF_ALT_FIRE )
{
distThreshold = 262144/*512*512*/;
}
break;
case WP_CONCUSSION:
if ( !(NPCInfo->scriptFlags&SCF_ALT_FIRE) )
{
distThreshold = 262144/*512*512*/;
}
break;
default:
break;
}
dist = DistanceSquared( impactPos, NPCInfo->enemyLastSeenLocation );
if ( dist > distThreshold )
{//impact would be too far from enemy
tooFar = qtrue;
}
}
if ( !tooClose && !tooFar )
{//okay too shoot at last pos
VectorSubtract( NPCInfo->enemyLastSeenLocation, muzzle, dir );
VectorNormalize( dir );
vectoangles( dir, angles );
NPCInfo->desiredYaw = angles[YAW];
NPCInfo->desiredPitch = angles[PITCH];
shoot = qtrue;
faceEnemy = qfalse;
//AI_GroupUpdateSquadstates( NPCInfo->group, NPC, SQUAD_STAND_AND_SHOOT );
return;
}
}
}
}
}
void ST_TrackEnemy( gentity_t *self, vec3_t enemyPos )
{
//clear timers
TIMER_Set( self, "attackDelay", Q_irand( 1000, 2000 ) );
//TIMER_Set( self, "duck", -1 );
TIMER_Set( self, "stick", Q_irand( 500, 1500 ) );
TIMER_Set( self, "stand", -1 );
TIMER_Set( self, "scoutTime", TIMER_Get( self, "stick" )-level.time+Q_irand(5000, 10000) );
//leave my combat point
NPC_FreeCombatPoint( self->NPC->combatPoint );
//go after his last seen pos
NPC_SetMoveGoal( self, enemyPos, 100.0f, qfalse );
if (Q_irand(0,3)==0)
{
NPCInfo->aiFlags |= NPCAI_STOP_AT_LOS;
}
}
int ST_ApproachEnemy( gentity_t *self )
{
TIMER_Set( self, "attackDelay", Q_irand( 250, 500 ) );
//TIMER_Set( self, "duck", -1 );
TIMER_Set( self, "stick", Q_irand( 1000, 2000 ) );
TIMER_Set( self, "stand", -1 );
TIMER_Set( self, "scoutTime", TIMER_Get( self, "stick" )-level.time+Q_irand(5000, 10000) );
//leave my combat point
NPC_FreeCombatPoint( self->NPC->combatPoint );
//return the relevant combat point flags
return (CP_CLEAR|CP_CLOSEST);
}
void ST_HuntEnemy( gentity_t *self )
{
//TIMER_Set( NPC, "attackDelay", Q_irand( 250, 500 ) );//Disabled this for now, guys who couldn't hunt would never attack
//TIMER_Set( NPC, "duck", -1 );
TIMER_Set( NPC, "stick", Q_irand( 250, 1000 ) );
TIMER_Set( NPC, "stand", -1 );
TIMER_Set( NPC, "scoutTime", TIMER_Get( NPC, "stick" )-level.time+Q_irand(5000, 10000) );
//leave my combat point
NPC_FreeCombatPoint( NPCInfo->combatPoint );
//go directly after the enemy
if ( NPCInfo->scriptFlags & SCF_CHASE_ENEMIES )
{
self->NPC->goalEntity = NPC->enemy;
}
}
void ST_TransferTimers( gentity_t *self, gentity_t *other )
{
TIMER_Set( other, "attackDelay", TIMER_Get( self, "attackDelay" )-level.time );
TIMER_Set( other, "duck", TIMER_Get( self, "duck" )-level.time );
TIMER_Set( other, "stick", TIMER_Get( self, "stick" )-level.time );
TIMER_Set( other, "scoutTime", TIMER_Get( self, "scoutTime" )-level.time );
TIMER_Set( other, "roamTime", TIMER_Get( self, "roamTime" )-level.time );
TIMER_Set( other, "stand", TIMER_Get( self, "stand" )-level.time );
TIMER_Set( self, "attackDelay", -1 );
TIMER_Set( self, "duck", -1 );
TIMER_Set( self, "stick", -1 );
TIMER_Set( self, "scoutTime", -1 );
TIMER_Set( self, "roamTime", -1 );
TIMER_Set( self, "stand", -1 );
}
void ST_TransferMoveGoal( gentity_t *self, gentity_t *other )
{
if ( Q3_TaskIDPending( self, TID_MOVE_NAV ) )
{//can't transfer movegoal when a script we're running is waiting to complete
return;
}
if ( self->NPC->combatPoint != -1 )
{//I've got a combatPoint I'm going to, give it to him
self->NPC->lastFailedCombatPoint = other->NPC->combatPoint = self->NPC->combatPoint;
self->NPC->combatPoint = -1;
}
else
{//I must be going for a goal, give that to him instead
if ( self->NPC->goalEntity == self->NPC->tempGoal )
{
NPC_SetMoveGoal( other, self->NPC->tempGoal->currentOrigin, self->NPC->goalRadius, (qboolean)((self->NPC->tempGoal->svFlags & SVF_NAVGOAL) != 0) );
}
else
{
other->NPC->goalEntity = self->NPC->goalEntity;
}
}
//give him my squadstate
AI_GroupUpdateSquadstates( self->NPC->group, other, NPCInfo->squadState );
//give him my timers and clear mine
ST_TransferTimers( self, other );
//now make me stand around for a second or two at least
AI_GroupUpdateSquadstates( self->NPC->group, self, SQUAD_STAND_AND_SHOOT );
TIMER_Set( self, "stand", Q_irand( 1000, 3000 ) );
}
int ST_GetCPFlags( void )
{
int cpFlags = 0;
if ( NPC && NPCInfo->group )
{
if ( NPC == NPCInfo->group->commander && NPC->client->NPC_class == CLASS_IMPERIAL )
{//imperials hang back and give orders
if ( NPCInfo->group->numGroup > 1 && Q_irand( -3, NPCInfo->group->numGroup ) > 1 )
{//FIXME: make sure he;s giving orders with these lines
if ( Q_irand( 0, 1 ) )
{
ST_Speech( NPC, SPEECH_CHASE, 0.5 );
}
else
{
ST_Speech( NPC, SPEECH_YELL, 0.5 );
}
}
cpFlags = (CP_CLEAR|CP_COVER|CP_AVOID|CP_SAFE|CP_RETREAT);
}
else if ( NPCInfo->group->morale < 0 )
{//hide
cpFlags = (CP_COVER|CP_AVOID|CP_SAFE|CP_RETREAT);
/*
if ( NPC->client->NPC_class == CLASS_SABOTEUR && !Q_irand( 0, 3 ) )
{
Saboteur_Cloak( NPC );
}
*/
}
/* else if ( NPCInfo->group->morale < NPCInfo->group->numGroup )
{//morale is low for our size
int moraleDrop = NPCInfo->group->numGroup - NPCInfo->group->morale;
if ( moraleDrop < -6 )
{//flee (no clear shot needed)
cpFlags = (CP_FLEE|CP_RETREAT|CP_COVER|CP_AVOID|CP_SAFE);
}
else if ( moraleDrop < -3 )
{//retreat (no clear shot needed)
cpFlags = (CP_RETREAT|CP_COVER|CP_AVOID|CP_SAFE);
}
else if ( moraleDrop < 0 )
{//cover (no clear shot needed)
cpFlags = (CP_COVER|CP_AVOID|CP_SAFE);
}
}*/
else
{
int moraleBoost = NPCInfo->group->morale - NPCInfo->group->numGroup;
if ( moraleBoost > 20 )
{//charge to any one and outflank (no cover needed)
cpFlags = (CP_CLEAR|CP_FLANK|CP_APPROACH_ENEMY);
//Saboteur_Decloak( NPC );
}
else if ( moraleBoost > 15 )
{//charge to closest one (no cover needed)
cpFlags = (CP_CLEAR|CP_CLOSEST|CP_APPROACH_ENEMY);
/*
if ( NPC->client->NPC_class == CLASS_SABOTEUR && !Q_irand( 0, 3 ) )
{
Saboteur_Decloak( NPC );
}
*/
}
else if ( moraleBoost > 10 )
{//charge closer (no cover needed)
cpFlags = (CP_CLEAR|CP_APPROACH_ENEMY);
/*
if ( NPC->client->NPC_class == CLASS_SABOTEUR && !Q_irand( 0, 6 ) )
{
Saboteur_Decloak( NPC );
}
*/
}
}
}
if ( !cpFlags )
{
//at some medium level of morale
switch( Q_irand( 0, 3 ) )
{
case 0://just take the nearest one
cpFlags = (CP_CLEAR|CP_COVER|CP_NEAREST);
break;
case 1://take one closer to the enemy
cpFlags = (CP_CLEAR|CP_COVER|CP_APPROACH_ENEMY);
break;
case 2://take the one closest to the enemy
cpFlags = (CP_CLEAR|CP_COVER|CP_CLOSEST|CP_APPROACH_ENEMY);
break;
case 3://take the one on the other side of the enemy
cpFlags = (CP_CLEAR|CP_COVER|CP_FLANK|CP_APPROACH_ENEMY);
break;
}
}
if ( NPC && (NPCInfo->scriptFlags&SCF_USE_CP_NEAREST) )
{
cpFlags &= ~(CP_FLANK|CP_APPROACH_ENEMY|CP_CLOSEST);
cpFlags |= CP_NEAREST;
}
return cpFlags;
}
/*
-------------------------
ST_Commander
Make decisions about who should go where, etc.
FIXME: leader (group-decision-making) AI?
FIXME: need alternate routes!
FIXME: more group voice interaction
FIXME: work in pairs?
-------------------------
*/
void ST_Commander( void )
{
int i;//, j;
int cp, cpFlags;
AIGroupInfo_t *group = NPCInfo->group;
gentity_t *member;//, *buddy;
qboolean enemyLost = qfalse;
float avoidDist;
group->processed = qtrue;
if ( group->enemy == NULL || group->enemy->client == NULL )
{//hmm, no enemy...?!
return;
}
//FIXME: have this group commander check the enemy group (if any) and see if they have
// superior numbers. If they do, fall back rather than advance. If you have
// superior numbers, advance on them.
//FIXME: find the group commander and have him occasionally give orders when there is speech
//FIXME: start fleeing when only a couple of you vs. a lightsaber, possibly give up if the only one left
SaveNPCGlobals();
if ( group->lastSeenEnemyTime < level.time - 180000 )
{//dissolve the group
ST_Speech( NPC, SPEECH_LOST, 0.0f );
group->enemy->waypoint = NAV::GetNearestNode(group->enemy);
for ( i = 0; i < group->numGroup; i++ )
{
member = &g_entities[group->member[i].number];
SetNPCGlobals( member );
if ( Q3_TaskIDPending( NPC, TID_MOVE_NAV ) )
{//running somewhere that a script requires us to go, don't break from that
continue;
}
if ( !(NPCInfo->scriptFlags&SCF_CHASE_ENEMIES) )
{//not allowed to doMove on my own
continue;
}
//Lost enemy for three minutes? go into search mode?
G_ClearEnemy( NPC );
NPC->waypoint = NAV::GetNearestNode(group->enemy);
if ( NPC->waypoint == WAYPOINT_NONE )
{
NPCInfo->behaviorState = BS_DEFAULT;//BS_PATROL;
}
else if ( group->enemy->waypoint == WAYPOINT_NONE || (NAV::EstimateCostToGoal( NPC->waypoint, group->enemy->waypoint ) >= Q3_INFINITE) )
{
NPC_BSSearchStart( NPC->waypoint, BS_SEARCH );
}
else
{
NPC_BSSearchStart( group->enemy->waypoint, BS_SEARCH );
}
}
group->enemy = NULL;
RestoreNPCGlobals();
return;
}
if ( /*!runner &&*/ group->lastSeenEnemyTime > level.time - 32000 && group->lastSeenEnemyTime < level.time - 30000 )
{//no-one has seen the enemy for 30 seconds// and no-one is running after him
if ( group->commander && !Q_irand( 0, 1 ) )
{
ST_Speech( group->commander, SPEECH_ESCAPING, 0.0f );
}
else
{
ST_Speech( NPC, SPEECH_ESCAPING, 0.0f );
}
//don't say this again
NPCInfo->blockedSpeechDebounceTime = level.time + 3000;
}
if ( group->lastSeenEnemyTime < level.time - 7000 )
{//no-one has seen the enemy for at least 10 seconds! Should send a scout
enemyLost = qtrue;
}
//Go through the list:
//Everyone should try to get to a combat point if possible
int curMemberNum, lastMemberNum;
if ( d_asynchronousGroupAI->integer )
{//do one member a turn
group->activeMemberNum++;
if ( group->activeMemberNum >= group->numGroup )
{
group->activeMemberNum = 0;
}
curMemberNum = group->activeMemberNum;
lastMemberNum = curMemberNum + 1;
}
else
{
curMemberNum = 0;
lastMemberNum = group->numGroup;
}
for ( i = curMemberNum; i < lastMemberNum; i++ )
{
//reset combat point flags
cp = -1;
cpFlags = 0;
avoidDist = 0;
//get the next guy
member = &g_entities[group->member[i].number];
if ( !member->enemy )
{//don't include guys that aren't angry
continue;
}
SetNPCGlobals( member );
if ( !TIMER_Done( NPC, "flee" ) )
{//running away
continue;
}
if ( Q3_TaskIDPending( NPC, TID_MOVE_NAV ) )
{//running somewhere that a script requires us to go
continue;
}
if ( NPC->s.weapon == WP_NONE
&& NPCInfo->goalEntity
&& NPCInfo->goalEntity == NPCInfo->tempGoal
&& NPCInfo->goalEntity->s.eType == ET_ITEM )
{//running to pick up a gun, don't do other logic
continue;
}
if ( !(NPCInfo->scriptFlags&SCF_CHASE_ENEMIES) )
{//not allowed to do combat-movement
continue;
}
if ( NPC->client->ps.weapon == WP_NONE )
{//weaponless, should be hiding
if ( NPCInfo->goalEntity == NULL || NPCInfo->goalEntity->enemy == NULL || NPCInfo->goalEntity->enemy->s.eType != ET_ITEM )
{//not running after a pickup
if ( TIMER_Done( NPC, "hideTime" ) || (DistanceSquared( group->enemy->currentOrigin, NPC->currentOrigin ) < 65536 && NPC_ClearLOS( NPC->enemy )) )
{//done hiding or enemy near and can see us
//er, start another flee I guess?
NPC_StartFlee( NPC->enemy, NPC->enemy->currentOrigin, AEL_DANGER_GREAT, 5000, 10000 );
}//else, just hang here
}
continue;
}
if (enemyLost && NAV::InSameRegion(NPC, NPC->enemy->currentOrigin))
{
ST_TrackEnemy( NPC, NPC->enemy->currentOrigin );
continue;
}
if (!NPC->enemy)
{
continue;
}
// Check To See We Have A Clear Shot To The Enemy Every Couple Seconds
//---------------------------------------------------------------------
if (TIMER_Done( NPC, "checkGrenadeTooCloseDebouncer" ))
{
TIMER_Set (NPC, "checkGrenadeTooCloseDebouncer", Q_irand(300, 600));
vec3_t mins;
vec3_t maxs;
bool fled = false;
gentity_t* ent;
gentity_t *entityList[MAX_GENTITIES];
for (int i = 0 ; i < 3 ; i++ )
{
mins[i] = NPC->currentOrigin[i] - 200;
maxs[i] = NPC->currentOrigin[i] + 200;
}
int numListedEntities = gi.EntitiesInBox( mins, maxs, entityList, MAX_GENTITIES );
for (int e = 0 ; e < numListedEntities ; e++ )
{
ent = entityList[ e ];
if (ent == NPC)
continue;
if (ent->owner == NPC)
continue;
if ( !(ent->inuse) )
continue;
if ( ent->s.eType == ET_MISSILE )
{
if ( ent->s.weapon == WP_THERMAL )
{//a thermal
if ( ent->has_bounced && (!ent->owner || !OnSameTeam(ent->owner, NPC)))
{//bounced and an enemy thermal
ST_Speech( NPC, SPEECH_COVER, 0 );//FIXME: flee sound?
NPC_StartFlee(NPC->enemy, ent->currentOrigin, AEL_DANGER_GREAT, 1000, 2000);
fled = true;
// cpFlags |= (CP_CLEAR|CP_COVER); // NOPE, Can't See The Enemy, So Find A New Combat Point
TIMER_Set (NPC, "checkGrenadeTooCloseDebouncer", Q_irand(2000, 4000));
break;
}
}
}
}
if (fled)
{
continue;
}
}
// Check To See We Have A Clear Shot To The Enemy Every Couple Seconds
//---------------------------------------------------------------------
if (TIMER_Done( NPC, "checkEnemyVisDebouncer" ))
{
TIMER_Set (NPC, "checkEnemyVisDebouncer", Q_irand(3000, 7000));
if (!NPC_ClearLOS(NPC->enemy))
{
cpFlags |= (CP_CLEAR|CP_COVER); // NOPE, Can't See The Enemy, So Find A New Combat Point
}
}
// Check To See If The Enemy Is Too Close For Comfort
//----------------------------------------------------
if (NPC->client->NPC_class!=CLASS_ASSASSIN_DROID)
{
if (TIMER_Done(NPC, "checkEnemyTooCloseDebouncer"))
{
TIMER_Set (NPC, "checkEnemyTooCloseDebouncer", Q_irand(1000, 6000));
float distThreshold = 16384/*128*128*/;//default
switch ( NPC->s.weapon )
{
case WP_ROCKET_LAUNCHER:
case WP_FLECHETTE:
case WP_THERMAL:
case WP_TRIP_MINE:
case WP_DET_PACK:
distThreshold = 65536/*256*256*/;
break;
case WP_REPEATER:
if ( NPCInfo->scriptFlags&SCF_ALT_FIRE )
{
distThreshold = 65536/*256*256*/;
}
break;
case WP_CONCUSSION:
if ( !(NPCInfo->scriptFlags&SCF_ALT_FIRE) )
{
distThreshold = 65536/*256*256*/;
}
break;
default:
break;
}
if ( DistanceSquared( group->enemy->currentOrigin, NPC->currentOrigin ) < distThreshold )
{
cpFlags |= (CP_CLEAR|CP_COVER);
}
}
}
//clear the local state
NPCInfo->localState = LSTATE_NONE;
cpFlags &= ~CP_NEAREST;
//Assign combat points
if ( cpFlags )
{//we want to run to a combat point
//always avoid enemy when picking combat points, and we always want to be able to get there
cpFlags |= CP_AVOID_ENEMY|CP_HAS_ROUTE|CP_TRYFAR;
avoidDist = 200;
//now get a combat point
if ( cp == -1 )
{//may have had sone set above
cp = NPC_FindCombatPointRetry( NPC->currentOrigin, NPC->currentOrigin, NPC->currentOrigin, &cpFlags, avoidDist, NPCInfo->lastFailedCombatPoint );
}
//see if we got a valid one
if ( cp != -1 )
{//found a combat point
//let others know that someone is now running
//don't change course again until we get to where we're going
TIMER_Set( NPC, "roamTime", Q3_INFINITE );
NPC_SetCombatPoint( cp );
NPC_SetMoveGoal( NPC, level.combatPoints[cp].origin, 8, qtrue, cp );
// If Successfully
if ((cpFlags&CP_FLANK) || ((cpFlags&CP_COVER) && (cpFlags&CP_CLEAR)))
{
}
else if (Q_irand(0,3)==0)
{
NPCInfo->aiFlags |= NPCAI_STOP_AT_LOS;
}
//okay, try a doMove right now to see if we can even get there
if ( (cpFlags&CP_FLANK) )
{
if ( group->numGroup > 1 )
{
NPC_ST_StoreMovementSpeech( SPEECH_OUTFLANK, -1 );
}
}
else if ( (cpFlags&CP_COVER) && !(cpFlags&CP_CLEAR) )
{//going into hiding
NPC_ST_StoreMovementSpeech( SPEECH_COVER, -1 );
}
else
{
if ( !Q_irand( 0, 20 ) )
{//hell, we're loading the sounds, use them every now and then!
if ( Q_irand( 0, 1 ) )
{
NPC_ST_StoreMovementSpeech( SPEECH_OUTFLANK, -1 );
}
else
{
NPC_ST_StoreMovementSpeech( SPEECH_ESCAPING, -1 );
}
}
}
}
}
}
RestoreNPCGlobals();
return;
}
extern void G_Knockdown( gentity_t *self, gentity_t *attacker, const vec3_t pushDir, float strength, qboolean breakSaberLock );
void Noghri_StickTrace( void )
{
if ( !NPC->ghoul2.size()
|| NPC->weaponModel[0] <= 0 )
{
return;
}
int boltIndex = gi.G2API_AddBolt(&NPC->ghoul2[NPC->weaponModel[0]], "*weapon");
if ( boltIndex != -1 )
{
int curTime = (cg.time?cg.time:level.time);
qboolean hit = qfalse;
int lastHit = ENTITYNUM_NONE;
for ( int time = curTime-25; time <= curTime+25&&!hit; time += 25 )
{
mdxaBone_t boltMatrix;
vec3_t tip, dir, base, angles={0,NPC->currentAngles[YAW],0};
vec3_t mins={-2,-2,-2},maxs={2,2,2};
trace_t trace;
gi.G2API_GetBoltMatrix( NPC->ghoul2, NPC->weaponModel[0],
boltIndex,
&boltMatrix, angles, NPC->currentOrigin, time,
NULL, NPC->s.modelScale );
gi.G2API_GiveMeVectorFromMatrix( boltMatrix, ORIGIN, base );
gi.G2API_GiveMeVectorFromMatrix( boltMatrix, POSITIVE_Y, dir );
VectorMA( base, 48, dir, tip );
#ifndef FINAL_BUILD
if ( d_saberCombat->integer > 1 )
{
G_DebugLine(base, tip, FRAMETIME, 0x000000ff, qtrue);
}
#endif
gi.trace( &trace, base, mins, maxs, tip, NPC->s.number, MASK_SHOT, G2_RETURNONHIT, 10 );
if ( trace.fraction < 1.0f && trace.entityNum != lastHit )
{//hit something
gentity_t *traceEnt = &g_entities[trace.entityNum];
if ( traceEnt->takedamage
&& (!traceEnt->client || traceEnt == NPC->enemy || traceEnt->client->NPC_class != NPC->client->NPC_class) )
{//smack
int dmg = Q_irand( 12, 20 );//FIXME: base on skill!
//FIXME: debounce?
G_Sound( traceEnt, G_SoundIndex( va( "sound/weapons/tusken_staff/stickhit%d.wav", Q_irand( 1, 4 ) ) ) );
G_Damage( traceEnt, NPC, NPC, vec3_origin, trace.endpos, dmg, DAMAGE_NO_KNOCKBACK, MOD_MELEE );
if ( traceEnt->health > 0 && dmg > 17 )
{//do pain on enemy
G_Knockdown( traceEnt, NPC, dir, 300, qtrue );
}
lastHit = trace.entityNum;
hit = qtrue;
}
}
}
}
}
/*
-------------------------
NPC_BSST_Attack
-------------------------
*/
void NPC_BSST_Attack( void )
{
//Don't do anything if we're hurt
if ( NPC->painDebounceTime > level.time )
{
NPC_UpdateAngles( qtrue, qtrue );
return;
}
//NPC_CheckEnemy( qtrue, qfalse );
//If we don't have an enemy, just idle
if ( NPC_CheckEnemyExt() == qfalse )//!NPC->enemy )//
{
if( NPC->client->playerTeam == TEAM_PLAYER )
{
NPC_BSPatrol();
}
else
{
NPC_BSST_Patrol();//FIXME: or patrol?
}
return;
}
//FIXME: put some sort of delay into the guys depending on how they saw you...?
//Get our group info
if ( TIMER_Done( NPC, "interrogating" ) )
{
AI_GetGroup( NPC );//, 45, 512, NPC->enemy );
}
else
{
//FIXME: when done interrogating, I should send out a team alert!
}
if ( NPCInfo->group )
{//I belong to a squad of guys - we should *always* have a group
if ( !NPCInfo->group->processed )
{//I'm the first ent in my group, I'll make the command decisions
#if AI_TIMERS
int startTime = GetTime(0);
#endif// AI_TIMERS
ST_Commander();
#if AI_TIMERS
int commTime = GetTime ( startTime );
if ( commTime > 20 )
{
gi.Printf( S_COLOR_RED"ERROR: Commander time: %d\n", commTime );
}
else if ( commTime > 10 )
{
gi.Printf( S_COLOR_YELLOW"WARNING: Commander time: %d\n", commTime );
}
else if ( commTime > 2 )
{
gi.Printf( S_COLOR_GREEN"Commander time: %d\n", commTime );
}
#endif// AI_TIMERS
}
}
else if ( TIMER_Done( NPC, "flee" ) && NPC_CheckForDanger( NPC_CheckAlertEvents( qtrue, qtrue, -1, qfalse, AEL_DANGER ) ) )
{//not already fleeing, and going to run
ST_Speech( NPC, SPEECH_COVER, 0 );
NPC_UpdateAngles( qtrue, qtrue );
return;
}
if ( !NPC->enemy )
{//WTF? somehow we lost our enemy?
NPC_BSST_Patrol();//FIXME: or patrol?
return;
}
if (NPCInfo->goalEntity && NPCInfo->goalEntity!=NPC->enemy)
{
NPCInfo->goalEntity = UpdateGoal();
}
enemyLOS = enemyCS = enemyInFOV = qfalse;
doMove = qtrue;
faceEnemy = qfalse;
shoot = qfalse;
hitAlly = qfalse;
VectorClear( impactPos );
enemyDist = DistanceSquared( NPC->currentOrigin, NPC->enemy->currentOrigin );
vec3_t enemyDir, shootDir;
VectorSubtract( NPC->enemy->currentOrigin, NPC->currentOrigin, enemyDir );
VectorNormalize( enemyDir );
AngleVectors( NPC->client->ps.viewangles, shootDir, NULL, NULL );
float dot = DotProduct( enemyDir, shootDir );
if ( dot > 0.5f ||( enemyDist * (1.0f-dot)) < 10000 )
{//enemy is in front of me or they're very close and not behind me
enemyInFOV = qtrue;
}
if ( enemyDist < MIN_ROCKET_DIST_SQUARED )//128
{//enemy within 128
if ( (NPC->client->ps.weapon == WP_FLECHETTE || NPC->client->ps.weapon == WP_REPEATER) &&
(NPCInfo->scriptFlags & SCF_ALT_FIRE) )
{//shooting an explosive, but enemy too close, switch to primary fire
NPCInfo->scriptFlags &= ~SCF_ALT_FIRE;
//FIXME: we can never go back to alt-fire this way since, after this, we don't know if we were initially supposed to use alt-fire or not...
}
}
else if ( enemyDist > 65536 )//256 squared
{
if ( NPC->client->ps.weapon == WP_DISRUPTOR )
{//sniping...
if ( !(NPCInfo->scriptFlags&SCF_ALT_FIRE) )
{//use primary fire
NPCInfo->scriptFlags |= SCF_ALT_FIRE;
//reset fire-timing variables
NPC_ChangeWeapon( NPC->client->ps.weapon );
NPC_UpdateAngles( qtrue, qtrue );
return;
}
}
}
//can we see our target?
if ( NPC_ClearLOS( NPC->enemy ) )
{
AI_GroupUpdateEnemyLastSeen( NPCInfo->group, NPC->enemy->currentOrigin );
NPCInfo->enemyLastSeenTime = level.time;
enemyLOS = qtrue;
if ( NPC->client->ps.weapon == WP_NONE )
{
enemyCS = qfalse;//not true, but should stop us from firing
NPC_AimAdjust( -1 );//adjust aim worse longer we have no weapon
}
else
{//can we shoot our target?
if ((enemyDist < MIN_ROCKET_DIST_SQUARED) &&
((level.time - NPC->lastMoveTime)<5000) &&
(
(NPC->client->ps.weapon == WP_ROCKET_LAUNCHER
|| (NPC->client->ps.weapon == WP_CONCUSSION && !(NPCInfo->scriptFlags&SCF_ALT_FIRE))
|| (NPC->client->ps.weapon == WP_FLECHETTE && (NPCInfo->scriptFlags&SCF_ALT_FIRE)))))
{
enemyCS = qfalse;//not true, but should stop us from firing
hitAlly = qtrue;//us!
//FIXME: if too close, run away!
}
else if ( enemyInFOV )
{//if enemy is FOV, go ahead and check for shooting
int hit = NPC_ShotEntity( NPC->enemy, impactPos );
gentity_t *hitEnt = &g_entities[hit];
if ( hit == NPC->enemy->s.number
|| ( hitEnt && hitEnt->client && hitEnt->client->playerTeam == NPC->client->enemyTeam )
|| ( hitEnt && hitEnt->takedamage && ((hitEnt->svFlags&SVF_GLASS_BRUSH)||hitEnt->health < 40||NPC->s.weapon == WP_EMPLACED_GUN) ) )
{//can hit enemy or enemy ally or will hit glass or other minor breakable (or in emplaced gun), so shoot anyway
AI_GroupUpdateClearShotTime( NPCInfo->group );
enemyCS = qtrue;
NPC_AimAdjust( 2 );//adjust aim better longer we have clear shot at enemy
VectorCopy( NPC->enemy->currentOrigin, NPCInfo->enemyLastSeenLocation );
}
else
{//Hmm, have to get around this bastard
NPC_AimAdjust( 1 );//adjust aim better longer we can see enemy
ST_ResolveBlockedShot( hit );
if ( hitEnt && hitEnt->client && hitEnt->client->playerTeam == NPC->client->playerTeam )
{//would hit an ally, don't fire!!!
hitAlly = qtrue;
}
else
{//Check and see where our shot *would* hit... if it's not close to the enemy (within 256?), then don't fire
}
}
}
else
{
enemyCS = qfalse;//not true, but should stop us from firing
}
}
}
else if ( gi.inPVS( NPC->enemy->currentOrigin, NPC->currentOrigin ) )
{
NPCInfo->enemyLastSeenTime = level.time;
faceEnemy = qtrue;
NPC_AimAdjust( -1 );//adjust aim worse longer we cannot see enemy
}
if ( NPC->client->ps.weapon == WP_NONE )
{
faceEnemy = qfalse;
shoot = qfalse;
}
else
{
if ( enemyLOS )
{//FIXME: no need to face enemy if we're moving to some other goal and he's too far away to shoot?
faceEnemy = qtrue;
}
if ( enemyCS )
{
shoot = qtrue;
}
}
//Check for movement to take care of
ST_CheckMoveState();
//See if we should override shooting decision with any special considerations
ST_CheckFireState();
if ( faceEnemy )
{//face the enemy
NPC_FaceEnemy( qtrue );
}
if ( !(NPCInfo->scriptFlags&SCF_CHASE_ENEMIES) )
{//not supposed to chase my enemies
if ( NPCInfo->goalEntity == NPC->enemy )
{//goal is my entity, so don't doMove
doMove = qfalse;
}
}
else if (NPC->NPC->scriptFlags&SCF_NO_GROUPS)
{
// NPCInfo->goalEntity = UpdateGoal();
NPCInfo->goalEntity = (enemyLOS)?(0):(NPC->enemy);
}
if ( NPC->client->fireDelay && NPC->s.weapon == WP_ROCKET_LAUNCHER )
{
doMove = qfalse;
}
if ( !ucmd.rightmove )
{//only if not already strafing for some strange reason...?
//NOTE: these are never set here, but can be set in AI_Jedi.cpp for those NPCs who are sort of Stormtrooper/Jedi hybrids
//NOTE: this stomps navigation movement entirely!
//FIXME: if enemy behind me and turning to face enemy, don't strafe in that direction, too
if ( !TIMER_Done( NPC, "strafeLeft" ) )
{
/*
if ( NPCInfo->desiredYaw > NPC->client->ps.viewangles[YAW] + 60 )
{//we want to turn left, don't apply the strafing
}
else
*/
{//go ahead and strafe left
ucmd.rightmove = -127;
//re-check the duck as we might want to be rolling
VectorClear( NPC->client->ps.moveDir );
doMove = qfalse;
}
}
else if ( !TIMER_Done( NPC, "strafeRight" ) )
{
/*if ( NPCInfo->desiredYaw < NPC->client->ps.viewangles[YAW] - 60 )
{//we want to turn right, don't apply the strafing
}
else
*/
{//go ahead and strafe left
ucmd.rightmove = 127;
VectorClear( NPC->client->ps.moveDir );
doMove = qfalse;
}
}
}
if ( NPC->client->ps.legsAnim == BOTH_GUARD_LOOKAROUND1 )
{//don't doMove when doing silly look around thing
doMove = qfalse;
}
if ( doMove )
{//doMove toward goal
if ( NPCInfo->goalEntity )//&& ( NPCInfo->goalEntity != NPC->enemy || enemyDist > 10000 ) )//100 squared
{
doMove = ST_Move();
if ( (NPC->client->NPC_class != CLASS_ROCKETTROOPER||NPC->s.weapon!=WP_ROCKET_LAUNCHER||enemyDist<MIN_ROCKET_DIST_SQUARED)//rockettroopers who use rocket launchers turn around and run if you get too close (closer than 128)
&& ucmd.forwardmove <= -32 )
{//moving backwards at least 45 degrees
if ( NPCInfo->goalEntity
&& DistanceSquared( NPCInfo->goalEntity->currentOrigin, NPC->currentOrigin ) > MIN_TURN_AROUND_DIST_SQ )
{//don't stop running backwards if your goal is less than 100 away
if ( TIMER_Done( NPC, "runBackwardsDebounce" ) )
{//not already waiting for next run backwards
if ( !TIMER_Exists( NPC, "runningBackwards" ) )
{//start running backwards
TIMER_Set( NPC, "runningBackwards", Q_irand( 500, 1000 ) );//Q_irand( 2000, 3500 ) );
}
else if ( TIMER_Done2( NPC, "runningBackwards", qtrue ) )
{//done running backwards
TIMER_Set( NPC, "runBackwardsDebounce", Q_irand( 3000, 5000 ) );
}
}
}
}
else
{//not running backwards
//TIMER_Remove( NPC, "runningBackwards" );
}
}
else
{
doMove = qfalse;
}
}
if ( !doMove )
{
if (NPC->client->NPC_class != CLASS_ASSASSIN_DROID)
{
if ( !TIMER_Done( NPC, "duck" ) )
{
ucmd.upmove = -127;
}
}
//FIXME: what about leaning?
}
else
{//stop ducking!
TIMER_Set( NPC, "duck", -1 );
}
if ( NPC->client->NPC_class == CLASS_REBORN//cultist using a gun
&& NPCInfo->rank >= RANK_LT_COMM //commando or better
&& NPC->enemy->s.weapon == WP_SABER )//fighting a saber-user
{//commando saboteur vs. jedi/reborn
//see if we need to avoid their saber
NPC_EvasionSaber();
}
if ( //!TIMER_Done( NPC, "flee" ) ||
(doMove&&!TIMER_Done( NPC, "runBackwardsDebounce" )) )
{//running away
faceEnemy = qfalse;
}
//FIXME: check scf_face_move_dir here?
if ( !faceEnemy )
{//we want to face in the dir we're running
if ( !doMove )
{//if we haven't moved, we should look in the direction we last looked?
VectorCopy( NPC->client->ps.viewangles, NPCInfo->lastPathAngles );
}
NPCInfo->desiredYaw = NPCInfo->lastPathAngles[YAW];
NPCInfo->desiredPitch = 0;
NPC_UpdateAngles( qtrue, qtrue );
if ( doMove )
{//don't run away and shoot
shoot = qfalse;
}
}
if ( NPCInfo->scriptFlags & SCF_DONT_FIRE )
{
shoot = qfalse;
}
if ( NPC->enemy && NPC->enemy->enemy )
{
if ( NPC->enemy->s.weapon == WP_SABER && NPC->enemy->enemy->s.weapon == WP_SABER )
{//don't shoot at an enemy jedi who is fighting another jedi, for fear of injuring one or causing rogue blaster deflections (a la Obi Wan/Vader duel at end of ANH)
shoot = qfalse;
}
}
//FIXME: don't shoot right away!
if ( NPC->client->fireDelay )
{
if ( NPC->client->NPC_class == CLASS_SABOTEUR )
{
Saboteur_Decloak( NPC );
}
if ( NPC->s.weapon == WP_ROCKET_LAUNCHER
|| (NPC->s.weapon==WP_CONCUSSION&&!(NPCInfo->scriptFlags&SCF_ALT_FIRE)) )
{
if ( !enemyLOS || !enemyCS )
{//cancel it
NPC->client->fireDelay = 0;
}
else
{//delay our next attempt
TIMER_Set( NPC, "attackDelay", Q_irand( 3000, 5000 ) );
}
}
}
else if ( shoot )
{//try to shoot if it's time
if ( NPC->client->NPC_class == CLASS_SABOTEUR )
{
Saboteur_Decloak( NPC );
}
if ( TIMER_Done( NPC, "attackDelay" ) )
{
if( !(NPCInfo->scriptFlags & SCF_FIRE_WEAPON) ) // we've already fired, no need to do it again here
{
WeaponThink( qtrue );
}
//NASTY
if ( NPC->s.weapon == WP_ROCKET_LAUNCHER )
{
if ( (ucmd.buttons&BUTTON_ATTACK)
&& !doMove
&& g_spskill->integer > 1
&& !Q_irand( 0, 3 ) )
{//every now and then, shoot a homing rocket
ucmd.buttons &= ~BUTTON_ATTACK;
ucmd.buttons |= BUTTON_ALT_ATTACK;
NPC->client->fireDelay = Q_irand( 1000, 2500 );
}
}
else if ( NPC->s.weapon == WP_NOGHRI_STICK
&& enemyDist < (48*48) )//?
{
ucmd.buttons &= ~BUTTON_ATTACK;
ucmd.buttons |= BUTTON_ALT_ATTACK;
NPC->client->fireDelay = Q_irand( 1500, 2000 );
}
}
}
else
{
if ( NPC->attackDebounceTime < level.time )
{
if ( NPC->client->NPC_class == CLASS_SABOTEUR )
{
Saboteur_Cloak( NPC );
}
}
}
}
extern qboolean G_TuskenAttackAnimDamage( gentity_t *self );
void NPC_BSST_Default( void )
{
if( NPCInfo->scriptFlags & SCF_FIRE_WEAPON )
{
WeaponThink( qtrue );
}
if ( NPC->s.weapon == WP_NOGHRI_STICK )
{
if ( G_TuskenAttackAnimDamage( NPC ) )
{
Noghri_StickTrace();
}
}
if( !NPC->enemy )
{//don't have an enemy, look for one
NPC_BSST_Patrol();
}
else //if ( NPC->enemy )
{//have an enemy
if ( NPC->enemy->client //enemy is a client
&& (NPC->enemy->client->NPC_class == CLASS_UGNAUGHT || NPC->enemy->client->NPC_class == CLASS_JAWA )//enemy is a lowly jawa or ugnaught
&& NPC->enemy->enemy != NPC//enemy's enemy is not me
&& (!NPC->enemy->enemy || !NPC->enemy->enemy->client || (NPC->enemy->enemy->client->NPC_class!=CLASS_RANCOR&&NPC->enemy->enemy->client->NPC_class!=CLASS_WAMPA)) )//enemy's enemy is not a client or is not a wampa or rancor (which is scarier than me)
{//they should be scared of ME and no-one else
G_SetEnemy( NPC->enemy, NPC );
}
NPC_CheckGetNewWeapon();
NPC_BSST_Attack();
}
}