2003-01-24 19:31:58 +00:00
|
|
|
/*
|
2003-03-03 19:17:04 +00:00
|
|
|
combat.qc
|
2003-01-24 19:31:58 +00:00
|
|
|
|
|
|
|
This program 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.
|
|
|
|
|
|
|
|
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, write to:
|
|
|
|
|
|
|
|
Free Software Foundation, Inc.
|
|
|
|
59 Temple Place - Suite 330
|
|
|
|
Boston, MA 02111-1307, USA
|
|
|
|
|
|
|
|
$Id$
|
|
|
|
*/
|
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
void () T_MissileTouch;
|
|
|
|
void () info_player_start;
|
|
|
|
void (entity targ, entity attacker) ClientObituary;
|
|
|
|
void (entity who) ResistanceSound;
|
2003-01-24 19:31:58 +00:00
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
/* SERVER
|
2003-01-24 19:31:58 +00:00
|
|
|
void() monster_death_use;
|
|
|
|
*/
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
|
|
|
|
/*
|
|
|
|
============
|
|
|
|
CanDamage
|
|
|
|
|
|
|
|
Returns true if the inflictor can directly damage the target. Used for
|
|
|
|
explosions and melee attacks.
|
|
|
|
============
|
|
|
|
*/
|
2003-03-03 19:17:04 +00:00
|
|
|
float (entity targ, entity inflictor)
|
|
|
|
CanDamage =
|
2003-01-24 19:31:58 +00:00
|
|
|
{
|
2003-03-03 19:17:04 +00:00
|
|
|
// bmodels need special checking because their origin is 0,0,0
|
|
|
|
if (targ.movetype == MOVETYPE_PUSH) {
|
|
|
|
traceline (inflictor.origin, 0.5 * (targ.absmin + targ.absmax), TRUE,
|
|
|
|
self);
|
2003-01-24 19:31:58 +00:00
|
|
|
if (trace_fraction == 1)
|
|
|
|
return TRUE;
|
|
|
|
if (trace_ent == targ)
|
|
|
|
return TRUE;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
traceline (inflictor.origin, targ.origin, TRUE, self);
|
2003-01-24 19:31:58 +00:00
|
|
|
if (trace_fraction == 1)
|
|
|
|
return TRUE;
|
2003-03-03 19:17:04 +00:00
|
|
|
traceline (inflictor.origin, targ.origin + '15 15 0', TRUE, self);
|
2003-01-24 19:31:58 +00:00
|
|
|
if (trace_fraction == 1)
|
|
|
|
return TRUE;
|
2003-03-03 19:17:04 +00:00
|
|
|
traceline (inflictor.origin, targ.origin + '-15 -15 0', TRUE, self);
|
2003-01-24 19:31:58 +00:00
|
|
|
if (trace_fraction == 1)
|
|
|
|
return TRUE;
|
2003-03-03 19:17:04 +00:00
|
|
|
traceline (inflictor.origin, targ.origin + '-15 15 0', TRUE, self);
|
2003-01-24 19:31:58 +00:00
|
|
|
if (trace_fraction == 1)
|
|
|
|
return TRUE;
|
2003-03-03 19:17:04 +00:00
|
|
|
traceline (inflictor.origin, targ.origin + '15 -15 0', TRUE, self);
|
2003-01-24 19:31:58 +00:00
|
|
|
if (trace_fraction == 1)
|
|
|
|
return TRUE;
|
|
|
|
return FALSE;
|
|
|
|
};
|
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
void (entity targ, entity attacker)
|
|
|
|
Killed =
|
2003-01-24 19:31:58 +00:00
|
|
|
{
|
2003-03-03 19:17:04 +00:00
|
|
|
local entity oself;
|
2003-01-24 19:31:58 +00:00
|
|
|
|
|
|
|
oself = self;
|
|
|
|
self = targ;
|
|
|
|
|
|
|
|
if (self.health < -99)
|
|
|
|
self.health = -99; // don't let sbar look bad if a player
|
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
if (self.movetype == MOVETYPE_PUSH || self.movetype == MOVETYPE_NONE) {
|
|
|
|
// doors, triggers, etc
|
2003-01-24 19:31:58 +00:00
|
|
|
self.th_die ();
|
|
|
|
self = oself;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
self.enemy = attacker;
|
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
// bump the monster counter
|
|
|
|
if (self.flags & FL_MONSTER) {
|
|
|
|
killed_monsters++;
|
2003-01-24 19:31:58 +00:00
|
|
|
WriteByte (MSG_ALL, SVC_KILLEDMONSTER);
|
|
|
|
}
|
|
|
|
|
|
|
|
ClientObituary(self, attacker);
|
|
|
|
|
|
|
|
self.takedamage = DAMAGE_NO;
|
|
|
|
self.touch = SUB_Null;
|
|
|
|
self.effects = 0;
|
|
|
|
|
|
|
|
/*SERVER
|
|
|
|
monster_death_use();
|
|
|
|
*/
|
|
|
|
self.th_die ();
|
|
|
|
|
|
|
|
self = oself;
|
|
|
|
};
|
|
|
|
|
|
|
|
// *TEAMPLAY*
|
|
|
|
// Prototypes
|
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
float (entity targ, entity inflictor, entity attacker, float damage)
|
|
|
|
TeamArmorDam;
|
|
|
|
float (entity targ, entity inflictor, entity attacker, float damage)
|
|
|
|
TeamHealthDam;
|
2003-01-24 19:31:58 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
============
|
|
|
|
T_Damage
|
|
|
|
|
|
|
|
The damage is coming from inflictor, but get mad at attacker
|
|
|
|
This should be the only function that ever reduces health.
|
|
|
|
============
|
|
|
|
*/
|
2003-03-03 19:17:04 +00:00
|
|
|
void (entity targ, entity inflictor, entity attacker, float damage)
|
|
|
|
T_Damage=
|
2003-01-24 19:31:58 +00:00
|
|
|
{
|
2003-03-03 19:17:04 +00:00
|
|
|
local entity oldself;
|
|
|
|
local float save, take;
|
|
|
|
local vector dir;
|
2003-01-24 19:31:58 +00:00
|
|
|
|
|
|
|
if (!targ.takedamage)
|
|
|
|
return;
|
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
// used by buttons and triggers to set activator for target firing
|
2003-01-24 19:31:58 +00:00
|
|
|
damage_attacker = attacker;
|
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
// check for quad damage powerup on the attacker
|
2003-01-24 19:31:58 +00:00
|
|
|
if (attacker.super_damage_finished > time)
|
2003-03-03 19:17:04 +00:00
|
|
|
damage *= 4;
|
|
|
|
// RUNE: check for double damage for rune of Black Magic powerup
|
2003-01-24 19:31:58 +00:00
|
|
|
if (attacker.player_flag & ITEM_RUNE2_FLAG)
|
2003-03-03 19:17:04 +00:00
|
|
|
damage *= 2;
|
2003-01-24 19:31:58 +00:00
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
//RUNE check if target has rune of Earth Magic (half damage)
|
2003-01-24 19:31:58 +00:00
|
|
|
if (targ.player_flag & ITEM_RUNE1_FLAG) {
|
2003-03-03 19:17:04 +00:00
|
|
|
damage /= 2;
|
|
|
|
ResistanceSound (targ);
|
2003-01-24 19:31:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// *XXX* EXPERT CTF mark players who hurt the flag carrier, so they
|
|
|
|
// are worth more points for a while.
|
2003-03-03 19:17:04 +00:00
|
|
|
if ((attacker.classname == "player") // attacker must be a player
|
|
|
|
&& (targ.player_flag & ITEM_ENEMY_FLAG) // target is a flag carrier
|
|
|
|
&& (attacker.steam != targ.steam)) // target and attacker on diff teams
|
|
|
|
attacker.last_hurt_carrier = time;
|
2003-01-24 19:31:58 +00:00
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
// save damage based on the target's armor level
|
2003-01-24 19:31:58 +00:00
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
// *TEAMPLAY*
|
|
|
|
// TeamArmorDam returns true iff the attacker can damage the target's armor
|
|
|
|
if (TeamArmorDam (targ, inflictor, attacker, damage))
|
|
|
|
save = ceil (targ.armortype * damage);
|
2003-01-24 19:31:58 +00:00
|
|
|
else
|
|
|
|
save = 0;
|
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
if (save >= targ.armorvalue) {
|
2003-01-24 19:31:58 +00:00
|
|
|
save = targ.armorvalue;
|
|
|
|
targ.armortype = 0; // lost all armor
|
2003-03-03 19:17:04 +00:00
|
|
|
targ.items &= ~(IT_ARMOR1 | IT_ARMOR2 | IT_ARMOR3);
|
2003-01-24 19:31:58 +00:00
|
|
|
}
|
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
targ.armorvalue -= save;
|
|
|
|
take = ceil (damage - save);
|
|
|
|
|
|
|
|
// add to the damage total for clients, which will be sent as a single
|
|
|
|
// message at the end of the frame
|
|
|
|
// FIXME: remove after combining shotgun blasts?
|
|
|
|
if (targ.flags & FL_CLIENT) {
|
|
|
|
targ.dmg_take += take;
|
|
|
|
targ.dmg_save += save;
|
2003-01-24 19:31:58 +00:00
|
|
|
targ.dmg_inflictor = inflictor;
|
|
|
|
}
|
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
// figure momentum add
|
|
|
|
if ((inflictor != world) && (targ.movetype == MOVETYPE_WALK)) {
|
2003-01-24 19:31:58 +00:00
|
|
|
dir = targ.origin - (inflictor.absmin + inflictor.absmax) * 0.5;
|
2003-03-03 19:17:04 +00:00
|
|
|
dir = normalize (dir);
|
|
|
|
targ.velocity += dir * damage * 8;
|
2003-01-24 19:31:58 +00:00
|
|
|
}
|
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
// check for godmode or invincibility
|
2003-01-24 19:31:58 +00:00
|
|
|
if (self.killed != 99) { // god or 666 doesn't save your from team change
|
|
|
|
if (targ.flags & FL_GODMODE)
|
|
|
|
return;
|
2003-03-03 19:17:04 +00:00
|
|
|
if (targ.invincible_finished >= time) {
|
|
|
|
if (self.invincible_sound < time) {
|
2003-01-24 19:31:58 +00:00
|
|
|
sound (targ, CHAN_ITEM, "items/protect3.wav", 1, ATTN_NORM);
|
|
|
|
self.invincible_sound = time + 2;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
// team play damage avoidance
|
|
|
|
if ((teamplay == 1) && (targ.team > 0) && (targ.team == attacker.team))
|
2003-01-24 19:31:58 +00:00
|
|
|
return;
|
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
// *TEAMPLAY*
|
|
|
|
// TeamHealthDam will return true if the attacker can damage the target's
|
|
|
|
// health
|
|
|
|
if (!TeamHealthDam (targ, inflictor, attacker, damage))
|
2003-01-24 19:31:58 +00:00
|
|
|
return;
|
2003-03-03 19:17:04 +00:00
|
|
|
|
|
|
|
// do the damage
|
2003-01-24 19:31:58 +00:00
|
|
|
targ.health = targ.health - take;
|
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
if (targ.health <= 0) {
|
2003-01-24 19:31:58 +00:00
|
|
|
Killed (targ, attacker);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
// react to the damage
|
2003-01-24 19:31:58 +00:00
|
|
|
oldself = self;
|
|
|
|
self = targ;
|
|
|
|
|
|
|
|
/*SERVER
|
2003-03-03 19:17:04 +00:00
|
|
|
if ((self.flags & FL_MONSTER) && attacker != world) {
|
2003-01-24 19:31:58 +00:00
|
|
|
// get mad unless of the same class (except for soldiers)
|
2003-03-03 19:17:04 +00:00
|
|
|
if (self != attacker && attacker != self.enemy) {
|
|
|
|
if ((self.classname != attacker.classname)
|
|
|
|
|| (self.classname == "monster_army")) {
|
2003-01-24 19:31:58 +00:00
|
|
|
if (self.enemy.classname == "player")
|
|
|
|
self.oldenemy = self.enemy;
|
|
|
|
self.enemy = attacker;
|
|
|
|
FoundTarget ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
if (self.th_pain)
|
|
|
|
self.th_pain (attacker, take);
|
|
|
|
|
|
|
|
self = oldself;
|
|
|
|
};
|
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
void (entity inflictor, entity attacker, float damage, entity ignore)
|
|
|
|
T_RadiusDamage =
|
2003-01-24 19:31:58 +00:00
|
|
|
{
|
2003-03-03 19:17:04 +00:00
|
|
|
local float points;
|
|
|
|
local entity head;
|
|
|
|
local vector org;
|
2003-01-24 19:31:58 +00:00
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
head = findradius (inflictor.origin, damage + 40);
|
2003-01-24 19:31:58 +00:00
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
while (head) {
|
|
|
|
if (head != ignore) {
|
|
|
|
if (head.takedamage) {
|
|
|
|
org = head.origin + (head.mins + head.maxs) * 0.5;
|
|
|
|
points = 0.5 * vlen (inflictor.origin - org);
|
2003-01-24 19:31:58 +00:00
|
|
|
if (points < 0)
|
|
|
|
points = 0;
|
|
|
|
points = damage - points;
|
|
|
|
if (head == attacker)
|
|
|
|
points = points * 0.5;
|
2003-03-03 19:17:04 +00:00
|
|
|
if (points > 0) {
|
|
|
|
if (CanDamage (head, inflictor)) {
|
|
|
|
// shambler takes half damage from all explosions
|
|
|
|
if (head.classname == "monster_shambler")
|
|
|
|
T_Damage (head, inflictor, attacker, points * 0.5);
|
2003-01-24 19:31:58 +00:00
|
|
|
else
|
|
|
|
T_Damage (head, inflictor, attacker, points);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
head = head.chain;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
void (entity attacker, float damage)
|
|
|
|
T_BeamDamage =
|
2003-01-24 19:31:58 +00:00
|
|
|
{
|
2003-03-03 19:17:04 +00:00
|
|
|
local float points;
|
|
|
|
local entity head;
|
2003-01-24 19:31:58 +00:00
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
head = findradius (attacker.origin, damage + 40);
|
2003-01-24 19:31:58 +00:00
|
|
|
|
2003-03-03 19:17:04 +00:00
|
|
|
while (head) {
|
|
|
|
if (head.takedamage) {
|
|
|
|
points = 0.5 * vlen (attacker.origin - head.origin);
|
2003-01-24 19:31:58 +00:00
|
|
|
if (points < 0)
|
|
|
|
points = 0;
|
|
|
|
points = damage - points;
|
|
|
|
if (head == attacker)
|
|
|
|
points = points * 0.5;
|
2003-03-03 19:17:04 +00:00
|
|
|
if (points > 0) {
|
|
|
|
if (CanDamage (head, attacker)) {
|
|
|
|
if (head.classname == "monster_shambler")
|
|
|
|
T_Damage (head, attacker, attacker, points * 0.5);
|
2003-01-24 19:31:58 +00:00
|
|
|
else
|
|
|
|
T_Damage (head, attacker, attacker, points);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
head = head.chain;
|
|
|
|
}
|
|
|
|
};
|