mirror of
https://git.code.sf.net/p/quake/prozac-qfcc
synced 2025-01-18 23:51:53 +00:00
8c2d5fdd12
should get frags for blowing people up with others' dispensers/mines/expbody, airfisting rockets, etc. 2) Redid Give_Frags_Out 3) Changed many uses of pointcontents and ugly hacks to use hullpointcontents and checkmove, now a lot cleaner and less buggy 4) You can grapple builds again. This caused really odd bugs. 5) You can now damage your own buildings again (gasp). Any time a tesla or sentry is damaged it turns on its attacker, friendly or otherwise. This is both a counter-TK and counter-spy mechanism. 6) Teslas are now entirely inside their bounding box 7) Now check every frame for players startsolid and for outside of the map cube. 8) #define WALL_HURT to make it hurt when you hit a wall 9) Used some cool ideas (aka laws of physics) to make the airfist a little less annoying 10) You now only get 1 mirv per slot without bandolier. Demoman and hwguy gain bandolier. demoman loses his extra mirv but gains an extra grenade and pair of detpacks. Hwguy gets extra grenade. 11) New and improved EMP grenade now does damage based on range. no longer blows up shells. Doesn't directly damage sentries anymore, but does significant damage to dispensers. EMP someone who's setting a det and it blows up in their face. 12) Players now do radius damage from getting EMPed (again) 13) EMPs now go through walls (again) 14) EMP number lowered by one (3 without, 4 with bandolier) and cost raised by $100 15) You can only have 2 frag grens, 3 with bandolier now. 16) Hover boots will now eat cells if they get low on charge. In addition, the silly bug where their charge wasn't restored when you die is fixed now. 17) EMPing a detpack now sets its timer to anywhere between 1 and 121 seconds from current time, with a logarithmic probability of it being lower. (random() * random() * 120 + 1). Also, probably more time the closer it is to the EMP. 18) Judo can now be blocked by people with close combat, knife or judo. Blocked judo means that the attacker loses 3 seconds of attack. 19) Judo missing now makes them unable to fire. 20) Shortened judo range (back to normal if w/ close combat) 21) Attempted to rework the railgun. Seems to be okay now. Probably still a lot of bugs in here, but since this is the devel version I thought I would commit all my changes so people could start testing it. I'll commit fixes as soon as I find the bugs.
542 lines
14 KiB
C++
542 lines
14 KiB
C++
#include "defs.qh"
|
|
/*======================================================
|
|
DEMOMAN.QC
|
|
|
|
TeamFortress v2.5 29/2/97
|
|
========================================================
|
|
Functions for the DEMOMAN class and associated weaponry
|
|
========================================================*/
|
|
// Functions outside this file
|
|
void() NormalGrenadeTouch;
|
|
void() NormalGrenadeExplode;
|
|
|
|
// Functions in this file
|
|
// Pipebomb Functions
|
|
void() TeamFortress_DetonatePipebombs;
|
|
void() PipebombTouch;
|
|
// Mirv Grenade Functions
|
|
void() MirvGrenadeTouch;
|
|
void() MirvGrenadeExplode;
|
|
void (vector org, entity shooter) MirvGrenadeLaunch;
|
|
// Detpack Functions
|
|
void(float timer) TeamFortress_SetDetpack;
|
|
void() TeamFortress_DetpackSet;
|
|
void(float krac) TeamFortress_DetpackStop;
|
|
void() TeamFortress_DetpackExplode;
|
|
void() TeamFortress_DetpackTouch;
|
|
void() TeamFortress_DetpackDisarm;
|
|
void() TeamFortress_DetpackCountDown;
|
|
|
|
//=========================================================================
|
|
// Detonate all thrown pipebombs
|
|
void() TeamFortress_DetonatePipebombs =
|
|
{
|
|
local entity e;
|
|
|
|
// Find all this players pipebombs
|
|
e = find(NIL, classname, "pipebomb");
|
|
while (e)
|
|
{
|
|
if(e.owner == self)
|
|
e.nextthink = time;
|
|
|
|
e = find(e, classname, "pipebomb");
|
|
}
|
|
};
|
|
|
|
//=========================================================================
|
|
// Pipebomb touch function
|
|
void() PipebombTouch =
|
|
{
|
|
sound (self, CHAN_WEAPON, "weapons/bounce.wav", 1, ATTN_NORM); // bounce sound
|
|
if (self.velocity == '0 0 0')
|
|
self.avelocity = '0 0 0';
|
|
};
|
|
|
|
//=========================================================================
|
|
// Touch Function for Mirv Grenade
|
|
// Mirv Grenade heavily influenced by the Firewall Grenade by Steve Bond (wedge@nuc.net)
|
|
void() MirvGrenadeTouch =
|
|
{
|
|
// If the Mirv Grenade hits a player, it just bounces off
|
|
sound (self, CHAN_WEAPON, "weapons/bounce.wav", 1, ATTN_NORM);
|
|
if (self.velocity == '0 0 0')
|
|
self.avelocity = '0 0 0';
|
|
};
|
|
|
|
//=========================================================================
|
|
// Mirv Grenade explode function, for when the PRIMETIME runs out
|
|
void() MirvGrenadeExplode =
|
|
{
|
|
local float i;
|
|
|
|
deathmsg = DMSG_GREN_MIRV;
|
|
T_RadiusDamage (self, self.owner, 100, NIL);
|
|
|
|
WriteByte (MSG_MULTICAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_MULTICAST, TE_EXPLOSION);
|
|
WriteCoord (MSG_MULTICAST, self.origin_x);
|
|
WriteCoord (MSG_MULTICAST, self.origin_y);
|
|
WriteCoord (MSG_MULTICAST, self.origin_z);
|
|
multicast (self.origin, MULTICAST_PHS);
|
|
|
|
self.solid = SOLID_NOT;
|
|
|
|
// Launch mirvs
|
|
i = 0;
|
|
while (i < GR_TYPE_MIRV_NO)
|
|
{
|
|
MirvGrenadeLaunch (self.origin + '0 0 -1',self.owner);
|
|
i = i + 1;
|
|
}
|
|
|
|
#ifdef DEMO_STUFF
|
|
// Remove any camera's locks on this missile
|
|
if (self.enemy)
|
|
CamProjectileLockOff();
|
|
#endif
|
|
|
|
BecomeExplosion();
|
|
};
|
|
|
|
//=========================================================================
|
|
// Launch a Mirv
|
|
void (vector org, entity shooter) MirvGrenadeLaunch =
|
|
{
|
|
local float xdir,ydir,zdir;
|
|
|
|
xdir = 150 * random() - 75;
|
|
ydir = 2 * (100 - fabs(xdir)) * random() - (75 - fabs(xdir));
|
|
zdir = 170 - (fabs(xdir) + fabs(ydir));
|
|
|
|
newmis = spawn ();
|
|
newmis.owner = shooter;
|
|
newmis.movetype = MOVETYPE_BOUNCE;
|
|
newmis.solid = SOLID_BBOX;
|
|
|
|
newmis.classname = "grenade";
|
|
newmis.weapon = DMSG_GREN_MIRV;
|
|
|
|
#ifdef NET_SERVER
|
|
newmis.touch = GrenadeTouch;
|
|
newmis.think = GrenadeExplode;
|
|
#else
|
|
newmis.touch = GrenadeTouch;
|
|
newmis.think = GrenadeExplode;
|
|
#endif
|
|
|
|
//WK 2 + random
|
|
newmis.nextthink = time + 0.75 + random();
|
|
|
|
newmis.velocity_x = xdir * 3;
|
|
newmis.velocity_y = ydir * 3;
|
|
newmis.velocity_z = zdir * 5;
|
|
|
|
newmis.avelocity='100 100 500';
|
|
newmis.AIRG_Flags = self.AIRG_Flags; //for tk stuff
|
|
newmis.AIRG_FlyTracker = self.AIRG_FlyTracker;
|
|
|
|
setmodel (newmis, "progs/grenade2.mdl");
|
|
setsize (newmis, VEC_ORIGIN, VEC_ORIGIN);
|
|
setorigin (newmis, org);
|
|
};
|
|
|
|
//=========================================================================
|
|
// Handles the Setting of Detpacks
|
|
void(float timer) TeamFortress_SetDetpack =
|
|
{
|
|
local string stimer;
|
|
|
|
// prevent detpack impulse from triggering anything else
|
|
self.impulse = 0;
|
|
self.last_impulse = 0;
|
|
|
|
if (!(self.cutf_items & CUTF_DETPACK))
|
|
return;
|
|
|
|
if (self.ammo_detpack <= 0)
|
|
return;
|
|
|
|
if (infokey (NIL, "no_spam") == "on")
|
|
{
|
|
sprint(self,PRINT_HIGH,"The admin has disabled spam devices on this map.\n");
|
|
return;
|
|
}
|
|
// Cant set detpack if you're in the air
|
|
if (!(self.flags & FL_ONGROUND))
|
|
{
|
|
sprint (self, PRINT_HIGH, "You can't set detpacks in the air!\n");
|
|
return;
|
|
}
|
|
//WK Prevents spy sliding bug
|
|
if (self.is_feigning)
|
|
{
|
|
sprint (self, PRINT_HIGH, "You can't set detpacks while playing dead!\n");
|
|
return;
|
|
}
|
|
if (self.is_haxxxoring)
|
|
{
|
|
sprint (self, PRINT_HIGH, "You can't set a detpack while you're hacking something.\n");
|
|
return;
|
|
}
|
|
if (timer < 5)
|
|
{
|
|
sprint (self, PRINT_HIGH, "You can't set detpacks for less that 5 seconds.\n");
|
|
return;
|
|
}
|
|
|
|
if (self.is_detpacking || self.is_toffingadet)
|
|
return;
|
|
|
|
self.is_detpacking = 1;
|
|
self.ammo_detpack = self.ammo_detpack - 1;
|
|
|
|
makeImmune(self,time + 2);
|
|
//self.immune_to_check = time + 2;
|
|
self.tfstate = self.tfstate | TFSTATE_CANT_MOVE;
|
|
// Save the current weapon and remove it
|
|
self.weapon = self.current_weapon;
|
|
self.current_weapon = 0;
|
|
self.weaponmodel = "";
|
|
self.weaponframe = 0;
|
|
|
|
TeamFortress_SetSpeed(self);
|
|
|
|
self.pausetime = time + WEAP_DETPACK_SETTIME;
|
|
|
|
stimer = ftos(timer);
|
|
|
|
sprint(self, PRINT_HIGH, "Setting detpack for ");
|
|
sprint(self, PRINT_HIGH, stimer);
|
|
sprint(self, PRINT_HIGH, " seconds...\n");
|
|
|
|
newmis = spawn();
|
|
newmis.owner = self;
|
|
newmis.classname = "timer";
|
|
newmis.netname = "detpack_timer";
|
|
newmis.movetype = MOVETYPE_NONE; //WK M3 Bug Hunt
|
|
newmis.nextthink = time + WEAP_DETPACK_SETTIME;
|
|
newmis.think = TeamFortress_DetpackSet;
|
|
newmis.health = timer;
|
|
|
|
//CH because its on the sbar :)
|
|
self.StatusRefreshTime = time + 0.2;
|
|
self.StatusBarScreen = 3;
|
|
};
|
|
|
|
//=========================================================================
|
|
// Stops the setting of the detpack
|
|
void(float krac) TeamFortress_DetpackStop =
|
|
{
|
|
local entity detpack_timer;
|
|
|
|
detpack_timer = find(NIL, netname, "detpack_timer");
|
|
while ((detpack_timer.owner != self) && (detpack_timer))
|
|
detpack_timer = find(detpack_timer, netname, "detpack_timer");
|
|
|
|
if (!detpack_timer)
|
|
return;
|
|
|
|
if (krac == TRUE) {
|
|
sprint(self, PRINT_HIGH, "Your detpack got Kraced!\n");
|
|
} else {
|
|
sprint(self, PRINT_HIGH, "Detpack retrieved.\n");
|
|
self.ammo_detpack = self.ammo_detpack + 1;
|
|
}
|
|
|
|
dremove(detpack_timer);
|
|
|
|
self.tfstate = self.tfstate - (self.tfstate & TFSTATE_CANT_MOVE);
|
|
|
|
self.is_detpacking = 0;
|
|
self.current_weapon = self.weapon;
|
|
W_SetCurrentAmmo();
|
|
|
|
TeamFortress_SetSpeed(self);
|
|
|
|
self.pausetime = time;
|
|
};
|
|
|
|
//=========================================================================
|
|
// The detpack is set, let the player go and start timer
|
|
void() TeamFortress_DetpackSet =
|
|
{
|
|
local entity countd, oldself;
|
|
|
|
self.is_detpacking = 0;
|
|
self.owner.tfstate = self.owner.tfstate - (self.owner.tfstate & TFSTATE_CANT_MOVE);
|
|
self.owner.is_detpacking = 0;
|
|
|
|
TeamFortress_SetSpeed(self.owner);
|
|
|
|
#ifdef SPEECH
|
|
stuffcmd(self.owner, "play speech/demo_dp.wav\n");
|
|
#endif
|
|
|
|
oldself = self;
|
|
self = self.owner;
|
|
//self.is_detpacking = 0;
|
|
self.current_weapon = self.weapon;
|
|
W_SetCurrentAmmo();
|
|
self = oldself;
|
|
|
|
newmis = spawn ();
|
|
newmis.owner = self.owner;
|
|
newmis.origin = self.owner.origin - '0 0 23';
|
|
newmis.movetype = MOVETYPE_BOUNCE;
|
|
newmis.solid = SOLID_BBOX;
|
|
newmis.classname = "detpack";
|
|
newmis.flags = FL_ITEM;
|
|
|
|
newmis.angles = '90 0 0';
|
|
newmis.angles_y = self.owner.angles_y;
|
|
newmis.velocity = '0 0 0';
|
|
newmis.avelocity = '0 0 0';
|
|
|
|
newmis.weaponmode = 0; // Detpack weaponmode = 1 when disarming
|
|
newmis.touch = TeamFortress_DetpackTouch;
|
|
|
|
setmodel (newmis, "progs/detpack.mdl");
|
|
setsize (newmis, '-16 -16 0', '16 16 8');
|
|
setorigin (newmis, self.owner.origin);
|
|
|
|
sound (newmis, CHAN_WEAPON, "doors/medtry.wav", 1, ATTN_NORM); //CH play set detpack sound
|
|
|
|
|
|
// Create the CountDown entity
|
|
countd = spawn();
|
|
newmis.linked_list = countd; // attach count to its detpack
|
|
countd.think = TeamFortress_DetpackCountDown;
|
|
countd.health = self.health - 1;
|
|
countd.owner = self.owner;
|
|
countd.movetype = MOVETYPE_NONE; //WK M3 Bug Hunt
|
|
countd.classname = "countdown_timer"; // Don't call it timer, because we
|
|
// don't want it removed if player dies
|
|
countd.enemy = newmis;
|
|
newmis.oldenemy = countd;
|
|
if (self.health <= 10)
|
|
countd.nextthink = time + 1;
|
|
else
|
|
{
|
|
countd.nextthink = time + self.health - 10;
|
|
countd.health = 9;
|
|
}
|
|
newmis.nextthink = time + self.health;
|
|
newmis.think = TeamFortress_DetpackExplode;
|
|
|
|
sprint(self.owner, PRINT_HIGH, "Detpack set!\n");
|
|
|
|
teamprefixsprint(self.team_no,self);
|
|
teamsprint6(self,self.netname," sets a detpack!\n","","","","");
|
|
|
|
dremove(self);
|
|
};
|
|
|
|
//- OfN - Used for extra det explosions
|
|
float (vector targ, vector check) vis2orig;
|
|
|
|
|
|
//=========================================================================
|
|
// The detpack goes BOOM!
|
|
void() TeamFortress_DetpackExplode =
|
|
{
|
|
local integer pos;
|
|
|
|
bprint(PRINT_MEDIUM, "FIRE IN THE HOLE!\n");
|
|
|
|
// Check the pointcontents to prevent detpack outside the world
|
|
pos = entpointcontents(self);
|
|
if (pos != CONTENTS_SOLID && pos != CONTENTS_SKY)
|
|
{
|
|
deathmsg = DMSG_DETPACK;
|
|
T_RadiusDamage (self, self.owner, WEAP_DETPACK_SIZE, NIL);
|
|
|
|
sound(self, CHAN_MISC, "weapons/detpack.wav", 1, ATTN_NONE);
|
|
|
|
WriteByte (MSG_MULTICAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_MULTICAST, TE_EXPLOSION);
|
|
WriteCoord (MSG_MULTICAST, self.origin_x);
|
|
WriteCoord (MSG_MULTICAST, self.origin_y);
|
|
WriteCoord (MSG_MULTICAST, self.origin_z);
|
|
multicast (self.origin, MULTICAST_PHS);
|
|
|
|
WriteByte (MSG_MULTICAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_MULTICAST, TE_LAVASPLASH);
|
|
WriteCoord (MSG_MULTICAST, self.origin_x);
|
|
WriteCoord (MSG_MULTICAST, self.origin_y);
|
|
WriteCoord (MSG_MULTICAST, self.origin_z);
|
|
multicast (self.origin, MULTICAST_PHS);
|
|
|
|
//- OfN - Should a detpack make the same explosion a gren does? dont think so...
|
|
#ifdef extra_detpack_explosions
|
|
|
|
local float loopc;
|
|
local vector rexp;
|
|
|
|
loopc = 0;
|
|
|
|
while (loopc < EXTRA_DETEXPOSIONS)
|
|
{
|
|
rexp_x = self.origin_x + random()*400 - 200;
|
|
rexp_y = self.origin_y + random()*400 - 200;
|
|
rexp_z = self.origin_z + random()*100;
|
|
|
|
if (vis2orig(self.origin,rexp))
|
|
{
|
|
WriteByte (MSG_MULTICAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_MULTICAST, TE_EXPLOSION);
|
|
WriteCoord (MSG_MULTICAST, rexp_x);
|
|
WriteCoord (MSG_MULTICAST, rexp_y);
|
|
WriteCoord (MSG_MULTICAST, rexp_z);
|
|
multicast (rexp, MULTICAST_PHS);
|
|
}
|
|
loopc = loopc + 1;
|
|
}
|
|
#endif
|
|
|
|
//- OfN - Moves a bit the players that are near the exploding detpack
|
|
#ifdef detpack_earthquake
|
|
|
|
local entity head;
|
|
|
|
local vector v;
|
|
|
|
head = findradius(self.origin, DETPACK_EARTHQUAKE_RANGE);
|
|
|
|
while (head)
|
|
{
|
|
if (head.classname == "player")
|
|
if (head.health > 0)
|
|
if (!head.has_disconnected)
|
|
if (head.playerclass != PC_UNDEFINED)
|
|
if (head.flags & FL_ONGROUND || head.waterlevel)
|
|
{
|
|
v_x = 100 * crandom();
|
|
v_y = 100 * crandom();
|
|
v_z = 150 + 200 * random();
|
|
head.velocity = head.velocity + v;
|
|
}
|
|
|
|
head = head.chain;
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
}
|
|
else
|
|
{
|
|
// bprint(PRINT_HIGH, "Your detpack fizzled out.\n");
|
|
sprint(self.owner, PRINT_HIGH, "Your detpack fizzled out.\n");
|
|
}
|
|
|
|
// This code handles a disarming scout with protection
|
|
if (self.weaponmode == 1) // Detpack was being disarmed
|
|
{
|
|
TeamFortress_SetSpeed(self.enemy);
|
|
|
|
dremove(self.oldenemy); // CountDown
|
|
dremove(self.observer_list); // Disarm timer
|
|
}
|
|
|
|
BecomeExplosion ();
|
|
};
|
|
|
|
//=========================================================================
|
|
// The detpack touch function. Scouts can disarm it.
|
|
void() TeamFortress_DetpackTouch =
|
|
{
|
|
local entity disarm;
|
|
|
|
if (other.classname != "player")
|
|
return;
|
|
|
|
//WK if (other.playerclass != PC_SCOUT)
|
|
if (!(other.tf_items & NIT_SCANNER))
|
|
return;
|
|
|
|
if (self.weaponmode == 1)
|
|
return;
|
|
|
|
if (Teammate(self.owner.team_no,other.team_no))
|
|
return;
|
|
|
|
makeImmune(other,time + 2);
|
|
//other.immune_to_check = time + 2;
|
|
other.tfstate = other.tfstate | TFSTATE_CANT_MOVE;
|
|
|
|
sprint(other, PRINT_HIGH, "Disarming detpack...\n");
|
|
TeamFortress_SetSpeed(other);
|
|
|
|
// Spawn disarming entity
|
|
disarm = spawn();
|
|
disarm.movetype = MOVETYPE_NONE; //WK M3 Bug Hunt
|
|
disarm.owner = other; // the scout
|
|
disarm.enemy = self; // the detpack
|
|
disarm.classname = "timer";
|
|
disarm.nextthink = time + WEAP_DETPACK_DISARMTIME;
|
|
disarm.think = TeamFortress_DetpackDisarm;
|
|
|
|
self.weaponmode = 1; // indicates disarming
|
|
self.enemy = other; // points to scout
|
|
self.observer_list = disarm;
|
|
};
|
|
|
|
//=========================================================================
|
|
// The detpack disarm function. Scout has finished disarming it
|
|
// .enemy is the detpack
|
|
// .owner is the scout
|
|
void() TeamFortress_DetpackDisarm =
|
|
{
|
|
if (self.owner.health <= 0) {
|
|
dremove(self);
|
|
return;
|
|
}
|
|
|
|
bprint (PRINT_MEDIUM, self.enemy.owner.netname);
|
|
bprint (PRINT_MEDIUM, "'s detpack was defused by ");
|
|
bprint (PRINT_MEDIUM, self.owner.netname);
|
|
bprint (PRINT_MEDIUM, "\n");
|
|
|
|
//if (!(self.owner.is_feigning))
|
|
//- OfN already checked on TeamFortress_SetSpeed()
|
|
//{
|
|
self.owner.tfstate = self.owner.tfstate - (self.owner.tfstate & TFSTATE_CANT_MOVE);
|
|
|
|
// Reset speeds of scout
|
|
TeamFortress_SetSpeed(self.owner);
|
|
//}
|
|
|
|
dremove(self.enemy.oldenemy); // remove count down
|
|
dremove(self.enemy); // remove detpack
|
|
dremove(self); // remove this timer
|
|
};
|
|
|
|
//=========================================================================
|
|
// The Detpack CountDown function. Displays the seconds left before the
|
|
// detpack detonates to the owner of the detpack, if <10
|
|
void() TeamFortress_DetpackCountDown =
|
|
{
|
|
local string cd;
|
|
|
|
cd = ftos(self.health);
|
|
|
|
sprint(self.owner, PRINT_HIGH, cd);
|
|
sprint(self.owner, PRINT_HIGH, "...\n");
|
|
|
|
self.nextthink = time + 1;
|
|
self.health = self.health - 1;
|
|
|
|
// Flash the red light
|
|
if (self.health <= 4)
|
|
self.enemy.skin = 1;
|
|
|
|
if (self.health <= 3 && self.health > 2) { //WK Detpack alerts
|
|
sound (self.owner, CHAN_VOICE, "weapons/fith.wav", 1, ATTN_NORM);
|
|
sound (self.enemy, CHAN_WEAPON, "doors/baseuse.wav", 1, ATTN_NORM); //CH play set detpack sound
|
|
}
|
|
|
|
if (self.health == 0)
|
|
dremove(self);
|
|
};
|
|
|