1275 lines
28 KiB
C++
1275 lines
28 KiB
C++
|
|
void() T_MissileTouch;
|
|
void() info_player_start;
|
|
void(entity targ, entity attacker) ClientObituary;
|
|
void(entity inflictor, entity attacker, float damage, entity ignore, string dtype) T_RadiusDamage;
|
|
float (float int) getperk;
|
|
void() monster_death_use;
|
|
|
|
//============================================================================
|
|
|
|
/*
|
|
============
|
|
CanDamage
|
|
|
|
Returns true if the inflictor can directly damage the target. Used for
|
|
explosions and melee attacks.
|
|
============
|
|
*/
|
|
float(entity targ, entity inflictor) CanDamage =
|
|
{
|
|
// 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);
|
|
if (trace_fraction == 1)
|
|
return TRUE;
|
|
if (trace_ent == targ)
|
|
return TRUE;
|
|
return FALSE;
|
|
}
|
|
|
|
traceline(inflictor.origin, targ.origin, TRUE, self);
|
|
if (trace_fraction == 1)
|
|
return TRUE;
|
|
traceline(inflictor.origin, targ.origin + '15 15 0', TRUE, self);
|
|
if (trace_fraction == 1)
|
|
return TRUE;
|
|
traceline(inflictor.origin, targ.origin + '-15 -15 0', TRUE, self);
|
|
if (trace_fraction == 1)
|
|
return TRUE;
|
|
traceline(inflictor.origin, targ.origin + '-15 15 0', TRUE, self);
|
|
if (trace_fraction == 1)
|
|
return TRUE;
|
|
traceline(inflictor.origin, targ.origin + '15 -15 0', TRUE, self);
|
|
if (trace_fraction == 1)
|
|
return TRUE;
|
|
|
|
return FALSE;
|
|
};
|
|
|
|
|
|
/*
|
|
============
|
|
Killed
|
|
============
|
|
*/
|
|
void(entity targ, entity attacker) Killed =
|
|
{
|
|
local entity oself;
|
|
|
|
oself = self;
|
|
self = targ;
|
|
|
|
if (self.health < -99)
|
|
self.health = -99; // don't let sbar look bad if a player
|
|
|
|
if (self.movetype == MOVETYPE_PUSH || self.movetype == MOVETYPE_NONE)
|
|
{ // doors, triggers, etc
|
|
self.th_die ();
|
|
self = oself;
|
|
return;
|
|
}
|
|
|
|
self.enemy = attacker;
|
|
|
|
targ.grab = 0;
|
|
targ.equipment_slot = 0; //turn off stealth-boys, climbing gear, etc
|
|
|
|
// bump the monster counter
|
|
if (self.flags & FL_MONSTER)
|
|
{
|
|
killed_monsters = killed_monsters + 1;
|
|
WriteByte (MSG_ALL, SVC_KILLEDMONSTER);
|
|
}
|
|
|
|
ClientObituary(self, attacker);
|
|
|
|
if (self.classname == "player") //so dead players can spectate
|
|
{
|
|
self.ghost = 1;
|
|
self.flags = self.flags | FL_FINDABLE_NONSOLID; //so tracelines can find them
|
|
}
|
|
|
|
self.takedamage = DAMAGE_NO;
|
|
self.touch = SUB_Null;
|
|
self.effects = 0;
|
|
|
|
monster_death_use();
|
|
|
|
self.th_die ();
|
|
|
|
self = oself;
|
|
};
|
|
|
|
|
|
/*
|
|
============
|
|
T_Damage
|
|
|
|
The damage is coming from inflictor, but get mad at attacker
|
|
This should be the only function that ever reduces health.
|
|
============
|
|
*/
|
|
|
|
void(entity targ, entity inflictor, entity attacker, float damage) T_Damage =
|
|
{
|
|
local entity oldself;
|
|
local float save, dm;
|
|
local float take, severity, helm;
|
|
local string attackerteam, targteam;
|
|
local string info1, info2;
|
|
|
|
if (!targ.takedamage)
|
|
return;
|
|
|
|
|
|
damage = floor(damage);
|
|
targ.health = floor(targ.health);
|
|
|
|
if (attacker.classname == "player" && total_players == 4)
|
|
damage = floor(damage * 0.75);
|
|
if (attacker.classname == "player" && total_players == 3)
|
|
damage = floor(damage * 0.80);
|
|
else if (attacker.classname == "player" && total_players == 2)
|
|
damage = floor(damage * 0.90);
|
|
|
|
if (attacker.classname == "monster")
|
|
damage = floor(damage * 0.80);
|
|
|
|
|
|
//getting hurt is slightly painful to the score, play cautiously!
|
|
if (attacker.classname == "monster" && targ.classname == "player")
|
|
targ.score = targ.score - 1;
|
|
|
|
//flanking monsters awards points
|
|
if (attacker.classname == "player" && targ.classname == "monster" && targ.enemy != attacker)
|
|
attacker.score = attacker.score + 1;
|
|
|
|
//aimed shots awards points
|
|
if (attacker.classname == "player" && targ.classname == "monster" && attacker.recoil <= 5)
|
|
attacker.score = attacker.score + 1;
|
|
|
|
//attacker a monster from behind awards points
|
|
|
|
local vector vec;
|
|
local float dot;
|
|
|
|
if (attacker.classname == "player" && targ.classname == "monster")
|
|
{
|
|
makevectors (targ.angles);
|
|
vec = normalize (attacker.origin - targ.origin);
|
|
dot = (vec * v_forward);
|
|
|
|
if (dot > 0.3)
|
|
attacker.score = attacker.score + 1;
|
|
}
|
|
|
|
//attacking a team-mate is a HUGE penalty
|
|
if (attacker.classname == "player" && targ.classname == "player")
|
|
{
|
|
attacker.score = attacker.score - 3;
|
|
sprint(attacker, 2, "you just shot a teammate! be careful!\n");
|
|
}
|
|
|
|
//attacking a hostage is a HUGE penalty
|
|
if (attacker.classname == "player" && targ.classname == "hostage")
|
|
{
|
|
attacker.score = attacker.score - 3;
|
|
sprint(attacker, 2, "you just shot a hostage! be careful!\n");
|
|
}
|
|
|
|
//using cover awards points
|
|
if (targ.classname == "monster" && attacker.classname == "player" && attacker.rage == 777)
|
|
attacker.score = attacker.score + 1;
|
|
|
|
//dodge chance
|
|
|
|
if (targ.position == 0 && random()<0.05)
|
|
{
|
|
sound (targ, CHAN_BODY, "effects/miss.wav", 1, ATTN_NORM);
|
|
return;
|
|
}
|
|
else if (targ.position == 1 && random()<0.05)
|
|
{
|
|
sound (targ, CHAN_BODY, "effects/miss.wav", 1, ATTN_NORM);
|
|
return;
|
|
}
|
|
else if (targ.position == 2 && random()<0.10)
|
|
{
|
|
sound (targ, CHAN_BODY, "effects/miss.wav", 1, ATTN_NORM);
|
|
return;
|
|
}
|
|
|
|
damage = damage * 1.25;
|
|
|
|
//nintendo gamer: going into the zone gives a 50% miss chance
|
|
|
|
if (targ.rage == 255)
|
|
{
|
|
if (random()<0.50)
|
|
{
|
|
sound (targ, CHAN_BODY, "effects/miss.wav", 1, ATTN_NORM);
|
|
return;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
//paranoia: if taking cover, 25% miss chance (others get 10% bonus)
|
|
if (targ.rage == 777)
|
|
{
|
|
if (targ.perk1 == 2 || targ.perk2 == 2)
|
|
{
|
|
if (random()<0.20)
|
|
{
|
|
sound (targ, CHAN_BODY, "effects/miss.wav", 1, ATTN_NORM);
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (random()<0.05)
|
|
{
|
|
sound (targ, CHAN_BODY, "effects/miss.wav", 1, ATTN_NORM);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
//master blaster: the more you burst, the higher the damage bonus
|
|
if (attacker.perk1 == 18 || attacker.perk2 == 18)
|
|
{
|
|
if (attacker.recoil > 10)
|
|
{
|
|
sprint(attacker, 2, "!MASTER BLASTER\n");
|
|
damage = damage * (1 + (attacker.recoil * 0.01));
|
|
}
|
|
}
|
|
|
|
|
|
//hit the deck
|
|
if (targ.perk1 == 4 || targ.perk2 == 4)
|
|
{
|
|
if (targ.position == 2)
|
|
damage = damage * 0.90;
|
|
}
|
|
|
|
//sneak attack: if you attack a creature who isn't fighting you, bonus damage
|
|
if (attacker.skill_sneak > 0)
|
|
{
|
|
if (targ.enemy != attacker)
|
|
{
|
|
sprint(attacker, 2, "!SNEAK ATTACK\n");
|
|
damage = damage * (1 + (self.skill_sneak/100));
|
|
}
|
|
}
|
|
|
|
//death wish: 20% damage bonus to targ, 10% to attacker
|
|
if (targ.perk1 == 19 || targ.perk2 == 19)
|
|
damage = damage * 1.10;
|
|
|
|
//death wish: 20% damage bonus to targ, 10% to attacker
|
|
if (attacker.perk1 == 19 || attacker.perk2 == 19)
|
|
damage = damage * 1.20;
|
|
|
|
//die hard: if below 50% health, 25% damage resistance
|
|
if (targ.perk1 == 1 || targ.perk2 == 1)
|
|
{
|
|
if (targ.health < (targ.max_health*0.5))
|
|
damage = damage * 0.75;
|
|
}
|
|
|
|
if ((targ.perk1 == 17 || targ.perk2 == 17) && random()<0.02)
|
|
{
|
|
damage = damage * 0.10;
|
|
sprint(self, 2, "!ONE IN A MILLION\n");
|
|
}
|
|
|
|
// used by buttons and triggers to set activator for target firing
|
|
damage_attacker = attacker;
|
|
|
|
if (attacker.classname == "monster")
|
|
{
|
|
if (random()*100>= 98)
|
|
attacker.critical = 3;
|
|
else
|
|
attacker.critical = 0;
|
|
}
|
|
|
|
if (attacker.critical == 3 && targ.classname != "car")//attacker scored a headshot/critical
|
|
{
|
|
if (attacker.critical == 3)
|
|
{
|
|
|
|
if (random()<0.005)
|
|
severity = 64;
|
|
|
|
|
|
severity = 0 + random()*20;
|
|
|
|
if ((attacker.perk1 == 17 || attacker.perk2 == 17) && random()<0.05)
|
|
{
|
|
severity = severity + 15;
|
|
sprint(self, 2, "!ONE IN A MILLION\n");
|
|
}
|
|
|
|
if (attacker.class == 2)
|
|
severity = severity + 5;
|
|
|
|
if (targ.classname == "player")
|
|
damage = (damage * 2);
|
|
else if (severity >= 64)
|
|
damage = (damage * 10);
|
|
else if (severity >= 30)
|
|
damage = (damage * 6);
|
|
else if (severity >= 20)
|
|
damage = (damage * 5);
|
|
else if (severity >= 15)
|
|
damage = (damage * 4);
|
|
else
|
|
damage = (damage * 3);
|
|
|
|
if (targ.classname == "monster")
|
|
attacker.score = attacker.score + 1;
|
|
|
|
helm = targ.armortype;
|
|
if (targ.helmet == 0)
|
|
{
|
|
sound (targ, CHAN_VOICE, "player/headshot.wav", 1, ATTN_NORM);
|
|
helm = 0;
|
|
}
|
|
if (targ.helmet == 1)
|
|
{
|
|
sound (targ, CHAN_BODY, "weapons/helmet.wav", 1, ATTN_NORM);
|
|
helm = 0.30;
|
|
}
|
|
if (targ.helmet == 2)
|
|
{
|
|
sound (targ, CHAN_BODY, "weapons/helmet.wav", 1, ATTN_NORM);
|
|
helm = 0.45;
|
|
}
|
|
|
|
damage = (damage - (damage * helm));
|
|
|
|
makevectors (targ.v_angle);
|
|
}
|
|
}
|
|
|
|
if (damage <= 0 && (attacker.classname == "monster" || attacker.classname == "player"))
|
|
{
|
|
damage = 0;
|
|
sound (targ, CHAN_BODY, targ.armornoise, 1, ATTN_NORM);
|
|
return;
|
|
}
|
|
|
|
if (attacker.classname == "monster" || attacker.classname == "player")
|
|
sound (targ, CHAN_ITEM, targ.armornoise, 1, ATTN_NORM);
|
|
|
|
|
|
// 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 = targ.dmg_take + take;
|
|
targ.dmg_save = targ.dmg_save + save;
|
|
targ.dmg_inflictor = inflictor;
|
|
}
|
|
|
|
damage_inflictor = inflictor;
|
|
|
|
/*
|
|
// figure momentum add
|
|
if ( (inflictor != world) && (targ.movetype == MOVETYPE_WALK) )
|
|
{
|
|
dir = targ.origin - (inflictor.absmin + inflictor.absmax) * 0.5;
|
|
dir = normalize(dir);
|
|
// Set kickback for smaller weapons
|
|
//Zoid -- use normal NQ kickback
|
|
// // Read: only if it's not yourself doing the damage
|
|
// if ( (damage < 60) & ((attacker.classname == "player") & (targ.classname == "player")) & ( attacker.netname != targ.netname))
|
|
// targ.velocity = targ.velocity + dir * damage * 11;
|
|
// else
|
|
// Otherwise, these rules apply to rockets and grenades
|
|
// for blast velocity
|
|
targ.velocity = targ.velocity + dir * damage * 8;
|
|
|
|
// Rocket Jump modifiers
|
|
if ( (rj > 1) & ((attacker.classname == "player") & (targ.classname == "player")) & ( attacker.netname == targ.netname))
|
|
targ.velocity = targ.velocity + dir * damage * rj;
|
|
|
|
}*/
|
|
|
|
|
|
|
|
// check for godmode or invincibility
|
|
if (targ.flags & FL_GODMODE)
|
|
return;
|
|
if (targ.invincible_finished >= time)
|
|
{
|
|
if (self.invincible_sound < time)
|
|
{
|
|
sound (targ, CHAN_ITEM, "items/protect3.wav", 1, ATTN_NORM);
|
|
self.invincible_sound = time + 2;
|
|
}
|
|
return;
|
|
}
|
|
|
|
// team play damage avoidance
|
|
//ZOID 12-13-96: self.team doesn't work in QW. Use keys
|
|
attackerteam = infokey(attacker, "team");
|
|
targteam = infokey(targ, "team");
|
|
|
|
if ((teamplay == 1) && (targteam == attackerteam) &&
|
|
(attacker.classname == "player") && (attackerteam != "") &&
|
|
inflictor.classname !="door")
|
|
return;
|
|
|
|
if ((teamplay == 3) && (targteam == attackerteam) &&
|
|
(attacker.classname == "player") && (attackerteam != "") &&
|
|
(targ != attacker)&& inflictor.classname !="door")
|
|
return;
|
|
|
|
|
|
take = damage;
|
|
|
|
// do the damage
|
|
//different sorts of armour simply subtract different amounts
|
|
//this makes armor like the force armor good against many small rounds
|
|
//(SMG, shotguns, swords, etc) but useless against big damage (grenades)
|
|
//power armor, which has the best of both worlds, is also the heaviest :p
|
|
|
|
//fix: soldiers inherent armor bonus applies to armor absorb
|
|
//fix: psycho drug gives similar benefits to a soldiers bonus
|
|
|
|
if (targ.class == 3)
|
|
take -= 1;
|
|
if (self.rage == IID_CHEM_PSYCHO)
|
|
take -= 3;
|
|
|
|
switch(ToIID(targ.islot3))
|
|
{
|
|
case IID_ARM_DESERT:
|
|
take -= 2;
|
|
break;
|
|
case IID_ARM_LEATHER:
|
|
take -= 3;
|
|
break;
|
|
case IID_ARM_VANDAL:
|
|
take -= 4;
|
|
break;
|
|
case IID_ARM_METAL:
|
|
take -= 5;
|
|
break;
|
|
case IID_ARM_TESLA:
|
|
take -= 9;
|
|
break;
|
|
case IID_ARM_COMBAT:
|
|
take -= 6;
|
|
break;
|
|
case IID_ARM_SEVA:
|
|
take -= 8;
|
|
break;
|
|
case IID_ARM_FORCE:
|
|
take -= 13;
|
|
break;
|
|
case IID_ARM_LPOWER:
|
|
take -= 11;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
switch(ToIID(targ.islot3))
|
|
{
|
|
case IID_ARM_LEATHER:
|
|
take = take * 0.80;
|
|
break;
|
|
case IID_ARM_VANDAL:
|
|
take = take * 0.70;
|
|
break;
|
|
case IID_ARM_METAL:
|
|
take = take * 0.50;
|
|
break;
|
|
case IID_ARM_TESLA:
|
|
take = take * 0.65;
|
|
break;
|
|
case IID_ARM_COMBAT:
|
|
take = take * 0.60;
|
|
break;
|
|
case IID_ARM_SEVA:
|
|
take = take * 0.55;
|
|
break;
|
|
case IID_ARM_FORCE:
|
|
take = take * 0.90;
|
|
break;
|
|
case IID_ARM_LPOWER:
|
|
take = take * 0.50;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (targ.classname == "player" && targ.zoom > 0)
|
|
{
|
|
targ.maxspeed = targ.maxspeed * 0.33;
|
|
stuffcmd(targ, "v_idlescale 20\n");
|
|
}
|
|
else if (take > 20 && targ.classname == "player")
|
|
{
|
|
targ.maxspeed = targ.maxspeed * 0.25;
|
|
stuffcmd(targ, "v_cshift 100 0 0 100\n");
|
|
stuffcmd(targ, "v_idlescale 30\n");
|
|
}
|
|
else if (take > 0)
|
|
{
|
|
targ.maxspeed = targ.maxspeed * 0.50;
|
|
stuffcmd(targ, "v_cshift 100 0 0 50\n");
|
|
}
|
|
|
|
if (take <= 0)
|
|
{
|
|
take = 0;
|
|
sound (targ, CHAN_BODY, targ.armornoise, 1, ATTN_NORM);
|
|
return;
|
|
}
|
|
|
|
|
|
if (targ.hold <= 0)
|
|
targ.health = targ.health - take;
|
|
|
|
if (targ.classname == "monster" && targ.enemy.classname != "player" && attacker.classname == "player")
|
|
{
|
|
targ.enemy = attacker;
|
|
targ.think = HuntTarget;
|
|
targ.nextthink = time + 0.1;
|
|
}
|
|
|
|
local string lm;
|
|
|
|
if (targ.health <= 0)
|
|
{
|
|
|
|
//vanquishing a monster earns points for all players
|
|
if (attacker.classname == "player" && targ.classname == "monster")
|
|
{
|
|
local entity te;
|
|
te = findradius(attacker.origin, 1000);
|
|
while (te)
|
|
{
|
|
if (te.classname == "player")
|
|
te.score = te.score + 2;
|
|
|
|
te = te.chain;
|
|
}
|
|
}
|
|
|
|
Killed (targ, attacker);
|
|
if (attacker.perk1 == 11 || attacker.perk2 == 11)
|
|
{
|
|
attacker.lastman += 1.5;
|
|
lm = ftos(attacker.lastman);
|
|
sprint(self, 2, "!LAST MAN STANDING: ");
|
|
sprint(self, 2, lm);
|
|
sprint(self, 2, "\n");
|
|
|
|
attacker.health += attacker.lastman;
|
|
if (attacker.health > attacker.max_health)
|
|
attacker.health = attacker.max_health;
|
|
}
|
|
return;
|
|
}
|
|
|
|
// react to the damage
|
|
oldself = self;
|
|
self = targ;
|
|
|
|
//SERVER
|
|
|
|
if ( (self.flags & FL_MONSTER) && attacker != world)
|
|
{
|
|
// get mad unless of the same class (except for soldiers)
|
|
if (self != attacker && attacker != self.enemy)
|
|
{
|
|
if ((self.classname != attacker.classname))
|
|
{
|
|
if (self.enemy.classname == "player")
|
|
self.oldenemy = self.enemy;
|
|
self.enemy = attacker;
|
|
FoundTarget ();
|
|
}
|
|
}
|
|
}
|
|
|
|
if (self.th_pain)
|
|
{
|
|
self.th_pain (attacker, take);
|
|
}
|
|
|
|
self = oldself;
|
|
};
|
|
|
|
|
|
/*
|
|
============
|
|
J_Damage
|
|
|
|
This is a special function that is called for weapons using JHP ammo.
|
|
JHP ammo does extra damage against things wearing no armor or little
|
|
armor, but less damage against heavily armored foes.
|
|
============
|
|
*/
|
|
|
|
void(entity targ, entity inflictor, entity attacker, float damage) J_Damage =
|
|
{
|
|
local entity oldself;
|
|
local float save;
|
|
local float take, severity, helm;
|
|
local string attackerteam, targteam;
|
|
|
|
|
|
if (!targ.takedamage)
|
|
return;
|
|
|
|
damage = damage * 1.25;
|
|
damage = floor(damage);
|
|
|
|
// used by buttons and triggers to set activator for target firing
|
|
damage_attacker = attacker;
|
|
|
|
if (attacker.critical == 3 && targ.classname != "car")//attacker scored a headshot/critical
|
|
{
|
|
if (attacker.critical == 3)
|
|
{
|
|
severity = 0 + random()*20;
|
|
|
|
if (attacker.class == 3)
|
|
severity = severity + 2;
|
|
|
|
if (severity >= 19)
|
|
damage = (damage * 5);
|
|
else if (severity >= 14)
|
|
damage = (damage * 4);
|
|
else
|
|
damage = (damage * 3);
|
|
|
|
if (targ.classname == "monster")
|
|
attacker.score = attacker.score + 1;
|
|
|
|
helm = targ.armortype;
|
|
if (targ.helmet == 0)
|
|
{
|
|
sound (targ, CHAN_VOICE, "player/headshot.wav", 1, ATTN_NORM);
|
|
helm = 0;
|
|
}
|
|
if (targ.helmet == AS_STRAIGHT)
|
|
{
|
|
sound (targ, CHAN_BODY, "weapons/helmet.wav", 1, ATTN_NORM);
|
|
helm = 0.30;
|
|
}
|
|
if (targ.helmet == AS_SLIDING)
|
|
{
|
|
sound (targ, CHAN_BODY, "weapons/helmet.wav", 1, ATTN_NORM);
|
|
helm = 0.45;
|
|
}
|
|
|
|
damage = (damage - (damage * helm));
|
|
|
|
makevectors (targ.v_angle);
|
|
}
|
|
}
|
|
|
|
//sneak attack: if you attack a creature who isn't fighting you, bonus damage
|
|
if (attacker.perk1 == 7 || attacker.perk2 == 7)
|
|
{
|
|
if (targ.enemy != attacker)
|
|
{
|
|
sprint(attacker, 2, "!SNEAK ATTACK\n");
|
|
if (damage <= 10)
|
|
damage = damage * 3;
|
|
else if (damage <= 20)
|
|
damage = damage * 2;
|
|
else if (damage <= 30)
|
|
damage = damage * 1.5;
|
|
else
|
|
damage = damage * 1.25;
|
|
}
|
|
}
|
|
|
|
if (damage <= 0)
|
|
{
|
|
damage = 0;
|
|
sound (targ, CHAN_BODY, targ.armornoise, 1, ATTN_NORM);
|
|
return;
|
|
}
|
|
|
|
sound (targ, CHAN_ITEM, targ.armornoise, 1, ATTN_NORM);
|
|
|
|
|
|
// 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 = targ.dmg_take + take;
|
|
targ.dmg_save = targ.dmg_save + save;
|
|
targ.dmg_inflictor = inflictor;
|
|
}
|
|
|
|
damage_inflictor = inflictor;
|
|
|
|
/*
|
|
// figure momentum add
|
|
if ( (inflictor != world) && (targ.movetype == MOVETYPE_WALK) )
|
|
{
|
|
dir = targ.origin - (inflictor.absmin + inflictor.absmax) * 0.5;
|
|
dir = normalize(dir);
|
|
// Set kickback for smaller weapons
|
|
//Zoid -- use normal NQ kickback
|
|
// // Read: only if it's not yourself doing the damage
|
|
// if ( (damage < 60) & ((attacker.classname == "player") & (targ.classname == "player")) & ( attacker.netname != targ.netname))
|
|
// targ.velocity = targ.velocity + dir * damage * 11;
|
|
// else
|
|
// Otherwise, these rules apply to rockets and grenades
|
|
// for blast velocity
|
|
targ.velocity = targ.velocity + dir * damage * 8;
|
|
|
|
// Rocket Jump modifiers
|
|
if ( (rj > 1) & ((attacker.classname == "player") & (targ.classname == "player")) & ( attacker.netname == targ.netname))
|
|
targ.velocity = targ.velocity + dir * damage * rj;
|
|
|
|
}*/
|
|
|
|
|
|
|
|
// check for godmode or invincibility
|
|
if (targ.flags & FL_GODMODE)
|
|
return;
|
|
if (targ.invincible_finished >= time)
|
|
{
|
|
if (self.invincible_sound < time)
|
|
{
|
|
sound (targ, CHAN_ITEM, "items/protect3.wav", 1, ATTN_NORM);
|
|
self.invincible_sound = time + 2;
|
|
}
|
|
return;
|
|
}
|
|
|
|
// team play damage avoidance
|
|
//ZOID 12-13-96: self.team doesn't work in QW. Use keys
|
|
attackerteam = infokey(attacker, "team");
|
|
targteam = infokey(targ, "team");
|
|
|
|
if ((teamplay == 1) && (targteam == attackerteam) &&
|
|
(attacker.classname == "player") && (attackerteam != "") &&
|
|
inflictor.classname !="door")
|
|
return;
|
|
|
|
if ((teamplay == 3) && (targteam == attackerteam) &&
|
|
(attacker.classname == "player") && (attackerteam != "") &&
|
|
(targ != attacker)&& inflictor.classname !="door")
|
|
return;
|
|
|
|
|
|
take = damage;
|
|
|
|
// do the damage
|
|
//different sorts of armour simply subtract different ammounts
|
|
//this makes armor like the force armor good against many small rounds
|
|
//(SMG, shotguns, swords, etc) but useless against big damage (grenades)
|
|
//power armor, which has the best of both worlds, is also the heaviest :p
|
|
|
|
switch(ToIID(targ.islot3))
|
|
{
|
|
case IID_ARM_DESERT:
|
|
take -= 1;
|
|
break;
|
|
case IID_ARM_LEATHER:
|
|
take -= 1;
|
|
break;
|
|
case IID_ARM_VANDAL:
|
|
take -= 15;
|
|
break;
|
|
case IID_ARM_METAL:
|
|
take -= 10;
|
|
break;
|
|
case IID_ARM_TESLA:
|
|
take -= 25;
|
|
break;
|
|
case IID_ARM_COMBAT:
|
|
take -= 20;
|
|
break;
|
|
case IID_ARM_SEVA:
|
|
take -= 20;
|
|
break;
|
|
case IID_ARM_FORCE:
|
|
take -= 30;
|
|
break;
|
|
case IID_ARM_LPOWER:
|
|
take -= 25;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
switch(ToIID(targ.islot3))
|
|
{
|
|
case IID_ARM_LEATHER:
|
|
take = take * 0.70;
|
|
break;
|
|
case IID_ARM_VANDAL:
|
|
take = take * 0.60;
|
|
break;
|
|
case IID_ARM_METAL:
|
|
take = take * 0.40;
|
|
break;
|
|
case IID_ARM_TESLA:
|
|
take = take * 0.55;
|
|
break;
|
|
case IID_ARM_COMBAT:
|
|
take = take * 0.50;
|
|
break;
|
|
case IID_ARM_SEVA:
|
|
take = take * 0.45;
|
|
break;
|
|
case IID_ARM_FORCE:
|
|
take = take * 0.80;
|
|
break;
|
|
case IID_ARM_LPOWER:
|
|
take = take * 0.40;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (targ.classname == "player" && targ.zoom > 0)
|
|
{
|
|
targ.maxspeed = targ.maxspeed * 0.20;
|
|
stuffcmd(targ, "v_idlescale 20\n");
|
|
}
|
|
else if (take > 20 && targ.classname == "player")
|
|
{
|
|
targ.maxspeed = targ.maxspeed * 0.20;
|
|
stuffcmd(targ, "v_cshift 100 0 0 100\n");
|
|
stuffcmd(targ, "v_idlescale 30\n");
|
|
}
|
|
else if (take > 0)
|
|
{
|
|
targ.maxspeed = targ.maxspeed * 0.30;
|
|
stuffcmd(targ, "v_cshift 100 0 0 50\n");
|
|
}
|
|
|
|
|
|
if (take <= 0)
|
|
{
|
|
take = 0;
|
|
sound (targ, CHAN_BODY, targ.armornoise, 1, ATTN_NORM);
|
|
return;
|
|
}
|
|
|
|
|
|
targ.health = targ.health - take;
|
|
|
|
if (targ.health <= 0)
|
|
{
|
|
Killed (targ, attacker);
|
|
return;
|
|
}
|
|
|
|
// react to the damage
|
|
oldself = self;
|
|
self = targ;
|
|
|
|
/*SERVER
|
|
if ( (self.flags & FL_MONSTER) && attacker != world)
|
|
{
|
|
// get mad unless of the same class (except for soldiers)
|
|
if (self != attacker && attacker != self.enemy)
|
|
{
|
|
if ( (self.classname != attacker.classname)
|
|
|| (self.classname == "monster" ) )
|
|
{
|
|
if (self.enemy.classname == "player")
|
|
self.oldenemy = self.enemy;
|
|
self.enemy = attacker;
|
|
FoundTarget ();
|
|
}
|
|
}
|
|
}
|
|
*/
|
|
if (self.th_pain)
|
|
{
|
|
self.th_pain (attacker, take);
|
|
}
|
|
|
|
self = oldself;
|
|
};
|
|
|
|
/*
|
|
============
|
|
X_Damage
|
|
|
|
The purpose of X_Damage is to allow certain weapons to bypass armor
|
|
which is a very large part of the game.
|
|
============
|
|
*/
|
|
|
|
void(entity targ, entity inflictor, entity attacker, float damage) X_Damage =
|
|
{
|
|
local entity oldself;
|
|
local float save;
|
|
local float take, severity, helm;
|
|
local string attackerteam, targteam;
|
|
|
|
|
|
if (!targ.takedamage)
|
|
return;
|
|
|
|
damage = damage * 0.75;
|
|
damage = floor(damage);
|
|
|
|
// used by buttons and triggers to set activator for target firing
|
|
damage_attacker = attacker;
|
|
|
|
if (attacker.critical == 3 && targ.classname != "car")//attacker scored a headshot/critical
|
|
{
|
|
if (attacker.critical == 3)
|
|
{
|
|
severity = 0 + random()*20;
|
|
|
|
if (attacker.class == 2)
|
|
severity = severity + 4;
|
|
|
|
if (severity >= 19)
|
|
damage = (damage * 5);
|
|
else if (severity >= 14)
|
|
damage = (damage * 4);
|
|
else
|
|
damage = (damage * 3);
|
|
|
|
if (targ.classname == "monster")
|
|
attacker.score = attacker.score + 1;
|
|
|
|
helm = targ.armortype;
|
|
if (targ.helmet == 0)
|
|
{
|
|
sound (targ, CHAN_VOICE, "player/headshot.wav", 1, ATTN_NORM);
|
|
helm = 0;
|
|
}
|
|
if (targ.helmet == AS_STRAIGHT)
|
|
{
|
|
sound (targ, CHAN_BODY, "weapons/helmet.wav", 1, ATTN_NORM);
|
|
helm = 0.15;
|
|
}
|
|
if (targ.helmet == AS_SLIDING)
|
|
{
|
|
sound (targ, CHAN_BODY, "weapons/helmet.wav", 1, ATTN_NORM);
|
|
helm = 0.22;
|
|
}
|
|
|
|
damage = (damage - (damage * helm));
|
|
|
|
makevectors (targ.v_angle);
|
|
}
|
|
}
|
|
|
|
if (damage <= 0)
|
|
{
|
|
damage = 0;
|
|
sound (targ, CHAN_BODY, targ.armornoise, 1, ATTN_NORM);
|
|
return;
|
|
}
|
|
|
|
sound (targ, CHAN_BODY, targ.armornoise, 1, ATTN_NORM);
|
|
|
|
// save damage based on the target's armor level
|
|
|
|
|
|
|
|
take = ceil(damage);
|
|
|
|
// 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 = targ.dmg_take + take;
|
|
targ.dmg_save = targ.dmg_save + save;
|
|
targ.dmg_inflictor = inflictor;
|
|
}
|
|
|
|
damage_inflictor = inflictor;
|
|
|
|
/*
|
|
// figure momentum add
|
|
if ( (inflictor != world) && (targ.movetype == MOVETYPE_WALK) )
|
|
{
|
|
dir = targ.origin - (inflictor.absmin + inflictor.absmax) * 0.5;
|
|
dir = normalize(dir);
|
|
// Set kickback for smaller weapons
|
|
//Zoid -- use normal NQ kickback
|
|
// // Read: only if it's not yourself doing the damage
|
|
// if ( (damage < 60) & ((attacker.classname == "player") & (targ.classname == "player")) & ( attacker.netname != targ.netname))
|
|
// targ.velocity = targ.velocity + dir * damage * 11;
|
|
// else
|
|
// Otherwise, these rules apply to rockets and grenades
|
|
// for blast velocity
|
|
targ.velocity = targ.velocity + dir * damage * 8;
|
|
|
|
// Rocket Jump modifiers
|
|
if ( (rj > 1) & ((attacker.classname == "player") & (targ.classname == "player")) & ( attacker.netname == targ.netname))
|
|
targ.velocity = targ.velocity + dir * damage * rj;
|
|
|
|
}*/
|
|
|
|
|
|
|
|
// check for godmode or invincibility
|
|
if (targ.flags & FL_GODMODE)
|
|
return;
|
|
if (targ.invincible_finished >= time)
|
|
{
|
|
if (self.invincible_sound < time)
|
|
{
|
|
sound (targ, CHAN_ITEM, "items/protect3.wav", 1, ATTN_NORM);
|
|
self.invincible_sound = time + 2;
|
|
}
|
|
return;
|
|
}
|
|
|
|
// team play damage avoidance
|
|
//ZOID 12-13-96: self.team doesn't work in QW. Use keys
|
|
attackerteam = infokey(attacker, "team");
|
|
targteam = infokey(targ, "team");
|
|
|
|
if ((teamplay == 1) && (targteam == attackerteam) &&
|
|
(attacker.classname == "player") && (attackerteam != "") &&
|
|
inflictor.classname !="door")
|
|
return;
|
|
|
|
if ((teamplay == 3) && (targteam == attackerteam) &&
|
|
(attacker.classname == "player") && (attackerteam != "") &&
|
|
(targ != attacker)&& inflictor.classname !="door")
|
|
return;
|
|
|
|
|
|
// do the damage
|
|
//different sorts of armour simply subtract different ammounts
|
|
//this makes armor like the force armor good against many small rounds
|
|
//(SMG, shotguns, swords, etc) but useless against big damage (grenades)
|
|
//power armor, which has the best of both worlds, is also the heaviest :p
|
|
|
|
switch(ToIID(targ.islot3))
|
|
{
|
|
case IID_ARM_METAL:
|
|
take -= 1;
|
|
break;
|
|
case IID_ARM_SEVA:
|
|
take -= 1;
|
|
break;
|
|
case IID_ARM_FORCE:
|
|
take -= 3;
|
|
break;
|
|
case IID_ARM_LPOWER:
|
|
take -= 2;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
switch(ToIID(targ.islot3))
|
|
{
|
|
case IID_ARM_LEATHER:
|
|
take = take * 0.90;
|
|
break;
|
|
case IID_ARM_VANDAL:
|
|
take = take * 0.75;
|
|
break;
|
|
case IID_ARM_METAL:
|
|
take = take * 0.90;
|
|
break;
|
|
case IID_ARM_COMBAT:
|
|
take = take * 0.90;
|
|
break;
|
|
case IID_ARM_SEVA:
|
|
take = take * 0.80;
|
|
break;
|
|
case IID_ARM_LPOWER:
|
|
take = take * 0.75;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (targ.classname == "player" && targ.zoom > 0)
|
|
{
|
|
targ.maxspeed = targ.maxspeed * 0.33;
|
|
stuffcmd(targ, "v_idlescale 20\n");
|
|
}
|
|
else if (take > 20 && targ.classname == "player")
|
|
{
|
|
targ.maxspeed = targ.maxspeed * 0.25;
|
|
stuffcmd(targ, "v_cshift 100 0 0 100\n");
|
|
stuffcmd(targ, "v_idlescale 30\n");
|
|
}
|
|
else if (take > 0)
|
|
{
|
|
targ.maxspeed = targ.maxspeed * 0.50;
|
|
stuffcmd(targ, "v_cshift 100 0 0 50\n");
|
|
}
|
|
|
|
|
|
if (take <= 0)
|
|
{
|
|
take = 0;
|
|
sound (targ, CHAN_BODY, targ.armornoise, 1, ATTN_NORM);
|
|
return;
|
|
}
|
|
|
|
|
|
targ.health = targ.health - take;
|
|
|
|
if (targ.health <= 0)
|
|
{
|
|
Killed (targ, attacker);
|
|
return;
|
|
}
|
|
|
|
// react to the damage
|
|
oldself = self;
|
|
self = targ;
|
|
|
|
/*SERVER
|
|
if ( (self.flags & FL_MONSTER) && attacker != world)
|
|
{
|
|
// get mad unless of the same class (except for soldiers)
|
|
if (self != attacker && attacker != self.enemy)
|
|
{
|
|
if ( (self.classname != attacker.classname)
|
|
|| (self.classname == "monster" ) )
|
|
{
|
|
if (self.enemy.classname == "player")
|
|
self.oldenemy = self.enemy;
|
|
self.enemy = attacker;
|
|
FoundTarget ();
|
|
}
|
|
}
|
|
}
|
|
*/
|
|
if (self.th_pain)
|
|
{
|
|
self.th_pain (attacker, take);
|
|
}
|
|
|
|
self = oldself;
|
|
};
|
|
|
|
/*
|
|
============
|
|
T_RadiusDamage
|
|
============
|
|
*/
|
|
void(entity inflictor, entity attacker, float damage, entity ignore, string dtype) T_RadiusDamage =
|
|
{
|
|
local float points, srange, brange;
|
|
local entity head;
|
|
local vector org;
|
|
|
|
head = findradius(inflictor.origin, damage+40);
|
|
|
|
while (head)
|
|
{
|
|
//bprint (PRINT_HIGH, head.classname);
|
|
//bprint (PRINT_HIGH, " | ");
|
|
//bprint (PRINT_HIGH, head.netname);
|
|
//bprint (PRINT_HIGH, "\n");
|
|
|
|
if (head != ignore)
|
|
{
|
|
if (head.takedamage)
|
|
{
|
|
org = head.origin + (head.mins + head.maxs)*0.5;
|
|
points = vlen(inflictor.origin - org);
|
|
points = points + damage*0.50;
|
|
if (points < 0)
|
|
points = 0;
|
|
points = damage - points;
|
|
|
|
if (points > 0)
|
|
{
|
|
if (CanDamage (head, inflictor))
|
|
{
|
|
head.deathtype = dtype;
|
|
T_Damage (head, inflictor, attacker, points);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
head = head.chain;
|
|
}
|
|
};
|
|
|
|
|
|
void (entity inflictor, entity attacker, float damage, entity ignore, float xrange) T_RadiusDamage2 =
|
|
{
|
|
local float points;
|
|
local entity head;
|
|
local vector org;
|
|
local float srange;
|
|
local float sub;
|
|
local float damaget;
|
|
|
|
head = findradius (inflictor.origin, xrange);
|
|
while (head)
|
|
{
|
|
if (head != ignore)
|
|
{
|
|
if (head.takedamage)
|
|
{
|
|
org = head.origin;
|
|
srange = vlen ((inflictor.origin - org));
|
|
points = (xrange / srange);
|
|
sub = (1 / points);
|
|
damaget = (damage - (damage * sub));
|
|
if (damaget > 0)
|
|
if (CanDamage (head, inflictor))
|
|
T_Damage (head, inflictor, attacker, damaget);
|
|
}
|
|
}
|
|
head = head.chain;
|
|
}
|
|
};
|
|
|
|
/*
|
|
============
|
|
T_BeamDamage
|
|
============
|
|
*/
|
|
void(entity attacker, float damage) T_BeamDamage =
|
|
{
|
|
local float points;
|
|
local entity head;
|
|
|
|
head = findradius(attacker.origin, damage+40);
|
|
|
|
while (head)
|
|
{
|
|
if (head.takedamage)
|
|
{
|
|
points = 0.5*vlen (attacker.origin - head.origin);
|
|
if (points < 0)
|
|
points = 0;
|
|
points = damage - points;
|
|
if (head == attacker)
|
|
points = points * 0.5;
|
|
if (points > 0)
|
|
{
|
|
if (CanDamage (head, attacker))
|
|
T_Damage (head, attacker, attacker, points);
|
|
}
|
|
}
|
|
head = head.chain;
|
|
}
|
|
};
|
|
|