super-coop/source/ai_subs.qc

298 lines
6.0 KiB
Plaintext
Raw Normal View History

// ai_subs.qc: subroutines for think frames
2019-09-11 11:47:16 -07:00
2019-09-20 20:24:56 -07:00
void() sub_null = {};
void(entity attacker, float damage) sub_pain_null = {};
void() sub_remove = {remove(self);};
2019-09-11 11:47:16 -07:00
/*
QuakeEd only writes a single float for angles(bad idea), so up and down are
just constant angles.
*/
2019-09-20 20:24:56 -07:00
void() set_move_dir = {
2019-09-11 11:47:16 -07:00
if(self.angles == '0 -1 0') {
self.movedir = '0 0 1';
} else if(self.angles == '0 -2 0') {
self.movedir = '0 0 -1';
} else {
2019-09-20 20:24:56 -07:00
make_vectors(self.angles);
2019-09-11 11:47:16 -07:00
self.movedir = v_forward;
}
2019-09-19 07:16:20 -07:00
self.angles = VEC_ORIGIN;
2019-09-11 11:47:16 -07:00
};
/*
================
2019-09-20 20:24:56 -07:00
init_trigger
2019-09-11 11:47:16 -07:00
================
*/
2019-09-20 20:24:56 -07:00
void() init_trigger = {
2019-09-11 11:47:16 -07:00
// trigger angles are used for one-way touches. An angle of 0 is assumed
// to mean no restrictions, so use a yaw of 360 instead.
2019-09-19 07:16:20 -07:00
if(self.angles != VEC_ORIGIN) {
2019-09-20 20:24:56 -07:00
set_move_dir();
2019-09-11 11:47:16 -07:00
}
self.solid = SOLID_TRIGGER;
2019-09-20 20:24:56 -07:00
set_model(self, self.model); // set size and link into world
2019-09-11 11:47:16 -07:00
self.movetype = MOVETYPE_NONE;
self.modelindex = 0;
2019-09-19 16:39:59 -07:00
self.model = string_null;
2019-09-11 11:47:16 -07:00
};
/*
=============
2019-09-20 20:24:56 -07:00
sub_calc_move
2019-09-11 11:47:16 -07:00
calculate self.velocity and self.nextthink to reach dest from
self.origin traveling at speed
===============
*/
2019-09-20 20:24:56 -07:00
void(entity ent, vector tdest, float tspeed, void() func) sub_calc_move_ent = {
2019-09-17 08:26:44 -07:00
entity stemp;
2019-09-11 11:47:16 -07:00
stemp = self;
self = ent;
2019-09-20 20:24:56 -07:00
sub_calc_move(tdest, tspeed, func);
2019-09-11 11:47:16 -07:00
self = stemp;
};
2019-09-20 20:24:56 -07:00
void(vector tdest, float tspeed, void() func) sub_calc_move = {
2019-09-17 08:26:44 -07:00
vector vdestdelta;
float len, traveltime;
2019-09-11 11:47:16 -07:00
if(!tspeed) {
2019-09-20 20:24:56 -07:00
error_obj("No speed is defined!");
2019-09-11 11:47:16 -07:00
}
self.think1 = func;
self.finaldest = tdest;
2019-09-20 20:24:56 -07:00
self.think = sub_calc_move_done;
2019-09-11 11:47:16 -07:00
if(tdest == self.origin) {
2019-09-19 07:16:20 -07:00
self.velocity = VEC_ORIGIN;
2019-09-11 11:47:16 -07:00
self.nextthink = self.ltime + 0.1;
return;
}
// set destdelta to the vector needed to move
vdestdelta = tdest - self.origin;
// calculate length of vector
2019-09-20 20:24:56 -07:00
len = vec_len(vdestdelta);
2019-09-11 11:47:16 -07:00
// divide by speed to get time to reach dest
traveltime = len / tspeed;
if(traveltime < 0.1) {
2019-09-19 07:16:20 -07:00
self.velocity = VEC_ORIGIN;
2019-09-11 11:47:16 -07:00
self.nextthink = self.ltime + 0.1;
return;
}
// set nextthink to trigger a think when dest is reached
self.nextthink = self.ltime + traveltime;
// scale the destdelta vector by the time spent traveling to get velocity
self.velocity = vdestdelta * (1 / traveltime); // qcc won't take vec/float
};
/*
============
After moving, set origin to exact final destination
============
*/
2019-09-20 20:24:56 -07:00
void() sub_calc_move_done = {
set_origin(self, self.finaldest);
2019-09-19 07:16:20 -07:00
self.velocity = VEC_ORIGIN;
2019-09-11 11:47:16 -07:00
self.nextthink = -1;
if(self.think1) {
self.think1();
}
};
/*
=============
2019-09-20 20:24:56 -07:00
sub_calc_angle_move
2019-09-11 11:47:16 -07:00
calculate self.avelocity and self.nextthink to reach destangle from
self.angles rotating
The calling function should make sure self.think is valid
===============
*/
2019-09-20 20:24:56 -07:00
void(entity ent, vector destangle, float tspeed, void() func) sub_calc_angle_moveEnt = {
2019-09-17 08:26:44 -07:00
entity stemp;
2019-09-11 11:47:16 -07:00
stemp = self;
self = ent;
2019-09-20 20:24:56 -07:00
sub_calc_angle_move(destangle, tspeed, func);
2019-09-11 11:47:16 -07:00
self = stemp;
};
2019-09-20 20:24:56 -07:00
void(vector destangle, float tspeed, void() func) sub_calc_angle_move = {
2019-09-17 08:26:44 -07:00
vector destdelta;
float len, traveltime;
2019-09-11 11:47:16 -07:00
if(!tspeed) {
2019-09-20 20:24:56 -07:00
error_obj("No speed is defined!");
2019-09-11 11:47:16 -07:00
}
// set destdelta to the vector needed to move
destdelta = destangle - self.angles;
// calculate length of vector
2019-09-20 20:24:56 -07:00
len = vec_len(destdelta);
2019-09-11 11:47:16 -07:00
// divide by speed to get time to reach dest
traveltime = len / tspeed;
// set nextthink to trigger a think when dest is reached
self.nextthink = self.ltime + traveltime;
// scale the destdelta vector by the time spent traveling to get velocity
self.avelocity = destdelta * (1 / traveltime);
self.think1 = func;
self.finalangle = destangle;
2019-09-20 20:24:56 -07:00
self.think = sub_calc_angle_move_done;
2019-09-11 11:47:16 -07:00
};
/*
============
After rotating, set angle to exact final angle
============
*/
2019-09-20 20:24:56 -07:00
void() sub_calc_angle_move_done = {
2019-09-11 11:47:16 -07:00
self.angles = self.finalangle;
2019-09-19 07:16:20 -07:00
self.avelocity = VEC_ORIGIN;
2019-09-11 11:47:16 -07:00
self.nextthink = -1;
if(self.think1) {
self.think1();
}
};
//=============================================================================
2019-09-20 20:24:56 -07:00
void() delay_think = {
2019-09-11 11:47:16 -07:00
activator = self.enemy;
2019-09-20 20:24:56 -07:00
sub_use_targets();
2019-09-11 11:47:16 -07:00
remove(self);
};
/*
==============================
2019-09-20 20:24:56 -07:00
sub_use_targets
2019-09-11 11:47:16 -07:00
the global "activator" should be set to the entity that initiated the firing.
If self.delay is set, a DelayedUse entity will be created that will actually
2019-09-20 20:24:56 -07:00
do the sub_use_targets after that many seconds have passed.
2019-09-11 11:47:16 -07:00
Centerprints any self.message to the activator.
Removes all entities with a targetname that match self.killtarget,
and removes them, so some events can remove other triggers.
Search for(string)targetname in all entities that
match(string)self.target and call their .use function
==============================
*/
2019-09-20 20:24:56 -07:00
void() sub_use_targets = {
2019-09-17 08:26:44 -07:00
entity t, stemp, otemp, act;
2019-09-11 11:47:16 -07:00
//
// check for a delay
//
if(self.delay) {
// create a temp object to fire at a later time
t = spawn();
t.classname = "DelayedUse";
t.nextthink = time + self.delay;
2019-09-20 20:24:56 -07:00
t.think = delay_think;
2019-09-11 11:47:16 -07:00
t.enemy = activator;
t.message = self.message;
t.killtarget = self.killtarget;
t.target = self.target;
return;
}
//
// print the message
//
2019-09-19 16:39:59 -07:00
if(activator.classname == "player" && self.message != string_null) {
2019-09-20 20:24:56 -07:00
print_center(activator, self.message);
2019-09-11 11:47:16 -07:00
if(!self.noise) {
sound(activator, CHAN_VOICE, "misc/talk.wav", 1, ATTN_NORM);
}
}
//
// kill the killtagets
//
if(self.killtarget) {
t = world;
do {
t = find(t, targetname, self.killtarget);
if(!t) {
return;
}
remove(t);
} while(1);
}
//
// fire targets
//
if(self.target) {
act = activator;
t = world;
do {
t = find(t, targetname, self.target);
if(!t) {
return;
}
stemp = self;
otemp = other;
self = t;
other = stemp;
2019-09-20 20:24:56 -07:00
if(self.use != sub_null) {
2019-09-11 11:47:16 -07:00
if(self.use) {
self.use();
}
}
self = stemp;
other = otemp;
activator = act;
} while(1);
}
};
/*
in nightmare mode, all attack_finished times become 0
some monsters refire twice automatically
*/
2019-09-20 20:24:56 -07:00
void(float normal) sub_attack_finished = {
2019-09-11 11:47:16 -07:00
self.cnt = 0; // refire count for nightmare
if(skill != 3) {
self.attack_finished = time + normal;
}
};
2019-09-20 20:24:56 -07:00
void(void() thinkst) sub_check_refire = {
2019-09-11 11:47:16 -07:00
if(skill != 3) {
return;
}
if(self.cnt == 1) {
return;
}
if(!visible(self.enemy)) {
return;
}
self.cnt = 1;
self.think = thinkst;
};