thirtyflightsofloving/game/m_actor.c
Knightmare66 e7db654934 Added entity class IDs for misc_actor and target_actor in default Lazarus DLL.
Restored updated GPL header in game/g_local.h.
2020-10-27 02:31:41 -04:00

2203 lines
56 KiB
C

/*
===========================================================================
Copyright (C) 1997-2001 Id Software, Inc.
Copyright (C) 2000-2002 Mr. Hyde and Mad Dog
This file is part of Lazarus Quake 2 Mod source code.
Lazarus Quake 2 Mod source code is free software; you can redistribute it
and/or modify it under the terms of the GNU General Public License as
published by the Free Software Foundation; either version 2 of the License,
or (at your option) any later version.
Lazarus Quake 2 Mod source code 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 Lazarus Quake 2 Mod source code; if not, write to the Free Software
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
===========================================================================
*/
// m_actor.c
//
// Lazarus 1.4: Adopted Mappack misc_actor code
//
#include "g_local.h"
#include "m_actor.h"
#include "pak.h"
static char wavname[NUM_ACTOR_SOUNDS][32] =
{ "jump1.wav",
"pain25_1.wav",
"pain25_2.wav",
"pain50_1.wav",
"pain50_2.wav",
"pain75_1.wav",
"pain75_2.wav",
"pain100_1.wav",
"pain100_2.wav",
"death1.wav",
"death2.wav",
"death3.wav",
"death4.wav" };
#define ACTOR_SOUND_JUMP 0 // Do NOT change this one
#define ACTOR_SOUND_PAIN_25_1 1
#define ACTOR_SOUND_PAIN_25_2 2
#define ACTOR_SOUND_PAIN_50_1 3
#define ACTOR_SOUND_PAIN_50_2 4
#define ACTOR_SOUND_PAIN_75_1 5
#define ACTOR_SOUND_PAIN_75_2 6
#define ACTOR_SOUND_PAIN_100_1 7
#define ACTOR_SOUND_PAIN_100_2 8
#define ACTOR_SOUND_DEATH1 9
#define ACTOR_SOUND_DEATH2 10
#define ACTOR_SOUND_DEATH3 11
#define ACTOR_SOUND_DEATH4 12
mframe_t actor_frames_stand [] =
{
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL
};
mmove_t actor_move_stand = {FRAME_stand01, FRAME_stand40, actor_frames_stand, NULL};
void actor_stand (edict_t *self)
{
self->s.sound = 0;
if (self->monsterinfo.aiflags & AI_CROUCH)
self->monsterinfo.currentmove = &actor_move_crouch;
else
self->monsterinfo.currentmove = &actor_move_stand;
// randomize on startup
if (level.time < 1.0)
self->s.frame = self->monsterinfo.currentmove->firstframe + (rand() % (self->monsterinfo.currentmove->lastframe - self->monsterinfo.currentmove->firstframe + 1));
}
mframe_t actor_frames_walk [] =
{
/* ai_walk, 4, NULL,
ai_walk, 15, NULL,
ai_walk, 15, NULL,
ai_walk, 8, NULL,
ai_walk, 20, NULL,
ai_walk, 15, NULL,
ai_walk, 8, NULL,
ai_walk, 17, NULL,
ai_walk, 12, NULL,
ai_walk, -2, NULL,
ai_walk, -2, NULL,
ai_walk, -1, NULL */
ai_walk, 10, NULL,
ai_walk, 10, NULL,
ai_walk, 10, NULL,
ai_walk, 10, NULL,
ai_walk, 10, NULL,
ai_walk, 10, NULL,
ai_walk, 10, NULL,
ai_walk, 10, NULL,
ai_walk, 10, NULL,
ai_walk, 10, NULL,
ai_walk, 10, NULL,
ai_walk, 10, NULL
};
mmove_t actor_move_walk = {FRAME_run1, FRAME_run6, actor_frames_walk, NULL};
mmove_t actor_move_run;
void actor_walk (edict_t *self)
{
// prevent foolishness:
if (self->monsterinfo.aiflags & AI_FOLLOW_LEADER)
{
if (!self->movetarget || !self->movetarget->inuse || (self->movetarget == world))
self->movetarget = self->monsterinfo.leader;
}
if ( (self->monsterinfo.aiflags & AI_FOLLOW_LEADER) &&
(self->movetarget) &&
(self->movetarget->inuse) &&
(self->movetarget->health > 0) )
{
float R;
R = realrange(self,self->movetarget);
if (R > ACTOR_FOLLOW_RUN_RANGE || self->enemy)
{
self->monsterinfo.currentmove = &actor_move_run;
if (self->monsterinfo.aiflags & AI_CROUCH)
{
self->monsterinfo.aiflags &= ~AI_CROUCH;
self->maxs[2] += 28;
self->viewheight += 28;
self->move_origin[2] += 28;
}
}
else if (R <= ACTOR_FOLLOW_STAND_RANGE && self->movetarget->client)
{
self->monsterinfo.pausetime = level.time + 0.5;
if (self->monsterinfo.aiflags & AI_CROUCH)
self->monsterinfo.currentmove = &actor_move_crouch;
else
self->monsterinfo.currentmove = &actor_move_stand;
}
else
{
if (self->monsterinfo.aiflags & AI_CROUCH)
self->monsterinfo.currentmove = &actor_move_crouchwalk;
else
self->monsterinfo.currentmove = &actor_move_walk;
}
}
else
{
if (self->monsterinfo.aiflags & AI_CROUCH)
self->monsterinfo.currentmove = &actor_move_crouchwalk;
else
self->monsterinfo.currentmove = &actor_move_walk;
}
}
mframe_t actor_frames_walk_back [] =
{
ai_walk, -10, NULL,
ai_walk, -10, NULL,
ai_walk, -10, NULL,
ai_walk, -10, NULL,
ai_walk, -10, NULL,
ai_walk, -10, NULL,
ai_walk, -10, NULL,
ai_walk, -10, NULL,
ai_walk, -10, NULL,
ai_walk, -10, NULL,
ai_walk, -10, NULL,
ai_walk, -10, NULL
};
mmove_t actor_move_walk_back = {FRAME_run1, FRAME_run6, actor_frames_walk_back, NULL};
mframe_t actor_frames_crouchwalk_back [] =
{
ai_walk, -10, NULL,
ai_walk, -10, NULL,
ai_walk, -10, NULL,
ai_walk, -10, NULL,
ai_walk, -10, NULL,
ai_walk, -10, NULL
};
mmove_t actor_move_crouchwalk_back = {FRAME_crwalk1, FRAME_crwalk6, actor_frames_crouchwalk_back, NULL};
void actor_walk_back (edict_t *self)
{
// prevent foolishness:
if (self->monsterinfo.aiflags & AI_FOLLOW_LEADER)
{
if (!self->movetarget || !self->movetarget->inuse || (self->movetarget == world))
self->movetarget = self->monsterinfo.leader;
}
if ( (self->monsterinfo.aiflags & AI_FOLLOW_LEADER) &&
(self->movetarget) &&
(self->movetarget->inuse) &&
(self->movetarget->health > 0) )
{
float R;
R = realrange(self,self->movetarget);
if (R <= ACTOR_FOLLOW_STAND_RANGE && self->movetarget->client)
{
self->monsterinfo.pausetime = level.time + 0.5;
if (self->monsterinfo.aiflags & AI_CROUCH)
self->monsterinfo.currentmove = &actor_move_crouch;
else
self->monsterinfo.currentmove = &actor_move_stand;
}
else
{
if (self->monsterinfo.aiflags & AI_CROUCH)
self->monsterinfo.currentmove = &actor_move_crouchwalk_back;
else
self->monsterinfo.currentmove = &actor_move_walk_back;
}
}
else
{
if (self->monsterinfo.aiflags & AI_CROUCH)
self->monsterinfo.currentmove = &actor_move_crouchwalk_back;
else
self->monsterinfo.currentmove = &actor_move_walk_back;
}
}
mframe_t actor_frames_crouch [] =
{
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL,
ai_stand, 0, NULL
};
mmove_t actor_move_crouch = {FRAME_crstnd01, FRAME_crstnd19, actor_frames_crouch, NULL};
mframe_t actor_frames_crouchwalk [] =
{
ai_walk, 10, NULL,
ai_walk, 10, NULL,
ai_walk, 10, NULL,
ai_walk, 10, NULL,
ai_walk, 10, NULL,
ai_walk, 10, NULL
};
mmove_t actor_move_crouchwalk = {FRAME_crwalk1, FRAME_crwalk6, actor_frames_crouchwalk, NULL};
// DWH: Changed running speed to a constant (equal to player running speed) for normal
// misc_actor and 2/3 that for "bad guys". Also eliminated excess frames.
mframe_t actor_frames_run [] =
{
ai_run, 40, NULL,
ai_run, 40, NULL,
ai_run, 40, NULL,
ai_run, 40, NULL,
ai_run, 40, NULL,
ai_run, 40, NULL
};
mmove_t actor_move_run = {FRAME_run1, FRAME_run6, actor_frames_run, NULL};
mframe_t actor_frames_run_bad [] =
{
ai_run, 30, NULL,
ai_run, 30, NULL,
ai_run, 30, NULL,
ai_run, 30, NULL,
ai_run, 30, NULL,
ai_run, 30, NULL
};
mmove_t actor_move_run_bad = {FRAME_run1, FRAME_run6, actor_frames_run_bad, NULL};
void actor_run (edict_t *self)
{
// prevent foolishness:
if (self->monsterinfo.aiflags & AI_FOLLOW_LEADER)
{
if (!self->movetarget || !self->movetarget->inuse || (self->movetarget == world))
self->movetarget = self->monsterinfo.leader;
}
if ((level.time < self->pain_debounce_time) && (!self->enemy))
{
if (self->movetarget)
actor_walk(self);
else
actor_stand(self);
return;
}
if ( self->monsterinfo.aiflags & AI_STAND_GROUND )
{
actor_stand(self);
return;
}
if ( self->monsterinfo.aiflags & AI_CROUCH)
{
self->monsterinfo.aiflags &= ~AI_CROUCH;
self->maxs[2] += 28;
self->viewheight += 28;
self->move_origin[2] += 28;
}
if ( self->monsterinfo.aiflags & AI_GOOD_GUY ) {
self->monsterinfo.currentmove = &actor_move_run;
}
else
{
self->monsterinfo.currentmove = &actor_move_run_bad;
}
}
mframe_t actor_frames_run_back [] =
{
ai_run, -40, NULL,
ai_run, -40, NULL,
ai_run, -40, NULL,
ai_run, -40, NULL,
ai_run, -40, NULL,
ai_run, -40, NULL
};
mmove_t actor_move_run_back = {FRAME_run1, FRAME_run6, actor_frames_run_back, NULL};
void actor_run_back (edict_t *self)
{
// prevent foolishness:
if (self->monsterinfo.aiflags & AI_FOLLOW_LEADER)
{
if (!self->movetarget || !self->movetarget->inuse || (self->movetarget == world))
self->movetarget = self->monsterinfo.leader;
}
if ((level.time < self->pain_debounce_time) && (!self->enemy))
{
if (self->movetarget)
actor_walk_back(self);
else
actor_stand(self);
return;
}
if ( self->monsterinfo.aiflags & AI_STAND_GROUND )
{
actor_stand(self);
return;
}
if ( self->monsterinfo.aiflags & AI_CROUCH)
{
self->monsterinfo.aiflags &= ~AI_CROUCH;
self->maxs[2] += 28;
self->viewheight += 28;
self->move_origin[2] += 28;
}
self->monsterinfo.currentmove = &actor_move_run_back;
}
mframe_t actor_frames_pain1 [] =
{
ai_move, -5, NULL,
ai_move, 4, NULL,
ai_move, 1, NULL,
ai_move, 1, NULL
};
mmove_t actor_move_pain1 = {FRAME_pain101, FRAME_pain104, actor_frames_pain1, actor_run};
mframe_t actor_frames_pain2 [] =
{
ai_move, -4, NULL,
ai_move, 4, NULL,
ai_move, 0, NULL,
ai_move, 0, NULL
};
mmove_t actor_move_pain2 = {FRAME_pain201, FRAME_pain204, actor_frames_pain2, actor_run};
mframe_t actor_frames_pain3 [] =
{
ai_move, -1, NULL,
ai_move, 1, NULL,
ai_move, 0, NULL,
ai_move, 1, NULL
};
mmove_t actor_move_pain3 = {FRAME_pain301, FRAME_pain304, actor_frames_pain3, actor_run};
mframe_t actor_frames_flipoff [] =
{
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL
};
mmove_t actor_move_flipoff = {FRAME_flip01, FRAME_flip12, actor_frames_flipoff, actor_run};
mframe_t actor_frames_taunt [] =
{
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL
};
mmove_t actor_move_taunt = {FRAME_taunt01, FRAME_taunt17, actor_frames_taunt, actor_run};
void actor_ideal_range(edict_t *self)
{
int weapon;
weapon = self->actor_weapon[self->actor_current_weapon];
switch (weapon)
{
case 2:
self->monsterinfo.ideal_range[0] = 0;
self->monsterinfo.ideal_range[1] = 270;
break;
case 3:
self->monsterinfo.ideal_range[0] = 0;
self->monsterinfo.ideal_range[1] = 90;
break;
case 4:
case 5:
self->monsterinfo.ideal_range[0] = 0;
self->monsterinfo.ideal_range[1] = 450;
break;
case 6:
self->monsterinfo.ideal_range[0] = 200;
self->monsterinfo.ideal_range[1] = 450;
break;
case 7:
self->monsterinfo.ideal_range[0] = 300;
self->monsterinfo.ideal_range[1] = 1000;
break;
case 8:
self->monsterinfo.ideal_range[0] = 200;
self->monsterinfo.ideal_range[1] = 500;
break;
case 9:
case 10:
self->monsterinfo.ideal_range[0] = 300;
self->monsterinfo.ideal_range[1] = 1000;
break;
default:
self->monsterinfo.ideal_range[0] = 0;
self->monsterinfo.ideal_range[1] = 0;
}
}
void actor_attack(edict_t *self);
void actor_switch (edict_t *self)
{
self->actor_current_weapon = 1 - self->actor_current_weapon;
self->s.modelindex2 = self->actor_model_index[self->actor_current_weapon];
actor_ideal_range(self);
gi.linkentity(self);
}
mframe_t actor_frames_switch [] =
{
ai_run, 0, actor_switch,
ai_run, 0, NULL,
ai_run, 0, NULL
};
mmove_t actor_move_switch = {FRAME_jump4, FRAME_jump6, actor_frames_switch, actor_attack};
void actor_pain (edict_t *self, edict_t *other, float kick, int damage)
{
int n;
int r, l;
// DWH: Players don't have pain skins!
// if (self->health < (self->max_health / 2))
// self->s.skinnum = 1;
// Stop weapon sound, if any
self->s.sound = 0;
if (level.time < self->pain_debounce_time)
return;
self->pain_debounce_time = level.time + 1;
// DWH: Use same scheme used for player pain sounds
if (!(self->flags & FL_GODMODE))
{
r = (rand()&1);
if (self->health < 25)
l = 0;
else if (self->health < 50)
l = 2;
else if (self->health < 75)
l = 4;
else
l = 6;
gi.sound (self, CHAN_VOICE, self->actor_sound_index[ACTOR_SOUND_PAIN_25_1 + l + r],
1, ATTN_NORM, 0);
}
// Lazarus: Removed printed message, but keep taunt (not for monster actors, though)
if ((other->client) && (random() < 0.4) && (self->monsterinfo.aiflags & AI_GOOD_GUY))
{
vec3_t v;
VectorSubtract (other->s.origin, self->s.origin, v);
self->ideal_yaw = vectoyaw (v);
if (random() < 0.5) {
self->monsterinfo.currentmove = &actor_move_flipoff;
}
else
{
self->monsterinfo.currentmove = &actor_move_taunt;
}
return;
}
n = rand() % 3;
if (n == 0)
self->monsterinfo.currentmove = &actor_move_pain1;
else if (n == 1)
self->monsterinfo.currentmove = &actor_move_pain2;
else
self->monsterinfo.currentmove = &actor_move_pain3;
}
//
// Attack code moved to m_actor_weap.c
//
void actor_dead (edict_t *self)
{
VectorSet (self->mins, -16, -16, -24);
VectorSet (self->maxs, 16, 16, -8);
self->movetype = MOVETYPE_TOSS;
self->svflags |= SVF_DEADMONSTER;
self->nextthink = 0;
gi.linkentity (self);
M_FlyCheck (self);
// Lazarus monster fade
if (world->effects & FX_WORLDSPAWN_CORPSEFADE)
{
self->think=FadeDieSink;
self->nextthink=level.time+corpse_fadetime->value;
}
}
mframe_t actor_frames_death1 [] =
{
ai_move, 0, NULL,
ai_move, 0, NULL,
ai_move, -13, NULL,
ai_move, 14, NULL,
ai_move, 3, NULL,
ai_move, -2, NULL
};
mmove_t actor_move_death1 = {FRAME_death101, FRAME_death106, actor_frames_death1, actor_dead};
mframe_t actor_frames_death2 [] =
{
ai_move, 0, NULL,
ai_move, 7, NULL,
ai_move, -6, NULL,
ai_move, -5, NULL,
ai_move, 1, NULL,
ai_move, 0, NULL
};
mmove_t actor_move_death2 = {FRAME_death201, FRAME_death206, actor_frames_death2, actor_dead};
void actor_die (edict_t *self, edict_t *inflictor, edict_t *attacker, int damage, vec3_t point)
{
int n;
//Remove the weapon model and turn off weapon sound, if any
self->s.modelindex2 = 0;
self->s.sound = 0;
// check for gib
if (self->health <= self->gib_health && !(self->spawnflags & SF_MONSTER_NOGIB))
{
gi.sound (self, CHAN_BODY, gi.soundindex ("misc/udeath.wav"), 1, ATTN_NORM, 0);
for (n= 0; n < 2; n++)
ThrowGib (self, "models/objects/gibs/bone/tris.md2", damage, GIB_ORGANIC);
for (n= 0; n < 4; n++)
ThrowGib (self, "models/objects/gibs/sm_meat/tris.md2", damage, GIB_ORGANIC);
ThrowHead (self, "models/objects/gibs/head2/tris.md2", damage, GIB_ORGANIC);
self->deadflag = DEAD_DEAD;
return;
}
if (self->deadflag == DEAD_DEAD)
return;
// regular death
gi.sound (self, CHAN_VOICE,
self->actor_sound_index[ACTOR_SOUND_DEATH1 + (rand()%4)], 1, ATTN_NORM, 0);
self->deadflag = DEAD_DEAD;
self->takedamage = DAMAGE_YES;
if (self->monsterinfo.aiflags & AI_CHASE_THING)
{
if (self->movetarget && !Q_stricmp(self->movetarget->classname,"thing"))
{
G_FreeEdict(self->movetarget);
self->movetarget = NULL;
}
}
self->monsterinfo.aiflags &= ~(AI_FOLLOW_LEADER | AI_CHASE_THING | AI_CHICKEN | AI_EVADE_GRENADE);
if (random() > 0.5)
self->monsterinfo.currentmove = &actor_move_death1;
else
self->monsterinfo.currentmove = &actor_move_death2;
}
void actor_fire (edict_t *self)
{
int weapon;
weapon = self->actor_weapon[self->actor_current_weapon];
switch (weapon)
{
case 1:
actorBlaster (self);
break;
case 2:
actorShotgun (self);
break;
case 3:
actorSuperShotgun (self);
break;
case 4:
actorMachineGun (self);
if (level.time >= self->monsterinfo.pausetime)
// self->monsterinfo.aiflags &= ~(AI_HOLD_FRAME|AI_STAND_GROUND);
self->monsterinfo.aiflags &= ~AI_HOLD_FRAME;
else
// self->monsterinfo.aiflags |= (AI_HOLD_FRAME|AI_STAND_GROUND);
self->monsterinfo.aiflags |= AI_HOLD_FRAME;
break;
case 5:
actorChaingun (self);
if (level.time >= self->monsterinfo.pausetime)
self->monsterinfo.aiflags &= ~AI_HOLD_FRAME;
else
self->monsterinfo.aiflags |= AI_HOLD_FRAME;
break;
case 6:
actorGrenadeLauncher (self);
break;
case 7:
actorRocket(self);
break;
case 8:
actorHyperblaster(self);
if (level.time >= self->monsterinfo.pausetime)
self->monsterinfo.aiflags &= ~AI_HOLD_FRAME;
else
self->monsterinfo.aiflags |= AI_HOLD_FRAME;
break;
case 9:
actorRailGun(self);
break;
case 10:
actorBFG(self);
if (level.time >= self->monsterinfo.pausetime)
self->monsterinfo.aiflags &= ~AI_HOLD_FRAME;
else
self->monsterinfo.aiflags |= AI_HOLD_FRAME;
break;
}
}
void actor_no_weapon_sound(edict_t *self)
{
self->s.sound = 0;
gi.linkentity(self);
}
static int chase_angle[] = {360,315,405,270,450,225,495,540};
void actor_seekcover (edict_t *self)
{
int i;
edict_t *thing;
vec3_t atk, dir, best_dir, end, forward;
vec_t travel, yaw;
vec3_t mins, maxs;
vec3_t testpos;
vec_t best_dist=0;
trace_t trace1, trace2;
// No point in hiding from enemy if.. we don't have an enemy
if (!self->enemy || !self->enemy->inuse)
{
actor_run(self);
return;
}
if (!actorscram->value)
{
actor_run(self);
return;
}
// Don't hide from non-humanoid stuff
if (!self->enemy->client && !(self->enemy->svflags & SVF_MONSTER))
{
actor_run(self);
return;
}
// This shouldn't happen, we're just being cautious. Quit now if
// already chasing a "thing"
if (self->movetarget && !Q_stricmp(self->movetarget->classname,"thing"))
{
actor_run(self);
return;
}
// Don't bother finding cover if we're within melee range of enemy
VectorSubtract(self->enemy->s.origin,self->s.origin,atk);
if (VectorLength(atk) < 80)
{
actor_run(self);
return;
}
VectorCopy(self->mins,mins);
mins[2] += 18;
if (mins[2] > 0) mins[2] = 0;
VectorCopy(self->maxs,maxs);
// Find a vector that will hide the actor from his enemy
VectorCopy(self->enemy->s.origin,atk);
atk[2] += self->enemy->viewheight;
VectorClear(best_dir);
AngleVectors(self->s.angles,forward,NULL,NULL);
dir[2] = 0;
for (travel=64; travel<257 && best_dist == 0; travel *= 2)
{
for (i=0; i<8 && best_dist == 0; i++)
{
yaw = self->s.angles[YAW] + chase_angle[i];
yaw = (int)(yaw/45)*45;
yaw = anglemod(yaw);
yaw *= M_PI/180;
dir[0] = cos(yaw);
dir[1] = sin(yaw);
VectorMA(self->s.origin,travel,dir,end);
trace1 = gi.trace(self->s.origin,mins,maxs,end,self,MASK_MONSTERSOLID);
// Test whether proposed position can be seen by enemy. Test
// isn't foolproof - tests against 1) new origin, 2-5) each corner of top
// of bounding box.
trace2 = gi.trace(trace1.endpos,NULL,NULL,atk,self,MASK_SOLID);
if (trace2.fraction == 1.0) continue;
VectorAdd(trace1.endpos,self->maxs,testpos);
trace2 = gi.trace(testpos,NULL,NULL,atk,self,MASK_SOLID);
if (trace2.fraction == 1.0) continue;
testpos[0] = trace1.endpos[0] + self->mins[0];
trace2 = gi.trace(testpos,NULL,NULL,atk,self,MASK_SOLID);
if (trace2.fraction == 1.0) continue;
testpos[1] = trace1.endpos[1] + self->mins[1];
trace2 = gi.trace(testpos,NULL,NULL,atk,self,MASK_SOLID);
if (trace2.fraction == 1.0) continue;
testpos[0] = trace1.endpos[0] + self->maxs[0];
trace2 = gi.trace(testpos,NULL,NULL,atk,self,MASK_SOLID);
if (trace2.fraction == 1.0) continue;
best_dist = trace1.fraction * travel;
if (best_dist < 32) // not much point to this move
continue;
VectorCopy(dir,best_dir);
}
}
if (best_dist < 32)
{
actor_run(self);
return;
}
// This snaps the angles, which may not be all that good but it sure
// is quicker than turning in place
vectoangles(best_dir,self->s.angles);
thing = SpawnThing();
VectorMA(self->s.origin,best_dist,best_dir,thing->s.origin);
thing->touch_debounce_time = level.time + 3.0;
thing->target_ent = self;
ED_CallSpawn(thing);
self->movetarget = self->goalentity = thing;
self->monsterinfo.aiflags &= ~(AI_SOUND_TARGET | AI_STAND_GROUND | AI_TEMP_STAND_GROUND);
self->monsterinfo.aiflags |= (AI_SEEK_COVER | AI_CHASE_THING);
gi.linkentity(self);
actor_run(self);
}
mframe_t actor_frames_attack [] =
{
ai_charge, 0, actor_fire,
ai_charge, 0, actor_no_weapon_sound,
ai_charge, 0, NULL,
ai_charge, 0, NULL,
ai_charge, 0, NULL,
ai_charge, 0, NULL,
ai_charge, 0, NULL,
ai_charge, 0, NULL
};
mmove_t actor_move_attack = {FRAME_attack1, FRAME_attack8, actor_frames_attack, actor_seekcover};
mframe_t actor_frames_crattack [] =
{
ai_charge, 0, actor_fire,
ai_charge, 0, actor_no_weapon_sound,
ai_charge, 0, NULL,
ai_charge, 0, NULL,
ai_charge, 0, NULL,
ai_charge, 0, NULL,
ai_charge, 0, NULL,
ai_charge, 0, NULL,
ai_charge, 0, NULL
};
mmove_t actor_move_crattack = {FRAME_crattak1, FRAME_crattak9, actor_frames_crattack, actor_run};
void actor_attack (edict_t *self)
{
int n;
int weapon, w_select;
mmove_t *attackmove;
vec3_t v;
w_select = self->actor_current_weapon;
weapon = self->actor_weapon[w_select];
if (self->enemy)
{
if ( w_select == 0 && self->actor_weapon[1] > 0 )
{
VectorSubtract(self->s.origin,self->enemy->s.origin,v);
if (VectorLength(v) < 200)
{
self->monsterinfo.currentmove = &actor_move_switch;
return;
}
}
else if ( w_select == 1 && self->actor_weapon[0] > 0 )
{
VectorSubtract(self->s.origin,self->enemy->s.origin,v);
if (VectorLength(v) > 300)
{
self->monsterinfo.currentmove = &actor_move_switch;
return;
}
}
}
self->actor_gunframe = 0;
// temporary deal to toggle crouch
/* if (self->actor_crouch_time < level.time)
self->actor_crouch_time = level.time + 5;
else
self->actor_crouch_time = 0; */
// end temp
if (self->actor_crouch_time < level.time)
attackmove = &actor_move_attack;
else
attackmove = &actor_move_crattack;
switch (weapon)
{
case 1:
self->monsterinfo.currentmove = attackmove;
self->monsterinfo.pausetime = level.time + 2 * FRAMETIME;
break;
case 2:
self->monsterinfo.currentmove = attackmove;
self->monsterinfo.pausetime = level.time + 6 * FRAMETIME;
break;
case 3:
self->monsterinfo.currentmove = attackmove;
self->monsterinfo.pausetime = level.time + 10 * FRAMETIME;
break;
case 4:
self->monsterinfo.currentmove = attackmove;
n = (rand() & 15) + 3 + 7;
self->monsterinfo.pausetime = level.time + n * FRAMETIME;
break;
case 5:
self->monsterinfo.currentmove = attackmove;
n = (rand() & 20) + 20;
self->monsterinfo.pausetime = level.time + n * FRAMETIME;
break;
case 6:
case 7:
self->monsterinfo.currentmove = attackmove;
if (self->monsterinfo.aiflags & AI_STAND_GROUND)
{ // if hes just standing there refire rate is normal
self->monsterinfo.pausetime = level.time + 7;
}
else
{ // otherwise, allow the target to fire back
self->monsterinfo.pausetime = level.time + 2;
}
break;
case 8:
self->monsterinfo.currentmove = attackmove;
n = (rand() & 15) + 3 + 7;
self->monsterinfo.pausetime = level.time + n * FRAMETIME;
break;
case 9:
self->monsterinfo.currentmove = attackmove;
self->monsterinfo.pausetime = level.time + 3;
break;
case 10:
if (level.time > self->endtime)
{
self->monsterinfo.currentmove = attackmove;
self->monsterinfo.pausetime = level.time + 1.5;
}
else
self->monsterinfo.currentmove = &actor_move_stand;
break;
}
}
void actor_use (edict_t *self, edict_t *other, edict_t *activator)
{
vec3_t v;
self->goalentity = self->movetarget = G_PickTarget(self->target);
if ((!self->movetarget) || (strcmp(self->movetarget->classname, "target_actor") != 0))
{
gi.dprintf ("%s has bad target %s at %s\n", self->classname, self->target, vtos(self->s.origin));
self->target = NULL;
self->monsterinfo.pausetime = 100000000;
self->monsterinfo.stand (self);
return;
}
VectorSubtract (self->goalentity->s.origin, self->s.origin, v);
self->ideal_yaw = self->s.angles[YAW] = vectoyaw(v);
self->monsterinfo.walk (self);
self->target = NULL;
if (self->pathtarget)
{
char *savetarget;
savetarget = self->target;
self->target = self->pathtarget;
G_UseTargets (self, other);
self->target = savetarget;
}
}
// Lazarus: checkattack - higher probabilities than normal monsters,
// also weapon-based
static float chancefar[11] = { 0.0, 0.2, 0.0, 0.0, 0.0, 0.0, 0.2, 0.4, 0.2, 0.5, 0.8 };
static float chancenear[11] = { 0.0, 0.4, 0.0, 0.0, 0.0, 0.0, 0.4, 0.4, 0.4, 0.4, 0.4 };
qboolean actor_checkattack (edict_t *self)
{
vec3_t v;
vec3_t forward, right, start, end;
float chance;
float range;
float goodchance, poorchance, lorange, hirange;
trace_t tr;
int weapon;
// Paranoia check
if (!self->enemy)
return false;
// If running to "thing", never attack
if (self->monsterinfo.aiflags & AI_CHASE_THING)
return false;
weapon = self->actor_weapon[self->actor_current_weapon];
// If actor has no weapon, well then of course he should not attack
if (weapon < 1 || weapon > 10)
return false;
if (self->enemy->health > 0)
{
// see if any entities are in the way of the shot
AngleVectors (self->s.angles, forward, right, NULL);
G_ProjectSource (self->s.origin, self->muzzle, forward, right, start);
VectorCopy (self->enemy->s.origin, end);
tr = gi.trace (start, NULL, NULL, end, self, CONTENTS_SOLID|CONTENTS_MONSTER|CONTENTS_SLIME|CONTENTS_LAVA|CONTENTS_WINDOW);
// do we have a clear shot?
if (tr.ent != self->enemy) {
return false;
}
}
VectorSubtract (self->s.origin, self->enemy->s.origin, v);
range = VectorLength (v);
// melee attack
if (range <= MELEE_DISTANCE)
{
// don't always melee in easy mode
if (skill->value == 0 && (rand()&3) )
return false;
self->monsterinfo.attack_state = AS_MISSILE;
return true;
}
// missile attack
if (!self->monsterinfo.attack)
return false;
if (level.time < self->monsterinfo.attack_finished)
return false;
if (range > self->monsterinfo.max_range)
return false;
if (self->monsterinfo.aiflags & AI_STAND_GROUND)
{
chance = 0.4;
}
else
{
if (weapon >= 2 && weapon <= 5)
{
// Scatter guns - probability of firing based on percentage of rounds
// that will hit target at a given range.
if (skill->value == 1)
goodchance = 0.6;
else if (skill->value > 1)
goodchance = 0.9;
else
goodchance = 0.3;
poorchance = 0.01;
switch (weapon)
{
case 2: lorange=270; hirange=500; break;
case 3: lorange= 90; hirange=200; break;
case 4: lorange=450; hirange=628; break;
case 5: lorange=450; hirange=628; break;
}
if (range <= lorange)
chance = goodchance;
else if (range > hirange)
chance = poorchance;
else
chance = goodchance + (range-lorange)/(hirange-lorange)*(poorchance-goodchance);
}
else
{
if (range <= 500)
chance = chancenear[weapon];
else
chance = chancefar[weapon];
if (self->monsterinfo.aiflags & AI_GOOD_GUY)
{
if (skill->value == 0)
chance *= 2;
else if (skill->value == 2)
chance *= 0.5;
else if (skill->value == 3)
chance *= 0.25;
}
else
{
if (skill->value == 0)
chance *= 0.5;
else if (skill->value == 2)
chance *= 2;
else if (skill->value == 3)
chance *= 4;
}
}
}
if (random () < chance)
{
self->monsterinfo.attack_state = AS_MISSILE;
self->monsterinfo.attack_finished = level.time + 2*random();
return true;
}
return false;
}
mmove_t actor_move_jump;
void actor_end_jump (edict_t *self)
{
if (self->flags & FL_ROBOT)
{
if (self->monsterinfo.savemove)
{
actor_run(self);
// self->monsterinfo.currentmove = self->monsterinfo.savemove;
/* gi.dprintf("savemove=%d\n",self->monsterinfo.currentmove);
gi.dprintf("actor_move_jump=%d\n",&actor_move_jump);
gi.dprintf("actor_move_run=%d\n",&actor_move_run);
gi.dprintf("actor_move_walk=%d\n",&actor_move_walk);
gi.dprintf("actor_move_stand=%d\n",&actor_move_stand); */
}
else if (self->enemy)
actor_run(self);
else if (self->movetarget)
actor_walk(self);
else
actor_stand(self);
}
else
actor_run(self);
}
mframe_t actor_frames_jump [] =
{
ai_move, 0, NULL,
ai_move, 0, NULL,
ai_move, 0, NULL,
ai_move, 0, NULL,
ai_move, 0, NULL,
ai_move, 0, actor_end_jump
};
mmove_t actor_move_jump = {FRAME_jump1, FRAME_jump6, actor_frames_jump, actor_jump};
void actor_jump (edict_t *self)
{
gi.sound (self, CHAN_VOICE, self->actor_sound_index[ACTOR_SOUND_JUMP], 1, ATTN_NORM, 0);
self->monsterinfo.currentmove = &actor_move_jump;
}
qboolean actor_blocked (edict_t *self, float dist)
{
if (check_shot_blocked (self, 0.25 + (0.05 * skill->value) ))
return true;
if (check_jump_blocked (self, dist, self->monsterinfo.jumpdn, self->monsterinfo.jumpup))
return true;
if (check_plat_blocked (self, dist))
return true;
return false;
}
mframe_t actor_frames_salute [] =
{
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL,
ai_turn, 0, NULL
};
mmove_t actor_move_salute = {FRAME_salute01, FRAME_salute11, actor_frames_salute, actor_run};
void actor_salute (edict_t *self)
{
self->monsterinfo.currentmove = &actor_move_salute;
}
/*QUAKED misc_actor (1 .5 0) (-16 -16 -24) (16 16 32)
*/
#define ACTOR_ALIEN 1
#define ACTOR_HUNTER 2
#define ACTOR_PARANOID 3
#define ACTOR_RATAMAHATTA 4
#define ACTOR_RHINO 5
#define ACTOR_SAS 6
#define ACTOR_SLITH 7
#define ACTOR_TERRAN 8
#define ACTOR_WALKER 9
#define ACTOR_WASTE 10
#define ACTOR_XENOID 11
#define ACTOR_ZUMLIN 12
#define NUM_ACTORPAK_ACTORS 12
char ActorNames[NUM_ACTORPAK_ACTORS][32] =
{ "alien", "hunter", "paranoid","ratamahatta",
"rhino", "sas", "slith", "terran",
"walker", "waste", "xenoid", "zumlin" };
void SP_misc_actor (edict_t *self)
{
char modelpath[256];
char *p;
int i;
int ActorID = 0;
if (deathmatch->value)
{
G_FreeEdict (self);
return;
}
self->class_id = ENTITY_MISC_ACTOR;
self->movetype = MOVETYPE_STEP;
self->solid = SOLID_BBOX;
if (self->usermodel) {
p = strstr(self->usermodel,"/tris.md2");
if (p) *p = 0;
}
else {
self->usermodel = gi.TagMalloc(5, TAG_LEVEL);
// strncpy(self->usermodel,"male");
Q_strncpyz(self->usermodel,"male", 5);
}
if ( (!Q_stricmp(self->usermodel,"male")) ||
(!Q_stricmp(self->usermodel,"female")) ||
(!Q_stricmp(self->usermodel,"cyborg")) )
{
self->actor_id_model = true;
if (PatchPlayerModels(self->usermodel))
level.restart_for_actor_models = true;
}
else
self->actor_id_model = false;
Com_sprintf(modelpath, sizeof(modelpath), "players/%s/tris.md2",self->usermodel);
self->s.modelindex = gi.modelindex(modelpath);
for (i=0; i<NUM_ACTORPAK_ACTORS && !ActorID; i++)
{
if (!Q_stricmp(self->usermodel,ActorNames[i]))
ActorID = i+1;
}
if (!VectorLength(self->bleft) && !VectorLength(self->tright))
{
switch (ActorID)
{
case ACTOR_ALIEN:
VectorSet (self->mins, -28, -28, -24);
VectorSet (self->maxs, 28, 28, 32);
break;
case ACTOR_HUNTER:
VectorSet (self->mins, -24, -24, -24);
VectorSet (self->maxs, 24, 24, 32);
break;
case ACTOR_RATAMAHATTA:
case ACTOR_TERRAN:
VectorSet (self->mins, -20, -20, -24);
VectorSet (self->maxs, 20, 20, 32);
break;
case ACTOR_RHINO:
VectorSet (self->mins, -30, -30, -24);
VectorSet (self->maxs, 30, 30, 32);
break;
case ACTOR_SAS:
case ACTOR_XENOID:
case ACTOR_ZUMLIN:
VectorSet (self->mins, -18, -18, -24);
VectorSet (self->maxs, 18, 18, 32);
break;
case ACTOR_WALKER:
VectorSet (self->mins, -24, -24, -24);
VectorSet (self->maxs, 24, 24, 30);
break;
default:
VectorSet (self->mins, -16, -16, -24);
VectorSet (self->maxs, 16, 16, 32);
}
}
else
{
VectorCopy (self->bleft, self->mins);
VectorCopy (self->tright, self->maxs);
}
if (!self->health)
self->health = 100;
if (!self->gib_health)
self->gib_health = -40;
if (!self->mass)
self->mass = 200;
if (self->sounds < 0)
{
self->actor_weapon[0] = 0;
self->actor_weapon[1] = -self->sounds;
}
else if (self->sounds < 10)
{
self->actor_weapon[0] = self->sounds;
self->actor_weapon[1] = 0;
}
else
{
self->actor_weapon[0] = self->sounds/100;
self->actor_weapon[1] = self->sounds % 100;
}
if (!VectorLength(self->muzzle))
{
switch (ActorID)
{
case ACTOR_ALIEN:
VectorSet(self->muzzle,42,5,15);
break;
case ACTOR_HUNTER:
switch (self->actor_weapon[0])
{
case 1: VectorSet(self->muzzle,32,5,15);break;
case 2: VectorSet(self->muzzle,36,5,15);break;
case 3: VectorSet(self->muzzle,36,5,15);break;
case 4: VectorSet(self->muzzle,38,4,19);break;
case 5: VectorSet(self->muzzle,45,4.5,15);break;
case 6: VectorSet(self->muzzle,32,5,15);break;
case 7: VectorSet(self->muzzle,40,5,15);break;
case 8: VectorSet(self->muzzle,41,4,19);break;
case 9: VectorSet(self->muzzle,40,4,19);break;
case 10: VectorSet(self->muzzle,42,5,20);break;
}
break;
case ACTOR_PARANOID:
switch (self->actor_weapon[0])
{
case 1: VectorSet(self->muzzle,18,7,10);break;
case 2: VectorSet(self->muzzle,22,7,10);break;
case 3: VectorSet(self->muzzle,22,7,10);break;
case 4: VectorSet(self->muzzle,18,7,12);break;
case 5: VectorSet(self->muzzle,26,7,16);break;
case 6: VectorSet(self->muzzle,24,7,10);break;
case 7: VectorSet(self->muzzle,26,7,10);break;
case 8: VectorSet(self->muzzle,18,7,14);break;
case 9: VectorSet(self->muzzle,28,7,10);break;
case 10: VectorSet(self->muzzle,28,7,10);break;
}
break;
case ACTOR_RATAMAHATTA:
VectorSet(self->muzzle,24,13,10);
break;
case ACTOR_RHINO:
VectorSet(self->muzzle,29,7,10);
break;
case ACTOR_SAS:
VectorSet(self->muzzle,17,6.5,17);
break;
case ACTOR_SLITH:
switch (self->actor_weapon[0])
{
case 1: VectorSet(self->muzzle,32,7,10);break;
case 2: VectorSet(self->muzzle,32,7,10);break;
case 3: VectorSet(self->muzzle,32,7,10);break;
case 4: VectorSet(self->muzzle,25,5,-1);break;
case 5: VectorSet(self->muzzle,25,5,-1);break;
case 6: VectorSet(self->muzzle,32,7,10);break;
case 7: VectorSet(self->muzzle,32,7,10);break;
case 8: VectorSet(self->muzzle,12,6,-1);break;
case 9: VectorSet(self->muzzle,32,7,10);break;
case 10: VectorSet(self->muzzle,20,5,-1);break;
}
break;
case ACTOR_TERRAN:
VectorSet(self->muzzle,42,7,11.5);
break;
case ACTOR_WALKER:
VectorSet(self->muzzle,9,16,7);
break;
case ACTOR_WASTE:
switch (self->actor_weapon[0])
{
case 1: VectorSet(self->muzzle,12, 9,9);break;
case 2: VectorSet(self->muzzle,22, 9,9);break;
case 3: VectorSet(self->muzzle,20, 9,9);break;
case 4: VectorSet(self->muzzle,11,11,7);break;
case 5: VectorSet(self->muzzle,26, 8,8);break;
case 6: VectorSet(self->muzzle,18, 9,7);break;
case 7: VectorSet(self->muzzle,26, 9,7);break;
case 8: VectorSet(self->muzzle,26, 7.5,8);break;
case 9: VectorSet(self->muzzle,26, 9,7);break;
case 10: VectorSet(self->muzzle,22,11,7);break;
}
break;
case ACTOR_XENOID:
VectorSet(self->muzzle,20,12,7);
break;
case ACTOR_ZUMLIN:
switch (self->actor_weapon[0])
{
case 1: VectorSet(self->muzzle,22, 3,8);break;
case 2: VectorSet(self->muzzle,20, 2,9);break;
case 3: VectorSet(self->muzzle,20, 2,9);break;
case 4: VectorSet(self->muzzle, 8, 5,4);break;
case 5: VectorSet(self->muzzle,22, 2,4);break;
case 6: VectorSet(self->muzzle,20, 2,7);break;
case 7: VectorSet(self->muzzle,30, 2,9);break;
case 8: VectorSet(self->muzzle,20, 3,2);break;
case 9: VectorSet(self->muzzle,26, 2,9);break;
case 10: VectorSet(self->muzzle,16, 5,-2);break;
}
break;
default:
switch (self->actor_weapon[0])
{
case 4: VectorSet(self->muzzle, 6 ,9 ,6 );break;
case 5: VectorSet(self->muzzle,20 ,9 ,8 );break;
case 8: VectorSet(self->muzzle,18 ,8 ,6 );break;
default:VectorSet(self->muzzle,18.4,7.4,9.6);break;
}
}
}
if (!VectorLength(self->muzzle2))
{
switch (ActorID)
{
case ACTOR_RHINO:
VectorSet(self->muzzle2,27,-15,13);
break;
case ACTOR_WALKER:
VectorSet(self->muzzle2,9,-11,7);
break;
}
}
if (VectorLength(self->muzzle2))
self->monsterinfo.aiflags |= AI_TWO_GUNS;
self->pain = actor_pain;
self->die = actor_die;
self->monsterinfo.stand = actor_stand;
self->monsterinfo.walk = actor_walk;
self->monsterinfo.run = actor_run;
self->monsterinfo.dodge = NULL;
self->monsterinfo.attack = actor_attack;
self->monsterinfo.melee = NULL;
self->monsterinfo.sight = NULL;
self->monsterinfo.idle = NULL;
self->monsterinfo.checkattack = actor_checkattack;
if (actorjump->value)
{
self->monsterinfo.jump = actor_jump;
self->monsterinfo.jumpup = 48;
self->monsterinfo.jumpdn = 160;
}
// self->monsterinfo.blocked = actor_blocked;
// There are several actions (mainly following a player leader) that
// are only applicable to misc_actor (not other monsters)
self->monsterinfo.aiflags |= AI_ACTOR;
if (!(self->spawnflags & SF_ACTOR_BAD_GUY) || (self->spawnflags & SF_MONSTER_GOODGUY))
self->monsterinfo.aiflags |= AI_GOOD_GUY;
if (self->powerarmor) {
self->monsterinfo.power_armor_type = POWER_ARMOR_SHIELD;
self->monsterinfo.power_armor_power = self->powerarmor;
}
// Minimum distance
if (self->actor_weapon[1])
self->monsterinfo.min_range = 0;
else
{
int weapon;
weapon = self->actor_weapon[0];
if (weapon == 6 || weapon == 7 || weapon == 10)
self->monsterinfo.min_range = 200;
else
self->monsterinfo.min_range = 0;
}
// Ideal range
actor_ideal_range(self);
gi.linkentity (self);
self->monsterinfo.currentmove = &actor_move_stand;
if (self->health < 0)
{
mmove_t *deathmoves[] = {&actor_move_death1,
&actor_move_death2,
NULL};
M_SetDeath(self,(mmove_t **)&deathmoves);
}
self->monsterinfo.scale = 0.8;
walkmonster_start (self);
// We've built the misc_actor model to include the standard
// Q2 male skins, specified with the style key. Default=grunt
self->s.skinnum = self->style;
// actors always start in a dormant state, they *must* be used to get going
self->use = actor_use;
// If health > 100000, actor is invulnerable
if (self->health >= 100000)
self->takedamage = DAMAGE_NO;
self->common_name = "Actor";
// Muzzle flash
self->flash = G_Spawn();
self->flash->classname = "muzzleflash";
self->flash->model = "models/objects/flash/tris.md2";
gi.setmodel(self->flash,self->flash->model);
self->flash->solid = SOLID_NOT;
self->flash->s.skinnum = 0;
self->flash->s.effects = EF_PLASMA;
self->flash->s.renderfx = RF_FULLBRIGHT;
self->flash->svflags |= SVF_NOCLIENT;
VectorCopy(self->s.origin,self->flash->s.origin);
gi.linkentity(self->flash);
}
/*QUAKED target_actor (.5 .3 0) (-8 -8 -8) (8 8 8) JUMP SHOOT ATTACK x HOLD BRUTAL
JUMP jump in set direction upon reaching this target
SHOOT take a single shot at the pathtarget
ATTACK attack pathtarget until it or actor is dead
"target" next target_actor
"pathtarget" target of any action to be taken at this point
"wait" amount of time actor should pause at this point
for JUMP only:
"speed" speed thrown forward (default 200)
"height" speed thrown upwards (default 200)
*/
void target_actor_touch (edict_t *self, edict_t *other, cplane_t *plane, csurface_t *surf)
{
vec3_t v;
if (other->movetarget != self)
return;
if (other->enemy)
return;
other->goalentity = other->movetarget = NULL;
if (self->spawnflags & 1) //jump
{
other->velocity[0] = self->movedir[0] * self->speed;
other->velocity[1] = self->movedir[1] * self->speed;
if (other->groundentity)
{
other->groundentity = NULL;
other->velocity[2] = self->movedir[2];
if (other->monsterinfo.aiflags & AI_ACTOR)
gi.sound (self, CHAN_VOICE, other->actor_sound_index[ACTOR_SOUND_JUMP], 1, ATTN_NORM, 0);
}
// NOTE: The jump animation won't work UNLESS this target_actor has a target. If this
// is the last target_actor in a sequence, the actor's run takes over and prevents
// the jump.
// if (!Q_stricmp(other->classname,"misc_actor"))
// other->monsterinfo.currentmove = &actor_move_jump;
}
if (self->spawnflags & 2) //shoot
{
if (self->pathtarget) {
if ( G_Find(NULL,FOFS(targetname),self->pathtarget) != NULL ) {
other->enemy = G_PickTarget(self->pathtarget);
if (self->spawnflags & 8)
{
other->monsterinfo.aiflags |= AI_STAND_GROUND;
actor_stand (other);
} else
actor_attack(other);
}
else
other->enemy = NULL;
} else {
other->enemy = NULL;
}
}
else if (self->spawnflags & 4) //attack
{
if (self->pathtarget) {
if ( G_Find(NULL,FOFS(targetname),self->pathtarget) != NULL )
other->enemy = G_PickTarget(self->pathtarget);
else
other->enemy = NULL;
} else {
other->enemy = NULL;
}
if (other->enemy)
{
other->goalentity = other->enemy;
if (self->spawnflags & 32)
other->monsterinfo.aiflags |= AI_BRUTAL;
if (self->spawnflags & 16)
{
other->monsterinfo.aiflags |= AI_STAND_GROUND;
actor_stand (other);
}
else
{
actor_run (other);
}
}
}
if (!(self->spawnflags & 6) && (self->pathtarget))
{
char *savetarget;
savetarget = self->target;
self->target = self->pathtarget;
G_UseTargets (self, other);
self->target = savetarget;
}
// DWH: Allow blank target field
if (self->target)
other->movetarget = G_PickTarget(self->target);
else
other->movetarget = NULL;
if (!other->goalentity)
other->goalentity = other->movetarget;
if (self->wait)
{
other->monsterinfo.pausetime = level.time + self->wait;
other->monsterinfo.stand (other);
}
else
{
if (!other->movetarget && !other->enemy)
{
other->monsterinfo.pausetime = level.time + 100000000;
other->monsterinfo.stand (other);
}
else if (other->movetarget == other->goalentity)
{
// DWH: Bug fix here... possible to get here with NULL movetarget and goalentity
if (other->movetarget) {
VectorSubtract (other->movetarget->s.origin, other->s.origin, v);
other->ideal_yaw = vectoyaw (v);
}
}
}
self->count--;
if (!self->count) {
self->think = G_FreeEdict;
self->nextthink = level.time + 1;
}
}
void SP_target_actor (edict_t *self)
{
if (deathmatch->value) {
G_FreeEdict(self);
return;
}
self->class_id = ENTITY_TARGET_ACTOR;
if (!self->targetname)
gi.dprintf ("%s with no targetname at %s\n", self->classname, vtos(self->s.origin));
self->solid = SOLID_TRIGGER;
self->touch = target_actor_touch;
VectorSet (self->mins, -8, -8, -8);
VectorSet (self->maxs, 8, 8, 8);
self->svflags = SVF_NOCLIENT;
if (self->spawnflags & 1)
{
if (!self->speed)
self->speed = 200;
if (!st.height)
st.height = 200;
if (self->s.angles[YAW] == 0)
self->s.angles[YAW] = 360;
G_SetMovedir (self->s.angles, self->movedir);
self->movedir[2] = st.height;
}
gi.linkentity (self);
}
qboolean InPak (char *basedir, char *gamedir, char *filename)
{
char pakfile[256];
int k, kk;
int num, numitems;
FILE *f;
pak_header_t pakheader;
pak_item_t pakitem;
qboolean found = false;
#ifdef KMQUAKE2_ENGINE_MOD // *.pak/pk3 support
char *file_data;
int file_size = 0;
file_size = gi.LoadFile(filename, (void **)&file_data);
if (file_data)
gi.FreeFile(file_data);
if (file_size > 2)
return true;
#endif
// Search paks in game folder
for (k=9; k>=0 && !found; k--)
{
strncpy(pakfile, basedir, sizeof(pakfile));
if (strlen(gamedir))
{
Q_strncatz(pakfile, "/", sizeof(pakfile));
Q_strncatz(pakfile, gamedir, sizeof(pakfile));
}
Q_strncatz(pakfile, va("/pak%d.pak",k), sizeof(pakfile));
if (NULL != (f = fopen(pakfile, "rb")))
{
num = (int)fread(&pakheader,1,sizeof(pak_header_t),f);
if (num >= sizeof(pak_header_t))
{
if ( pakheader.id[0] == 'P' &&
pakheader.id[1] == 'A' &&
pakheader.id[2] == 'C' &&
pakheader.id[3] == 'K' )
{
numitems = pakheader.dsize/sizeof(pak_item_t);
fseek(f,pakheader.dstart,SEEK_SET);
for (kk=0; kk<numitems && !found; kk++)
{
fread(&pakitem,1,sizeof(pak_item_t),f);
if (!Q_stricmp(pakitem.name,filename))
found = true;
}
}
}
fclose(f);
}
}
return found;
}
typedef struct
{
int index;
} actorlist;
void actor_files ()
{
char path[256];
char filename[256];
int s_match, w_match[2];
int i, j, k;
int num_actors = 0;
actorlist actors[MAX_EDICTS];
cvar_t *basedir, *cddir, *gamedir;
edict_t *e, *e0;
FILE *f;
if (deathmatch->value)
return;
basedir = gi.cvar("basedir", "", 0);
cddir = gi.cvar("cddir", "", 0);
gamedir = gi.cvar("gamedir", "", 0);
memset(&actors,0,MAX_EDICTS*sizeof(actorlist));
for (i=game.maxclients+1; i<globals.num_edicts; i++) {
e = &g_edicts[i];
if (!e->inuse) continue;
if (!e->classname) continue;
if (!(e->monsterinfo.aiflags & AI_ACTOR)) continue;
for (j=0; j<NUM_ACTOR_SOUNDS; j++)
e->actor_sound_index[j] = 0;
s_match = 0;
w_match[0] = 0;
w_match[1] = 0;
if (num_actors > 0) {
for (j=0; j<num_actors && (s_match == 0 || w_match[0] == 0 || w_match[1] == 0); j++) {
e0 = &g_edicts[actors[j].index];
if (!Q_stricmp(e->usermodel,e0->usermodel)) {
s_match = j+1;
if (e->actor_weapon[0] == e0->actor_weapon[0])
w_match[0] = j*2+1;
else if (e->actor_weapon[0] == e0->actor_weapon[1])
w_match[0] = j*2+2;
if (e->actor_weapon[1] == e0->actor_weapon[0])
w_match[1] = j*2+1;
else if (e->actor_weapon[1] == e0->actor_weapon[1])
w_match[1] = j*2+2;
}
}
if (s_match) {
// copy sound indices from previous actor
e0 = &g_edicts[actors[s_match-1].index];
for (j=0; j<NUM_ACTOR_SOUNDS; j++)
e->actor_sound_index[j] = e0->actor_sound_index[j];
}
if (w_match[0]) {
k = (w_match[0]-1) % 2;
e0 = &g_edicts[actors[ (w_match[0]-k-1)/2 ].index];
e->s.modelindex2 = e->actor_model_index[0] = e0->actor_model_index[k];
}
if (w_match[1]) {
k = (w_match[1]-1) % 2;
e0 = &g_edicts[actors[ (w_match[1]-k-1)/2 ].index];
e->actor_model_index[1] = e0->actor_model_index[k];
}
}
if (!s_match) {
// search for sounds on hard disk and in paks
actors[num_actors].index = i;
num_actors++;
if (!Q_stricmp(e->usermodel,"male") || !Q_stricmp(e->usermodel,"female")) {
Com_sprintf(path, sizeof(path), "player/%s/",e->usermodel);
} else {
Com_sprintf(path, sizeof(path), "../players/%s/",e->usermodel);
}
for (j=0; j<NUM_ACTOR_SOUNDS; j++) {
if (e->actor_sound_index[j])
continue;
// If it's NOT a custom model, start by looking in game folder
if (strlen(gamedir->string))
{
Com_sprintf(filename, sizeof(filename), "%s/%s/sound/%s%s", basedir->string, gamedir->string, path,wavname[j]);
f = fopen(filename,"r");
if (f) {
fclose(f);
Q_strncpyz(filename, path, sizeof(filename));
Q_strncatz(filename, wavname[j], sizeof(filename));
e->actor_sound_index[j] = gi.soundindex(filename);
continue;
}
// Search paks in game folder
Com_sprintf(filename, sizeof(filename), "sound/%s%s",path,wavname[j]);
if (InPak(basedir->string, gamedir->string, filename)) {
Q_strncpyz(filename, path, sizeof(filename));
Q_strncatz(filename, wavname[j], sizeof(filename));
e->actor_sound_index[j] = gi.soundindex(filename);
continue;
}
}
// Search in baseq2 for external file
Com_sprintf(filename, sizeof(filename), "%s/baseq2/sound/%s%s", basedir->string, path, wavname[j]);
f = fopen(filename,"r");
if (f) {
fclose(f);
Q_strncpyz(filename, path, sizeof(filename));
Q_strncatz(filename, wavname[j], sizeof(filename));
e->actor_sound_index[j] = gi.soundindex(filename);
continue;
}
// Search paks in baseq2
Com_sprintf(filename, sizeof(filename), "sound/%s%s",path,wavname[j]);
if (InPak(basedir->string, "baseq2", filename)) {
Q_strncpyz(filename, path, sizeof(filename));
Q_strncatz(filename, wavname[j], sizeof(filename));
e->actor_sound_index[j] = gi.soundindex(filename);
continue;
}
if (strlen(cddir->string))
{
// Search in cddir (minimal installation)
Com_sprintf(filename, sizeof(filename), "%s/baseq2/sound/%s%s",cddir->string,path,wavname[j]);
f = fopen(filename,"r");
if (f) {
fclose(f);
Q_strncpyz(filename, path, sizeof(filename));
Q_strncatz(filename, wavname[j], sizeof(filename));
e->actor_sound_index[j] = gi.soundindex(filename);
continue;
}
// Search paks in baseq2
Com_sprintf(filename, sizeof(filename), "sound/%s%s",path,wavname[j]);
if (InPak(cddir->string,"baseq2",filename)) {
Q_strncpyz(filename, path, sizeof(filename));
Q_strncatz(filename, wavname[j], sizeof(filename));
e->actor_sound_index[j] = gi.soundindex(filename);
continue;
}
}
// If sound is STILL not found, use normal male sounds
Com_sprintf(filename, sizeof(filename), "player/male/%s",wavname[j]);
e->actor_sound_index[j] = gi.soundindex(filename);
}
}
// repeat this WHOLE DAMN THING for weapons
for (k=0; k<2; k++) {
if (w_match[k]) continue;
if (!e->actor_weapon[k]) continue;
if ((k==1) && (e->actor_weapon[0] == e->actor_weapon[1])) {
e->actor_model_index[1] = e->actor_model_index[0];
continue;
}
if (s_match)
{
// Wasn't added to table on account of sounds
if (k==0 || w_match[0] > 0) {
// Either this is weapon 0, or weapon 0 was a match. Either
// way, this guy has something unique and hasn't been added
// to the table
actors[num_actors].index = i;
num_actors++;
}
}
Com_sprintf(filename, sizeof(filename), "players/%s/",e->usermodel);
switch (e->actor_weapon[k]) {
case 2: Q_strncatz(filename, "w_shotgun.md2", sizeof(filename)); break;
case 3: Q_strncatz(filename, "w_sshotgun.md2", sizeof(filename)); break;
case 4: Q_strncatz(filename, "w_machinegun.md2", sizeof(filename)); break;
case 5: Q_strncatz(filename, "w_chaingun.md2", sizeof(filename)); break;
case 6: Q_strncatz(filename, "w_glauncher.md2", sizeof(filename)); break;
case 7: Q_strncatz(filename, "w_rlauncher.md2", sizeof(filename)); break;
case 8: Q_strncatz(filename, "w_hyperblaster.md2", sizeof(filename)); break;
case 9: Q_strncatz(filename, "w_railgun.md2", sizeof(filename)); break;
case 10:Q_strncatz(filename, "w_bfg.md2", sizeof(filename)); break;
default:Q_strncatz(filename, "w_blaster.md2", sizeof(filename)); break;
}
if (strlen(gamedir->string))
{
// Start in game folder
Com_sprintf(path, sizeof(path), "%s/%s/%s", basedir->string, gamedir->string, filename);
f = fopen(path,"r");
if (f) {
fclose(f);
e->actor_model_index[k] = gi.modelindex(filename);
continue;
}
// Search pak files in gamedir
if (InPak(basedir->string,gamedir->string,filename)) {
e->actor_model_index[k] = gi.modelindex(filename);
continue;
}
}
// Search in baseq2 for external file
Com_sprintf(path, sizeof(path), "%s/baseq2/%s",basedir->string,filename);
f = fopen(path,"r");
if (f) {
fclose(f);
e->actor_model_index[k] = gi.modelindex(filename);
continue;
}
// Search paks in baseq2
if (InPak(basedir->string,"baseq2",filename)) {
e->actor_model_index[k] = gi.modelindex(filename);
continue;
}
if (strlen(cddir->string))
{
// Search CD for minimal installations
Com_sprintf(path, sizeof(path), "%s/baseq2/%s",cddir->string,filename);
f = fopen(path,"r");
if (f) {
fclose(f);
e->actor_model_index[k] = gi.modelindex(filename);
continue;
}
// Search paks in baseq2
if (InPak(cddir->string,"baseq2",filename)) {
e->actor_model_index[k] = gi.modelindex(filename);
continue;
}
}
// If sound is STILL not found, start the fuck over and look for weapon.md2
Com_sprintf(filename, sizeof(filename), "players/%s/weapon.md2",e->usermodel);
if (strlen(gamedir->string))
{
// Start in game folder
Com_sprintf(path, sizeof(path), "%s/%s/%s",basedir->string,gamedir->string,filename);
f = fopen(path,"r");
if (f) {
fclose(f);
e->actor_model_index[k] = gi.modelindex(filename);
continue;
}
// Search pak files in gamedir
if (InPak(basedir->string,gamedir->string,filename)) {
e->actor_model_index[k] = gi.modelindex(filename);
continue;
}
}
// Search in baseq2 for external file
Com_sprintf(path, sizeof(path), "%s/baseq2/%s",basedir->string,filename);
f = fopen(path,"r");
if (f) {
fclose(f);
e->actor_model_index[k] = gi.modelindex(filename);
continue;
}
// Search paks in baseq2
if (InPak(basedir->string,"baseq2",filename)) {
e->actor_model_index[k] = gi.modelindex(filename);
continue;
}
if (strlen(cddir->string))
{
// Search CD for minimal installations
Com_sprintf(path, sizeof(path), "%s/baseq2/%s",cddir->string,filename);
f = fopen(path,"r");
if (f) {
fclose(f);
e->actor_model_index[k] = gi.modelindex(filename);
continue;
}
// Search paks in baseq2
if (InPak(cddir->string,"baseq2",filename)) {
e->actor_model_index[k] = gi.modelindex(filename);
continue;
}
}
// And if it's STILL not found, use
Com_sprintf(filename, sizeof(filename), "players/male/weapon.md2");
e->actor_model_index[k] = gi.modelindex(filename);
}
if (e->health > 0)
e->s.modelindex2 = e->actor_model_index[e->actor_current_weapon];
else
e->s.modelindex2 = 0;
gi.linkentity(e);
}
}
void actor_moveit (edict_t *player, edict_t *actor)
{
edict_t *thing;
trace_t tr;
vec3_t dir, end;
vec_t d[3];
vec_t temp;
vec_t travel;
int best=0;
if (!(actor->monsterinfo.aiflags & AI_FOLLOW_LEADER))
return;
if (actor->enemy)
return;
if (actor->health <= 0)
return;
travel = 256 + 128*crandom();
thing = actor->vehicle;
if (!thing || !thing->inuse || Q_stricmp(thing->classname,"thing"))
thing = actor->vehicle = SpawnThing();
VectorSubtract(actor->s.origin,player->s.origin,dir);
dir[2] = 0;
VectorNormalize(dir);
if (!VectorLength(dir))
VectorSet(dir,1.0,0.,0.);
VectorMA(actor->s.origin,travel,dir,end);
tr = gi.trace(actor->s.origin,NULL,NULL,end,actor,MASK_MONSTERSOLID);
d[best] = tr.fraction * travel;
if (d[best] < 64)
{
temp = dir[0];
dir[0] = -dir[1];
dir[1] = temp;
VectorMA(actor->s.origin,travel,dir,end);
tr = gi.trace(actor->s.origin,NULL,NULL,end,actor,MASK_MONSTERSOLID);
best = 1;
d[best] = tr.fraction * travel;
if (d[best] < 64)
{
dir[0] = -dir[0];
dir[1] = -dir[1];
VectorMA(actor->s.origin,travel,dir,end);
tr = gi.trace(actor->s.origin,NULL,NULL,end,actor,MASK_MONSTERSOLID);
best = 2;
d[best] = tr.fraction * travel;
if (d[best] < 64)
{
if (d[0] > d[1] && d[0] > d[2])
best = 0;
else if (d[1] > d[0] && d[1] > d[2])
best = 1;
if (best==1)
{
dir[0] = -dir[0];
dir[1] = -dir[1];
}
else if (best==0)
{
temp = -dir[1];
dir[1] = dir[0];
dir[0] = temp;
}
}
}
}
VectorCopy(tr.endpos, thing->s.origin);
thing->touch_debounce_time = level.time + max(5.0,d[best]/50.);
thing->target_ent = actor;
ED_CallSpawn(thing);
actor->monsterinfo.aiflags |= AI_CHASE_THING;
actor->movetarget = actor->goalentity = thing;
actor->monsterinfo.old_leader = player;
actor->monsterinfo.leader = thing;
VectorSubtract (thing->s.origin, actor->s.origin, dir);
actor->ideal_yaw = vectoyaw(dir);
actor->monsterinfo.run(actor);
}