mirror of
https://git.code.sf.net/p/quake/prozac-qfcc
synced 2025-01-18 23:51:53 +00:00
4727 lines
127 KiB
C++
4727 lines
127 KiB
C++
|
|
/*======================================================
|
|
WEAPONS.QC Custom TeamFortress v3.1
|
|
|
|
(c) TeamFortress Software Pty Ltd 29/2/97
|
|
(c) William Kerney 5/21/00
|
|
========================================================
|
|
All the functions for firing all the weapons, holds
|
|
all new precache() functions, finding best weapons,
|
|
choosing next weapon, and all impulse commands.
|
|
======================================================*/
|
|
#include "defs.qh"
|
|
#include "menu.qh"
|
|
|
|
void() SniperSight_Update2; //CH for RL
|
|
void() I_DID_CHEAT_ONE; //CH for speed.qc
|
|
void() I_DID_CHEAT_TWO;
|
|
void() I_DID_CHEAT_THREE;
|
|
void(string gibname, float dm) ThrowGib;
|
|
void (entity targ, entity inflictor, entity attacker, float damage) T_Damage;
|
|
void (entity targ, entity inflictor, entity attacker, float damage, float T_flags, float T_AttackType) TF_T_Damage;
|
|
void () player_run;
|
|
void(entity bomb, entity attacker, float rad, entity ignore) T_RadiusDamage;
|
|
void() TeamFortress_DisplayDetectionItems;
|
|
vector(vector veca, vector vecb) crossproduct;
|
|
void(vector org, float damage) SpawnBlood;
|
|
void(entity rhook) Reset_Grapple;
|
|
void() SuperDamageSound;
|
|
float() W_BestWeapon;
|
|
void() ConcussionGrenadeTimer;
|
|
void() W_PrintWeaponMessage;
|
|
void() button_touch;
|
|
void() button_fire;
|
|
void() MauserRecoilThink;
|
|
float() ReturnWeaponVelocity;
|
|
vector() Grunty_LeadShot;
|
|
|
|
// TeamFortress Impulse Commands
|
|
void() TeamFortress_ChangeClass;
|
|
void() TeamFortress_DisplayLegalClasses;
|
|
void() TeamFortress_Inventory;
|
|
void() TeamFortress_SaveMe;
|
|
void(float inAuto) TeamFortress_ID;
|
|
void() TeamFortress_ShowTF;
|
|
void() TeamFortress_SniperWeapon;
|
|
void() TeamFortress_AssaultWeapon;
|
|
void() TeamFortress_IncendiaryCannon;
|
|
void() TeamFortress_FlameThrower;
|
|
void() TeamFortress_PrimeGrenade;
|
|
void() TeamFortress_ThrowGrenade;
|
|
void() TeamFortress_Discard;
|
|
void(entity p) TeamFortress_SetSpeed;
|
|
void() TeamFortress_DetonatePipebombs;
|
|
void() PipebombTouch;
|
|
void(float foo) TeamFortress_DetpackStop;
|
|
void(float type) SniperSight_Create;
|
|
void(float zoom_level) TF_zoom;
|
|
void() TeamFortress_ReloadCurrentWeapon;
|
|
void() TeamFortress_AutoZoomToggle;
|
|
void() TeamFortress_StatusQuery;
|
|
void() TeamFortress_SpyGoUndercover;
|
|
void(float type) TeamFortress_SpyFeignDeath;
|
|
void() TeamFortress_EngineerBuild;
|
|
void() DropKey;
|
|
void() UseSpecialSkill;
|
|
void() UseJobSkill; //WK Function for handling professions
|
|
void (entity targ, float pain) RevealThief;
|
|
void(vector startpos) GuerillaMineSweep;
|
|
void(entity foo,float bar) makeImmune;
|
|
float() CheckForReload;
|
|
void() SBBuildSensor;
|
|
void() SBFireInterface;
|
|
//void() SBInitiateInterface;
|
|
void() W_FireMauser;
|
|
void() W_FireDaedalus;
|
|
|
|
// TeamFortress Pre-Impulse Commands
|
|
void(float scanrange,float inAuto) TeamFortress_Scan;
|
|
void(float timer) TeamFortress_SetDetpack;
|
|
|
|
// Team Functions
|
|
float(float tno) TeamFortress_TeamSet;
|
|
void(float tno) TeamFortress_TeamShowScores;
|
|
void(entity Player) TeamFortress_TeamShowMemberClasses;
|
|
|
|
#ifdef DEMO_STUFF
|
|
// Camera Functions
|
|
void() CamLock;
|
|
void() CamDistLock;
|
|
void() CamVecLock;
|
|
void() CamAngleLock;
|
|
void() CamRevAngleLock;
|
|
void() CamProjectileLock;
|
|
void() CamProjectileZoom;
|
|
void() CamProjectileLockOn;
|
|
void() CamProjectileLockOff;
|
|
void() CamOffset;
|
|
void() CamDrop;
|
|
void() fadetoblack;
|
|
void() fadefromblack;
|
|
void() fadetowhite;
|
|
void() fadefromwhite;
|
|
#endif
|
|
|
|
// Engineer Functions
|
|
void(entity disp) Engineer_UseDispenser;
|
|
void(entity cam) Engineer_UseSensor;
|
|
void(entity gun) Engineer_UseSentryGun;
|
|
void(entity gun) Engineer_UseTesla;
|
|
void(entity cam) Engineer_UseCamera;
|
|
void(entity tele) Engineer_UseTeleporter;
|
|
void(entity field) Engineer_UseFieldGen;
|
|
void() Engineer_AutoUse;
|
|
|
|
void(entity spy) Spy_RemoveDisguise;
|
|
|
|
// Help functions
|
|
void() TeamFortress_MOTD;
|
|
void() TeamFortress_HelpMap;
|
|
void(float res) StatusRes;
|
|
|
|
// BioInfection functions
|
|
void() BioInfection_Decay;
|
|
void() BioInfection_MonsterDecay;
|
|
|
|
// Attacking functions
|
|
void() W_FireFlame;
|
|
void() W_FireIncendiaryCannon;
|
|
void() W_FireTranq;
|
|
void() W_FireLaser;
|
|
|
|
// Timer Functions
|
|
void() HallucinationTimer;
|
|
void() TranquiliserTimer;
|
|
|
|
// CTF Support functions
|
|
void() TeamFortress_CTF_FlagInfo;
|
|
|
|
// PC_UNDEFINED viewing functions
|
|
void() TF_MovePlayer;
|
|
|
|
//WK CustomTF Functions
|
|
void() custom_demon_precache;
|
|
void() custom_lay;
|
|
//void() CameraSwitchView;
|
|
|
|
//CH dropitems
|
|
void() TeamFortress_DropItems;
|
|
|
|
//- OfN -
|
|
void(entity player) ActivateHolo;
|
|
void(vector org, vector dir) grunty_spike;
|
|
void () UpdateInfos;
|
|
void() launch_horn;
|
|
void() player_laser1;
|
|
void(entity field) PutFieldWork;
|
|
void(entity tfield, vector where, entity thing) FieldExplosion;
|
|
|
|
// called by worldspawn
|
|
void() W_Precache =
|
|
{
|
|
precache_sound ("weapons/r_exp3.wav"); // new rocket explosion
|
|
precache_sound ("weapons/rocket1i.wav"); // spike gun
|
|
precache_sound ("weapons/sgun1.wav");
|
|
precache_sound ("weapons/guncock.wav"); // player shotgun
|
|
precache_sound ("weapons/ric1.wav"); // ricochet (used in c code)
|
|
precache_sound ("weapons/ric2.wav"); // ricochet (used in c code)
|
|
precache_sound ("weapons/ric3.wav"); // ricochet (used in c code)
|
|
precache_sound ("weapons/spike2.wav"); // super spikes
|
|
precache_sound ("weapons/tink1.wav"); // spikes tink (used in c code)
|
|
precache_sound ("weapons/grenade.wav"); // grenade launcher
|
|
precache_sound ("weapons/bounce.wav"); // grenade bounce
|
|
precache_sound ("weapons/shotgn2.wav"); // super shotgun
|
|
precache_sound ("wizard/wattack.wav"); // sniper rifle
|
|
precache_sound ("items/r_item1.wav"); // Medikit
|
|
precache_sound ("items/r_item2.wav"); // Medikit
|
|
precache_model ("progs/flame2.mdl"); // Flamethrower
|
|
precache_sound ("ambience/fire1.wav");
|
|
precache_sound2("blob/land1.wav"); // Hook
|
|
precache_model2("progs/v_spike.mdl"); // Hook
|
|
precache_sound ("hknight/hit.wav"); // Hook
|
|
// NEW FILES
|
|
precache_sound ("weapons/turrset.wav"); // Sentry Gun Setup
|
|
precache_sound ("weapons/turrspot.wav"); // Sentry Gun Spot
|
|
precache_sound ("weapons/turridle.wav"); // Sentry Gun Idle
|
|
precache_sound ("weapons/sniper.wav"); // sniper rifle
|
|
precache_sound ("weapons/flmfire2.wav"); // flamethrower
|
|
precache_sound ("weapons/flmgrexp.wav"); // flamethrower
|
|
precache_sound ("misc/vapeur2.wav"); // flamethrower
|
|
precache_sound ("weapons/asscan1.wav"); // Assault Cannon Powerup
|
|
precache_sound ("weapons/asscan2.wav"); // Assault Cannon Churning
|
|
precache_sound ("weapons/asscan3.wav"); // Assault Cannon Powerdown
|
|
precache_sound ("weapons/railgun.wav"); // Railgun
|
|
precache_sound ("weapons/dartgun.wav"); // Spy's dart gun
|
|
precache_sound ("ambience/thunder1.wav"); // Connect sound
|
|
|
|
//WK
|
|
// precache_model ("progs/v_merc.mdl"); //Carbine Model
|
|
// precache_sound ("weapons/carbfire.wav"); // Carbine shooting
|
|
// precache_sound ("weapons/carbreld.wav"); // Carbine shooting
|
|
// precache_sound ("weapons/carbrock.wav"); // Carbine shooting
|
|
// precache_sound ("weapons/nishi.wav"); // Nishi shooting
|
|
precache_model ("progs/telepad.mdl"); //Teleporter pad
|
|
precache_model ("progs/camera.mdl"); // Security Camera
|
|
precache_sound ("ambience/orff.wav"); // Chaplan sound
|
|
precache_sound ("weapons/guerilla_blip.wav"); // Landmine beep
|
|
precache_sound ("weapons/guerilla_set.wav"); // Landmine set
|
|
precache_sound ("weapons/camera_beep.wav"); // CH camera beep
|
|
precache_sound ("misc/enemy.wav"); // sensor alert
|
|
precache_sound ("weapons/fith.wav"); // WK Fire in the hole
|
|
// precache_model2("progs/guerilla_mine.mdl"); // The Landmine
|
|
// precache_model ("progs/tesla.mdl"); // The Tesla (by --Warrior--)
|
|
precache_model ("progs/minimissile.mdl"); // Swarm missile (BLOG)
|
|
// precache_model ("progs/newtesla.mdl"); // The Tesla (by --Warrior-- & BLOG)
|
|
precache_model ("progs/coil.mdl"); // The Tesla (by --Warrior-- & BLOG)
|
|
// precache_model ("progs/tscloak.mdl");
|
|
precache_model ("progs/tesgib1.mdl"); // Tesla Gib
|
|
precache_model ("progs/tesgib2.mdl"); // Tesla Gib
|
|
precache_model ("progs/tesgib3.mdl"); // Tesla Gib
|
|
precache_sound ("doors/medtry.wav"); //CH Dropping detpack sound
|
|
precache_sound ("doors/baseuse.wav"); //CH Detpack warning sound
|
|
precache_sound ("enforcer/sight1.wav"); // WK Taunt
|
|
precache_sound ("enforcer/sight2.wav"); // WK Taunt
|
|
precache_sound ("enforcer/sight3.wav"); // WK Taunt
|
|
precache_sound ("enforcer/sight4.wav"); // WK Taunt
|
|
|
|
custom_demon_precache();
|
|
};
|
|
|
|
float() crandom =
|
|
{
|
|
return 2*(random() - 0.5);
|
|
};
|
|
|
|
//======================================================================
|
|
// Calculate the attack_finished time
|
|
void(float att_delay) Attack_Finished =
|
|
{
|
|
if (self.tfstate & TFSTATE_TRANQUILISED)
|
|
self.attack_finished = time + (att_delay * 1.5);
|
|
else if (self.aura == AURA_HASTE)
|
|
self.attack_finished = time + (att_delay * 0.5);
|
|
else
|
|
self.attack_finished = time + att_delay;
|
|
};
|
|
|
|
void (entity atk, boolean do_cc, float dist) find_melee =
|
|
{
|
|
local vector dir;
|
|
|
|
if (self.classname == "player") {
|
|
makevectors (atk.v_angle);
|
|
dir = v_forward;
|
|
} else {
|
|
dir = normalize (atk.enemy.origin - atk.origin);
|
|
}
|
|
|
|
local vector source = atk.origin + '0 0 16';
|
|
|
|
traceline (source, source + dir*dist, FALSE, atk);
|
|
|
|
if (trace_fraction != 1.0)
|
|
return;
|
|
|
|
if (do_cc && (atk.cutf_items & CUTF_CLOSECOMBAT)) {
|
|
local entity te = findradius (source, dist);
|
|
local float tdist = -0.6;
|
|
|
|
do {
|
|
local vector diff = source - (te.origin + '0 0 16');
|
|
diff_z *= 0.75; //hack
|
|
|
|
local float l;
|
|
|
|
l = (normalize (diff) * dir);
|
|
|
|
if (l < tdist && te.takedamage == DAMAGE_AIM) {
|
|
tdist = l;
|
|
trace_ent = te;
|
|
trace_fraction = 0.5;
|
|
}
|
|
} while ((te = te.chain));
|
|
}
|
|
};
|
|
|
|
/*
|
|
================
|
|
W_FireAxe
|
|
================
|
|
*/
|
|
void() W_FireAxe =
|
|
{
|
|
local vector org;
|
|
|
|
find_melee (self, TRUE, 64);
|
|
|
|
if (trace_fraction == 1.0)
|
|
return;
|
|
|
|
org = trace_endpos - v_forward*4;
|
|
|
|
if (trace_ent.classname == "force_field") {
|
|
FieldExplosion(trace_ent,trace_endpos,trace_ent);
|
|
PutFieldWork(trace_ent);
|
|
|
|
return;
|
|
}
|
|
|
|
if (trace_ent.takedamage) {
|
|
trace_ent.axhitme = 1;
|
|
|
|
SpawnBlood (org, 20);
|
|
|
|
deathmsg = DMSG_AXE;
|
|
|
|
if (!(self.cutf_items & CUTF_KNIFE)) {
|
|
TF_T_Damage (trace_ent, self, self, 30, TF_TD_NOTTEAM, TF_TD_MELEE);
|
|
} else {
|
|
//WK Only give blood if you hit an enemy when being a warlock
|
|
if ((!Teammate(trace_ent.team_no, self.team_no) || !(self.job & JOB_WARLOCK)) && prematch < time) {
|
|
self.job = self.job | JOB_BLOODY_KNIFE;
|
|
self.weaponmode = 1; // Put blood on the knife
|
|
self.weaponmodel = "progs/v_knife2.mdl";
|
|
} else if (self.job & JOB_WARLOCK) {
|
|
sprint(self,PRINT_HIGH,"You may only draw blood from enemies\n");
|
|
}
|
|
|
|
TF_T_Damage (trace_ent, self, self, 50, TF_TD_NOTTEAM, TF_TD_MELEE);
|
|
}
|
|
} else { // hit wall
|
|
sound (self, CHAN_WEAPON, "player/axhit2.wav", 1, ATTN_NORM);
|
|
WriteByte (MSG_MULTICAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_MULTICAST, TE_GUNSHOT);
|
|
WriteByte (MSG_MULTICAST, 3);
|
|
WriteCoord (MSG_MULTICAST, org_x);
|
|
WriteCoord (MSG_MULTICAST, org_y);
|
|
WriteCoord (MSG_MULTICAST, org_z);
|
|
multicast (org, MULTICAST_PVS);
|
|
}
|
|
};
|
|
|
|
boolean (boolean strict) is_backstab =
|
|
{
|
|
local float d;
|
|
local vector dir;
|
|
|
|
makevectors (trace_ent.angles);
|
|
|
|
dir = trace_ent.origin - self.origin;
|
|
|
|
dir_z *= 0.75; // hack
|
|
|
|
d = normalize (dir) * v_forward;
|
|
|
|
if (d < 0.3)
|
|
return FALSE;
|
|
|
|
if (!strict)
|
|
return TRUE;
|
|
|
|
if (d < 0.7)
|
|
return FALSE;
|
|
|
|
d = normalize (trace_ent.origin - trace_endpos) * v_forward;
|
|
|
|
if (d < 0.8)
|
|
return FALSE;
|
|
|
|
return TRUE;
|
|
};
|
|
|
|
/*
|
|
================
|
|
W_FireBackstab
|
|
================
|
|
*/
|
|
|
|
void() W_FireBackstab =
|
|
{
|
|
local vector org;
|
|
|
|
if (!(self.cutf_items & CUTF_KNIFE)) {
|
|
W_FireAxe ();
|
|
return;
|
|
}
|
|
|
|
find_melee (self, FALSE, 64);
|
|
|
|
if (trace_fraction == 1.0)
|
|
return;
|
|
|
|
org = trace_endpos - v_forward*4;
|
|
|
|
if (trace_ent.classname == "force_field") {
|
|
FieldExplosion(trace_ent,trace_endpos,trace_ent);
|
|
PutFieldWork(trace_ent);
|
|
|
|
return;
|
|
}
|
|
|
|
local boolean living = FALSE;
|
|
|
|
if (trace_ent.classname == "player" || IsMonster (trace_ent))
|
|
living = TRUE;
|
|
|
|
if (trace_ent.takedamage) {
|
|
trace_ent.axhitme = 1;
|
|
|
|
deathmsg = DMSG_BACKSTAB;
|
|
|
|
if (living) {
|
|
if (trace_ent.classname != "player" && trace_ent.real_owner)
|
|
trace_ent.team_no = trace_ent.real_owner.team_no;
|
|
|
|
//WK Only give blood if you hit an enemy when being a warlock
|
|
if ((!Teammate(trace_ent.team_no, self.team_no) || !(self.job & JOB_WARLOCK)) && prematch < time) {
|
|
self.job = self.job | JOB_BLOODY_KNIFE;
|
|
self.weaponmode = 1;
|
|
self.weaponmodel = "progs/v_knife2.mdl";
|
|
} else if (self.job & JOB_WARLOCK) {
|
|
sprint(self,PRINT_HIGH,"You may only draw blood from enemies\n");
|
|
}
|
|
}
|
|
|
|
if (is_backstab (TRUE) && living) {
|
|
|
|
SpawnBlood (org, 50);
|
|
|
|
local entity oself = self;
|
|
self = trace_ent;
|
|
|
|
ThrowGib ("progs/gib1.mdl", -50);
|
|
ThrowGib ("progs/gib2.mdl", 10);
|
|
ThrowGib ("progs/gib3.mdl", 50);
|
|
ThrowGib ("progs/gib2.mdl", 25);
|
|
|
|
self = oself;
|
|
|
|
if (trace_ent.classname == "player")
|
|
stuffcmd (trace_ent, "bf; bf\n");
|
|
|
|
if (!(trace_ent.armorclass & AT_SAVEMELEE))
|
|
TF_T_Damage (trace_ent, self, self, 100, TF_TD_NOTTEAM | TF_TD_IGNOREARMOUR,
|
|
TF_TD_MELEE);
|
|
else
|
|
TF_T_Damage (trace_ent, self, self, 100, TF_TD_NOTTEAM, TF_TD_MELEE);
|
|
} else {
|
|
deathmsg = DMSG_AXE;
|
|
|
|
SpawnBlood (org, 10);
|
|
|
|
TF_T_Damage (trace_ent, self, self, 30, TF_TD_NOTTEAM, TF_TD_MELEE);
|
|
}
|
|
|
|
} else { // hit wall
|
|
sound (self, CHAN_WEAPON, "player/axhit2.wav", 1, ATTN_NORM);
|
|
WriteByte (MSG_MULTICAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_MULTICAST, TE_GUNSHOT);
|
|
WriteByte (MSG_MULTICAST, 3);
|
|
WriteCoord (MSG_MULTICAST, org_x);
|
|
WriteCoord (MSG_MULTICAST, org_y);
|
|
WriteCoord (MSG_MULTICAST, org_z);
|
|
multicast (org, MULTICAST_PVS);
|
|
}
|
|
}
|
|
/*
|
|
================
|
|
W_FireSpanner
|
|
================
|
|
*/
|
|
void() W_FireSpanner =
|
|
{
|
|
local vector org;
|
|
local float healam;
|
|
local entity te;
|
|
|
|
find_melee (self, TRUE, 64);
|
|
|
|
if (trace_fraction == 1.0)
|
|
return;
|
|
|
|
org = trace_endpos - v_forward*4;
|
|
|
|
if (trace_ent.classname == "force_field") {
|
|
FieldExplosion(trace_ent,trace_endpos,trace_ent);
|
|
PutFieldWork(trace_ent);
|
|
|
|
deathmsg = DMSG_SPANNERFIELD;
|
|
TF_T_Damage(self, self, self, self.health + 50, TF_TD_IGNOREARMOUR, TF_TD_OTHER);
|
|
sound (self, CHAN_MISC, "effects/crunch.wav", 1, ATTN_NONE);
|
|
|
|
return;
|
|
}
|
|
|
|
// It may be a trigger that can be activated by the engineer's spanner
|
|
if (trace_ent.goal_activation & TFGA_SPANNER) {
|
|
// Does the AP match the AP Criteria?
|
|
if (Activated(trace_ent,self)) {
|
|
DoResults(trace_ent, self, TRUE);
|
|
|
|
if (trace_ent.classname == "func_button") {
|
|
trace_ent.enemy = self;
|
|
other = self;
|
|
self = trace_ent;
|
|
self.dont_do_triggerwork = TRUE; // Already done in DoResults
|
|
button_fire();
|
|
self = other;
|
|
}
|
|
} else {
|
|
// If an else goal should be activated, activate it
|
|
if (trace_ent.else_goal != 0) {
|
|
te = Findgoal(trace_ent.else_goal);
|
|
if (te)
|
|
DoResults(te, self, (trace_ent.goal_result & TFGR_ADD_BONUSES));
|
|
} else {
|
|
sound (self, CHAN_WEAPON, "player/axhit2.wav", 1, ATTN_NORM);
|
|
WriteByte (MSG_MULTICAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_MULTICAST, TE_GUNSHOT);
|
|
WriteByte (MSG_MULTICAST, 3);
|
|
WriteCoord (MSG_MULTICAST, org_x);
|
|
WriteCoord (MSG_MULTICAST, org_y);
|
|
WriteCoord (MSG_MULTICAST, org_z);
|
|
multicast (org, MULTICAST_PVS);
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
if (trace_ent.takedamage) {
|
|
// auto-repair/dismantle if hit twice
|
|
if (trace_ent == self.building) {
|
|
Engineer_AutoUse();
|
|
}
|
|
|
|
if (trace_ent.classname == "building_dispenser") {
|
|
Engineer_UseDispenser(trace_ent);
|
|
return;
|
|
} else if (trace_ent.classname == "building_sentrygun") {
|
|
Engineer_UseSentryGun(trace_ent);
|
|
return;
|
|
} else if (trace_ent.classname == "building_sentrygun_base") {
|
|
if (trace_ent.oldenemy)
|
|
Engineer_UseSentryGun(trace_ent.oldenemy);
|
|
return;
|
|
} else if (trace_ent.classname == "building_tesla") {
|
|
Engineer_UseTesla(trace_ent);
|
|
return;
|
|
} else if (trace_ent.classname == "building_camera") {
|
|
Engineer_UseCamera(trace_ent);
|
|
return;
|
|
} else if (trace_ent.classname == "building_sensor") {
|
|
Engineer_UseSensor(trace_ent);
|
|
return;
|
|
} else if (trace_ent.classname == "building_teleporter") {
|
|
Engineer_UseTeleporter(trace_ent);
|
|
return;
|
|
} else if (trace_ent.classname == "building_fieldgen") {
|
|
Engineer_UseFieldGen(trace_ent);
|
|
return;
|
|
} else {
|
|
if (trace_ent.classname == "player") {
|
|
if (Teammate(trace_ent.team_no, self.team_no) && (teamplay) || (coop)) {
|
|
healam = WEAP_SPANNER_REPAIR;
|
|
if (self.ammo_cells < healam)
|
|
healam = self.ammo_cells;
|
|
|
|
// Only fix armor if they've got some
|
|
if (trace_ent.armor_allowed == 0) //WK
|
|
return;
|
|
|
|
if (trace_ent.armorvalue <= 0) // SB
|
|
return;
|
|
|
|
if (trace_ent.maxarmor - trace_ent.armorvalue < (healam * 4))
|
|
healam = ceil((trace_ent.maxarmor - trace_ent.armorvalue) / 4);
|
|
|
|
if (healam > 0) {
|
|
trace_ent.armorvalue = trace_ent.armorvalue + (healam * 4);
|
|
if (trace_ent.armorvalue > trace_ent.maxarmor)
|
|
trace_ent.armorvalue = trace_ent.maxarmor;
|
|
|
|
//WK Give them full armor color
|
|
trace_ent.armortype = trace_ent.armor_allowed;
|
|
|
|
self.ammo_cells = self.ammo_cells - healam;
|
|
|
|
sound(trace_ent, CHAN_WEAPON, "items/r_item1.wav", 1, ATTN_NORM);
|
|
WriteByte (MSG_MULTICAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_MULTICAST, TE_GUNSHOT);
|
|
WriteByte (MSG_MULTICAST, 3);
|
|
WriteCoord (MSG_MULTICAST, org_x);
|
|
WriteCoord (MSG_MULTICAST, org_y);
|
|
WriteCoord (MSG_MULTICAST, org_z);
|
|
multicast (org, MULTICAST_PVS);
|
|
|
|
W_SetCurrentAmmo ();
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
|
|
trace_ent.axhitme = 1;
|
|
SpawnBlood (org, 20);
|
|
|
|
deathmsg = DMSG_SPANNER;
|
|
//WK 20
|
|
TF_T_Damage (trace_ent, self, self, 20, TF_TD_NOTTEAM, TF_TD_MELEE);
|
|
}
|
|
} else { // hit wall
|
|
sound (self, CHAN_WEAPON, "player/axhit2.wav", 1, ATTN_NORM);
|
|
WriteByte (MSG_MULTICAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_MULTICAST, TE_GUNSHOT);
|
|
WriteByte (MSG_MULTICAST, 3);
|
|
WriteCoord (MSG_MULTICAST, org_x);
|
|
WriteCoord (MSG_MULTICAST, org_y);
|
|
WriteCoord (MSG_MULTICAST, org_z);
|
|
multicast (org, MULTICAST_PVS);
|
|
}
|
|
};
|
|
|
|
/*
|
|
================
|
|
W_FireMedikit
|
|
================
|
|
*/
|
|
|
|
void(entity ignore, entity ignore2, string st, string st2, string st3, string st4, string st5, string st6) teamsprintbi;
|
|
void(float tno, entity ignore, entity ignore2) teamprefixsprintbi;
|
|
|
|
void(float inAuto) W_FireMedikit =
|
|
{
|
|
local vector org;
|
|
local float healam;
|
|
local entity te, BioInfection;
|
|
|
|
find_melee (self, TRUE, 64);
|
|
|
|
if (trace_fraction == 1.0)
|
|
return;
|
|
|
|
org = trace_endpos - v_forward*4;
|
|
|
|
if (trace_ent.takedamage) {
|
|
if (trace_ent.classname == "player") {
|
|
if (Teammate(trace_ent.team_no, self.team_no) || (coop)) {
|
|
healam = WEAP_MEDIKIT_HEAL;
|
|
|
|
// remove concussion from player
|
|
// Try to find a concusstimer entity for this player
|
|
te = find(NIL, classname, "timer");
|
|
while (((te.owner != trace_ent) || (te.think != ConcussionGrenadeTimer)) && (te)) {
|
|
te = find(te, classname, "timer");
|
|
}
|
|
|
|
if (te) {
|
|
stuffcmd(trace_ent ,"v_idlescale 0\n");
|
|
trace_ent.mangle = '0 0 0';
|
|
SpawnBlood(org, 20);
|
|
sprint(trace_ent, PRINT_HIGH, "you have been healed of your concussion\n");
|
|
trace_ent.tfstate &= ~TFSTATE_CONCUSSIONED;
|
|
|
|
// Give the medic a frag for doing it, only if it was caused by an enemy
|
|
if (!Teammate(self.team_no,te.team_no)) {
|
|
self.real_frags = self.real_frags + 1;
|
|
if (!(toggleflags & TFLAG_TEAMFRAGS))
|
|
self.frags = self.real_frags;
|
|
}
|
|
|
|
dremove(te);
|
|
}
|
|
|
|
// remove hallucination from player
|
|
// Try to find a hallucination timer entity for this player
|
|
if (trace_ent.tfstate & TFSTATE_HALLUCINATING) {
|
|
te = find(NIL, classname, "timer");
|
|
while (((te.owner != trace_ent)
|
|
|| (te.think != HallucinationTimer)) && (te)) {
|
|
te = find(te, classname, "timer");
|
|
}
|
|
|
|
if (te) {
|
|
trace_ent.tfstate = trace_ent.tfstate - (trace_ent.tfstate & TFSTATE_HALLUCINATING);
|
|
|
|
SpawnBlood(org, 20);
|
|
sprint(trace_ent, PRINT_HIGH, "you have been healed of your hallucinations\n");
|
|
|
|
// Give the medic a frag for doing it, only if it was caused by an enemy
|
|
if (!Teammate(self.team_no,te.team_no)) {
|
|
self.real_frags = self.real_frags + 1;
|
|
if (!(toggleflags & TFLAG_TEAMFRAGS))
|
|
self.frags = self.real_frags;
|
|
}
|
|
|
|
dremove(te);
|
|
} else {
|
|
RPrint("Warning: Error in Hallucination Timer logic.\n");
|
|
}
|
|
}
|
|
|
|
// remove tranquilisation from player
|
|
// Try to find a tranquilisation timer entity for this player
|
|
if (trace_ent.tfstate & TFSTATE_TRANQUILISED) {
|
|
te = find(NIL, classname, "timer");
|
|
while (((te.owner != trace_ent) || (te.think != TranquiliserTimer)) && (te)) {
|
|
te = find(te, classname, "timer");
|
|
}
|
|
|
|
if (te) {
|
|
trace_ent.tfstate = trace_ent.tfstate - (trace_ent.tfstate & TFSTATE_TRANQUILISED);
|
|
TeamFortress_SetSpeed(trace_ent);
|
|
|
|
SpawnBlood(org, 20);
|
|
sprint(trace_ent, PRINT_HIGH, "you have been healed of your tranquilisation\n");
|
|
|
|
// Give the medic a frag for doing it, only if it was caused by an enemy
|
|
if (!Teammate(self.team_no,te.team_no)) {
|
|
self.real_frags = self.real_frags + 1;
|
|
if (!(toggleflags & TFLAG_TEAMFRAGS))
|
|
self.frags = self.real_frags;
|
|
}
|
|
|
|
dremove(te);
|
|
} else {
|
|
RPrint("Warning: Error in Tranquilisation Timer logic.\n");
|
|
}
|
|
}
|
|
|
|
// check if the healed player is blinded
|
|
if (trace_ent.FlashTime > 0) {
|
|
te = find(NIL, netname, "flashtimer");
|
|
while ((te.owner != trace_ent || te.classname != "timer") && (te))
|
|
te = find(te, netname, "flashtimer");
|
|
|
|
if (te) {
|
|
trace_ent.FlashTime = 0;
|
|
SpawnBlood(org, 20);
|
|
|
|
// Give the medic a frag for doing it, only if it was caused by an enemy
|
|
stuffcmd(trace_ent, "v_cshift 0\n"); //WK -- /CH te -> trace_ent
|
|
stuffcmd(trace_ent, "r_norefresh 0;wait;echo;wait;echo;wait;echo;wait;echo\n"); //WK -- /CH te -> trace_ent
|
|
|
|
if (!Teammate(self.team_no,te.team_no)) {
|
|
self.real_frags = self.real_frags + 1;
|
|
if (!(toggleflags & TFLAG_TEAMFRAGS))
|
|
self.frags = self.real_frags;
|
|
}
|
|
|
|
dremove(te);
|
|
} else {
|
|
RPrint("Warning: Error in Flash Timer logic.\n");
|
|
trace_ent.FlashTime = 0;
|
|
}
|
|
}
|
|
|
|
// check if the healed player is infected
|
|
if (trace_ent.tfstate & TFSTATE_INFECTED) {
|
|
healam = rint(trace_ent.health / 2);
|
|
|
|
// remove the infection
|
|
trace_ent.tfstate = trace_ent.tfstate - (trace_ent.tfstate & TFSTATE_INFECTED);
|
|
|
|
// some damage is caused (because of the use of leeches!)
|
|
// remove half their remaining health
|
|
deathmsg = DMSG_MEDIKIT;
|
|
T_Damage(trace_ent, self, self, healam);
|
|
|
|
SpawnBlood(org, 30);
|
|
sprint(trace_ent, PRINT_HIGH,self.netname);
|
|
sprint(trace_ent, PRINT_HIGH, " cures your infection!\n");
|
|
|
|
if (self.classname == "player") {
|
|
sprint(self, PRINT_HIGH, "You have healed ");
|
|
sprint(self, PRINT_HIGH, trace_ent.netname);
|
|
sprint(self, PRINT_HIGH, " of the infection.\n");
|
|
|
|
teamprefixsprintbi(self.team_no, self, trace_ent);
|
|
teamsprintbi(self, trace_ent, trace_ent.netname,
|
|
" infection has been cured by ",
|
|
self.netname, "\n", "", "");
|
|
|
|
// Give the medic a frag for doing it, only if it was caused by an enemy
|
|
if (!Teammate(trace_ent.infection_team_no, self.team_no)) {
|
|
self.real_frags = self.real_frags + 1;
|
|
if (!(toggleflags & TFLAG_TEAMFRAGS))
|
|
self.frags = self.real_frags;
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
// put out the fire if they are burning
|
|
if (trace_ent.numflames > 0) {
|
|
sound(trace_ent, CHAN_WEAPON, "items/r_item1.wav", 1, ATTN_NORM);
|
|
trace_ent.numflames = 0;
|
|
|
|
sprint(trace_ent, PRINT_HIGH, "The flames have been doused!\n");
|
|
|
|
if (self.classname == "player") {
|
|
sprint(self, PRINT_MEDIUM, "You have put out ");
|
|
sprint(self, PRINT_MEDIUM, trace_ent.netname);
|
|
sprint(self, PRINT_MEDIUM, "'s fire.\n");
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
if (healam > 0 && trace_ent.health < trace_ent.max_health) {
|
|
sound(trace_ent, CHAN_WEAPON, "items/r_item1.wav", 1, ATTN_NORM);
|
|
trace_ent.axhitme = 1;
|
|
SpawnBlood (org, 20);
|
|
|
|
T_Heal(trace_ent, healam, 0);
|
|
} else if (trace_ent.health >= trace_ent.max_health && trace_ent.health < (trace_ent.max_health + WEAP_MEDIKIT_OVERHEAL)) {
|
|
healam = 10;
|
|
if (healam > (self.ammo_medikit * 10))
|
|
healam = (self.ammo_medikit * 10);
|
|
if (healam > 0) {
|
|
sound(trace_ent, CHAN_ITEM, "items/r_item2.wav", 1, ATTN_NORM);
|
|
T_Heal(trace_ent, healam, 1);
|
|
self.ammo_medikit = self.ammo_medikit - rint(healam / 10);
|
|
if (!(trace_ent.items & IT_SUPERHEALTH)) {
|
|
trace_ent.items = trace_ent.items | IT_SUPERHEALTH;
|
|
newmis = spawn();
|
|
newmis.classname = "medikit_rot";
|
|
newmis.nextthink = time + 20;
|
|
newmis.think = item_megahealth_rot;
|
|
newmis.owner = trace_ent;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
#ifdef MEDIKIT_IS_BIOWEAPON
|
|
//WK Don't infect if they're invincible or observing
|
|
else if (!(trace_ent.invincible_finished) && !(trace_ent.playerclass == PC_UNDEFINED)) {
|
|
if (inAuto)
|
|
return;
|
|
trace_ent.axhitme = 1;
|
|
SpawnBlood (org, 20);
|
|
|
|
deathmsg = DMSG_BIOWEAPON_ATT;
|
|
if (!(self.cutf_items & CUTF_CLOSECOMBAT))
|
|
T_Damage (trace_ent, self, self, 10);
|
|
else
|
|
T_Damage (trace_ent, self, self, 20);
|
|
|
|
if (trace_ent.weapons_carried & WEAP_MEDIKIT) //WK
|
|
return;
|
|
|
|
//Melee armor stops infection 75% of the time
|
|
if (trace_ent.tf_items & NIT_GEL && random() < 0.75)
|
|
return;
|
|
|
|
trace_ent.tfstate = trace_ent.tfstate | TFSTATE_INFECTED;
|
|
|
|
BioInfection = spawn ();
|
|
BioInfection.classname = "timer";
|
|
BioInfection.netname = "biotimer";
|
|
BioInfection.owner = trace_ent;
|
|
BioInfection.nextthink = time + 2;
|
|
BioInfection.think = BioInfection_Decay;
|
|
BioInfection.enemy = self;
|
|
|
|
trace_ent.infection_team_no = self.team_no;
|
|
}
|
|
#endif
|
|
|
|
} else if (IsMonster(trace_ent)) { //- OfN - //|| trace_ent.classname == "monster_fish")
|
|
if (Teammate(trace_ent.real_owner.team_no, self.team_no)) {
|
|
if (trace_ent.health < trace_ent.max_health) {
|
|
if (trace_ent.max_health - trace_ent.health < 50)
|
|
healam = trace_ent.max_health - trace_ent.health;
|
|
else
|
|
healam = 50;
|
|
if (healam > self.ammo_medikit)
|
|
healam = self.ammo_medikit;
|
|
|
|
if (healam == 0)
|
|
return;
|
|
sound(trace_ent, CHAN_WEAPON, "items/r_item1.wav", 1, ATTN_NORM);
|
|
trace_ent.axhitme = 1;
|
|
SpawnBlood (org, 20);
|
|
|
|
T_Heal(trace_ent, healam, 0);
|
|
self.ammo_medikit = self.ammo_medikit - healam;
|
|
if (trace_ent.health > trace_ent.max_health)
|
|
trace_ent.health = trace_ent.max_health;
|
|
}
|
|
}
|
|
} else {
|
|
trace_ent.axhitme = 1;
|
|
SpawnBlood (org, 30);
|
|
T_Damage (trace_ent, self, self, 10);
|
|
}
|
|
} else {
|
|
if (inAuto) return; //Don't click for automedic
|
|
// hit wall
|
|
sound (self, CHAN_WEAPON, "player/axhit2.wav", 1, ATTN_NORM);
|
|
WriteByte (MSG_MULTICAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_MULTICAST, TE_GUNSHOT);
|
|
WriteByte (MSG_MULTICAST, 3);
|
|
WriteCoord (MSG_MULTICAST, org_x);
|
|
WriteCoord (MSG_MULTICAST, org_y);
|
|
WriteCoord (MSG_MULTICAST, org_z);
|
|
multicast (org, MULTICAST_PVS);
|
|
}
|
|
};
|
|
|
|
/*
|
|
================
|
|
W_FireBioweapon
|
|
================
|
|
*/
|
|
void() W_FireBioweapon =
|
|
{
|
|
local vector source;
|
|
local vector org;
|
|
|
|
local entity BioInfection;
|
|
|
|
source = self.origin + '0 0 16';
|
|
traceline (source, source + v_forward*64, FALSE, self);
|
|
if (trace_fraction == 1.0)
|
|
return;
|
|
|
|
org = trace_endpos - v_forward*4;
|
|
|
|
if (trace_ent.takedamage) {
|
|
if (trace_ent.classname == "player") {
|
|
if ((!Teammate(trace_ent.team_no, self.team_no) && teamplay) || teamplay == 0) {
|
|
trace_ent.axhitme = 1;
|
|
SpawnBlood (org, 20);
|
|
|
|
deathmsg = DMSG_BIOWEAPON_ATT;
|
|
if (!(self.cutf_items & CUTF_CLOSECOMBAT))
|
|
T_Damage (trace_ent, self, self, 10);
|
|
else
|
|
T_Damage (trace_ent, self, self, 20);
|
|
|
|
if (trace_ent.weapons_carried & WEAP_MEDIKIT) //WK
|
|
return;
|
|
|
|
trace_ent.tfstate = trace_ent.tfstate | TFSTATE_INFECTED;
|
|
|
|
BioInfection = spawn ();
|
|
BioInfection.classname = "timer";
|
|
BioInfection.netname = "biotimer";
|
|
BioInfection.owner = trace_ent;
|
|
BioInfection.nextthink = time + 2;
|
|
BioInfection.think = BioInfection_Decay;
|
|
BioInfection.enemy = self;
|
|
|
|
trace_ent.infection_team_no = self.team_no;
|
|
}
|
|
} else if (trace_ent.flags & FL_MONSTER) {
|
|
if (trace_ent.classname == "monster_zombie") {
|
|
// zombie slayer!
|
|
T_Damage (trace_ent, self, self, 200);
|
|
}
|
|
|
|
trace_ent.axhitme = 1;
|
|
SpawnBlood(org, 20);
|
|
if (!(self.cutf_items & CUTF_CLOSECOMBAT))
|
|
T_Damage (trace_ent, self, self, 10);
|
|
else
|
|
T_Damage (trace_ent, self, self, 20);
|
|
|
|
BioInfection = spawn ();
|
|
BioInfection.classname = "timer";
|
|
BioInfection.classname = "biotimer";
|
|
BioInfection.nextthink = time + 2;
|
|
BioInfection.think = BioInfection_MonsterDecay;
|
|
BioInfection.owner = self;
|
|
BioInfection.enemy = trace_ent;
|
|
} else { // must be a switch
|
|
trace_ent.axhitme = 1;
|
|
SpawnBlood (org, 30);
|
|
|
|
T_Damage(trace_ent, self, self, 40);
|
|
}
|
|
} else { // hit wall
|
|
sound (self, CHAN_WEAPON, "player/axhit2.wav", 1, ATTN_NORM);
|
|
WriteByte (MSG_MULTICAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_MULTICAST, TE_GUNSHOT);
|
|
WriteByte (MSG_MULTICAST, 3);
|
|
WriteCoord (MSG_MULTICAST, org_x);
|
|
WriteCoord (MSG_MULTICAST, org_y);
|
|
WriteCoord (MSG_MULTICAST, org_z);
|
|
multicast (org, MULTICAST_PVS);
|
|
}
|
|
};
|
|
|
|
//============================================================================
|
|
|
|
vector() wall_velocity =
|
|
{
|
|
local vector vel;
|
|
|
|
vel = normalize (self.velocity);
|
|
vel = normalize(vel + v_up*(random()- 0.5) + v_right*(random()- 0.5));
|
|
vel = vel + 2*trace_plane_normal;
|
|
vel = vel * 200;
|
|
|
|
return vel;
|
|
};
|
|
|
|
/*
|
|
================
|
|
SpawnMeatSpray
|
|
================
|
|
*/
|
|
void(vector org, vector vel) SpawnMeatSpray =
|
|
{
|
|
local entity missile;
|
|
|
|
missile = spawn ();
|
|
missile.owner = self;
|
|
missile.movetype = MOVETYPE_BOUNCE;
|
|
missile.solid = SOLID_NOT;
|
|
|
|
makevectors (self.angles);
|
|
|
|
missile.velocity = vel;
|
|
missile.velocity_z = missile.velocity_z + 250 + 50*random();
|
|
|
|
missile.avelocity = '3000 1000 2000';
|
|
|
|
// set missile duration
|
|
missile.nextthink = time + 1;
|
|
missile.think = SUB_Remove;
|
|
|
|
setmodel (missile, "progs/zom_gib.mdl");
|
|
setsize (missile, '0 0 0', '0 0 0');
|
|
setorigin (missile, org);
|
|
};
|
|
|
|
/*
|
|
================
|
|
SpawnBlood
|
|
================
|
|
*/
|
|
void(vector org, float damage) SpawnBlood =
|
|
{
|
|
WriteByte (MSG_MULTICAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_MULTICAST, TE_BLOOD);
|
|
WriteByte (MSG_MULTICAST, 1);
|
|
WriteCoord (MSG_MULTICAST, org_x);
|
|
WriteCoord (MSG_MULTICAST, org_y);
|
|
WriteCoord (MSG_MULTICAST, org_z);
|
|
multicast (org, MULTICAST_PVS);
|
|
};
|
|
|
|
/*
|
|
================
|
|
spawn_touchblood
|
|
================
|
|
*/
|
|
void(float damage) spawn_touchblood =
|
|
{
|
|
local vector vel;
|
|
|
|
vel = wall_velocity () * 0.2;
|
|
SpawnBlood (self.origin + vel*0.01, damage);
|
|
};
|
|
|
|
/*
|
|
================
|
|
SpawnChunk
|
|
================
|
|
*/
|
|
void(vector org, vector vel) SpawnChunk =
|
|
{
|
|
// particle (org, vel*0.02, 0, 10);
|
|
};
|
|
|
|
/*
|
|
==============================================================================
|
|
|
|
MULTI-DAMAGE
|
|
|
|
Collects multiple small damages into a single damage
|
|
|
|
==============================================================================
|
|
*/
|
|
|
|
entity multi_ent;
|
|
float multi_damage;
|
|
|
|
vector blood_org;
|
|
float blood_count;
|
|
|
|
vector puff_org;
|
|
float puff_count;
|
|
|
|
void() ClearMultiDamage =
|
|
{
|
|
multi_ent = NIL;
|
|
multi_damage = 0;
|
|
blood_count = 0;
|
|
puff_count = 0;
|
|
};
|
|
|
|
void() ApplyMultiDamage =
|
|
{
|
|
if (!multi_ent)
|
|
return;
|
|
// don't set deathmsg here, since it'll be set by the weapon that fired
|
|
if (self.current_weapon & WEAP_LIGHT_ASSAULT)
|
|
TF_T_Damage (multi_ent, self, self, multi_damage, TF_TD_NOTTEAM, TF_TD_NAIL);
|
|
else
|
|
TF_T_Damage (multi_ent, self, self, multi_damage, TF_TD_NOTTEAM, TF_TD_SHOT);
|
|
};
|
|
|
|
void(entity hit, float damage) AddMultiDamage =
|
|
{
|
|
if (!hit)
|
|
return;
|
|
|
|
if (hit != multi_ent) {
|
|
ApplyMultiDamage ();
|
|
multi_damage = damage;
|
|
multi_ent = hit;
|
|
} else
|
|
multi_damage = multi_damage + damage;
|
|
};
|
|
|
|
void (integer big) Multi_Finish =
|
|
{
|
|
if (puff_count) {
|
|
WriteByte (MSG_MULTICAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_MULTICAST, TE_GUNSHOT);
|
|
WriteByte (MSG_MULTICAST, big ? puff_count * 2 : puff_count);
|
|
WriteCoord (MSG_MULTICAST, puff_org_x);
|
|
WriteCoord (MSG_MULTICAST, puff_org_y);
|
|
WriteCoord (MSG_MULTICAST, puff_org_z);
|
|
multicast (puff_org, MULTICAST_PVS);
|
|
}
|
|
|
|
if (blood_count) {
|
|
WriteByte (MSG_MULTICAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_MULTICAST, TE_BLOOD);
|
|
WriteByte (MSG_MULTICAST, big ? blood_count * 2 : puff_count);
|
|
WriteCoord (MSG_MULTICAST, blood_org_x);
|
|
WriteCoord (MSG_MULTICAST, blood_org_y);
|
|
WriteCoord (MSG_MULTICAST, blood_org_z);
|
|
multicast (puff_org, MULTICAST_PVS);
|
|
}
|
|
};
|
|
|
|
/*
|
|
==============================================================================
|
|
|
|
BULLETS
|
|
|
|
==============================================================================
|
|
*/
|
|
|
|
/*
|
|
================
|
|
TraceAttack
|
|
================
|
|
*/
|
|
void (float damage, vector dir) TraceAttack =
|
|
{
|
|
local vector vel, org;
|
|
|
|
vel = normalize (dir + v_up * crandom() + v_right * crandom());
|
|
vel = vel + 2 * trace_plane_normal;
|
|
vel = vel * 200;
|
|
|
|
org = trace_endpos - v_forward * 4;
|
|
|
|
//WK Sweep mines at the end of the attack
|
|
GuerillaMineSweep (trace_endpos);
|
|
|
|
if (damage && trace_ent.takedamage) {
|
|
blood_org = org;
|
|
blood_count++;
|
|
|
|
AddMultiDamage (trace_ent, damage);
|
|
} else {
|
|
puff_org = org;
|
|
puff_count++;
|
|
if (trace_ent.classname == "force_field") { //- OfN - Makes field explosion b4 removing it
|
|
FieldExplosion(trace_ent,trace_endpos,trace_ent);
|
|
PutFieldWork(trace_ent);
|
|
}
|
|
}
|
|
};
|
|
|
|
/*
|
|
================
|
|
FireBullets
|
|
|
|
Used by shotgun, super shotgun, assault cannon, and enemy soldier firing
|
|
Go to the trouble of combining multiple pellets into a single damage call.
|
|
================
|
|
*/
|
|
void(float shotcount, vector dir, vector spread) FireBullets =
|
|
{
|
|
local vector direction;
|
|
local vector src;
|
|
local string st = infokey (NIL, "numpuffs");
|
|
local float numpuffs = 0;
|
|
local integer puffdiv;
|
|
|
|
if (st)
|
|
numpuffs = stof (st) - 1;
|
|
|
|
if (numpuffs <= 0)
|
|
puffdiv = 0;
|
|
else
|
|
puffdiv = (shotcount / numpuffs) + 1;
|
|
|
|
makevectors (self.v_angle);
|
|
|
|
src = self.origin + v_forward * 10;
|
|
src_z = self.absmin_z + self.size_z * 0.7;
|
|
|
|
ClearMultiDamage ();
|
|
while (shotcount > 0) {
|
|
|
|
direction = dir + crandom() * spread_x * v_right + crandom() * spread_y * v_up;
|
|
|
|
traceline (src, src + direction * 4096, FALSE, self);
|
|
|
|
if (trace_fraction == 1.0)
|
|
TraceAttack (0, direction);
|
|
else
|
|
TraceAttack (6, direction); //WK 4
|
|
|
|
if (puffdiv) {
|
|
if (shotcount % puffdiv == 0)
|
|
Multi_Finish (TRUE);
|
|
}
|
|
|
|
shotcount--;
|
|
}
|
|
ApplyMultiDamage ();
|
|
Multi_Finish (TRUE);
|
|
};
|
|
|
|
/*
|
|
================
|
|
W_FireShotgun
|
|
================
|
|
*/
|
|
void() W_FireShotgun =
|
|
{
|
|
local vector dir;
|
|
|
|
sound (self, CHAN_WEAPON, "weapons/guncock.wav", 1, ATTN_NORM);
|
|
|
|
if (self.classname == "player")
|
|
KickPlayer(-2, self);
|
|
|
|
self.currentammo = self.ammo_shells = self.ammo_shells - 1;
|
|
if (self.classname == "player")
|
|
dir = aim (self, 100000);
|
|
else {
|
|
dir = self.enemy.origin - self.enemy.velocity * (0.08 / self.has_sensor);
|
|
dir = normalize (dir - self.origin);
|
|
}
|
|
|
|
deathmsg = DMSG_SHOTGUN;
|
|
FireBullets (6, dir, '0.03 0.03 0');
|
|
};
|
|
|
|
|
|
/*
|
|
================
|
|
W_FireSuperShotgun
|
|
================
|
|
*/
|
|
void() W_FireSuperShotgun =
|
|
{
|
|
local vector dir;
|
|
|
|
if (self.currentammo == 1) {
|
|
W_FireShotgun ();
|
|
return;
|
|
}
|
|
|
|
sound (self ,CHAN_WEAPON, "weapons/shotgn2.wav", 1, ATTN_NORM);
|
|
|
|
if (self.classname == "player")
|
|
KickPlayer(-4, self);
|
|
|
|
self.currentammo = self.ammo_shells = self.ammo_shells - 2;
|
|
if (self.classname == "player")
|
|
dir = aim (self, 100000);
|
|
else {
|
|
dir = self.enemy.origin - self.enemy.velocity * (0.08 / self.has_sensor);
|
|
dir = normalize (dir - self.origin);
|
|
}
|
|
deathmsg = DMSG_SSHOTGUN;
|
|
FireBullets (6, dir, '0.11 0.06 0'); //WK 14, 0.14, 0.08
|
|
FireBullets (6, dir, '0.11 0.06 0');
|
|
};
|
|
|
|
/*
|
|
================
|
|
FireSniperBullet
|
|
Used by sniper rifle firing (W_FireSniperRifle)
|
|
================
|
|
*/
|
|
void(vector direction, float damage) FireSniperBullet =
|
|
{
|
|
local vector src;
|
|
|
|
makevectors (self.v_angle);
|
|
|
|
src = self.origin + v_forward * 10;
|
|
src_z = self.absmin_z + self.size_z * 0.7;
|
|
|
|
ClearMultiDamage ();
|
|
|
|
traceline (src, src + direction * 4096, FALSE, self);
|
|
|
|
if (trace_fraction == 1.0)
|
|
TraceAttack (0, direction);
|
|
else
|
|
TraceAttack (damage, direction);
|
|
|
|
ApplyMultiDamage ();
|
|
Multi_Finish (FALSE);
|
|
};
|
|
|
|
/*
|
|
=================================
|
|
TeamFortress : W_FireSniperRifle
|
|
=================================
|
|
*/
|
|
void() W_FireSniperRifle =
|
|
{
|
|
local vector dir, src;
|
|
local float dam_mult, zdif, use_this, ign_mult = 0;
|
|
local vector angle;
|
|
|
|
sound (self ,CHAN_WEAPON, "weapons/sniper.wav", 1, ATTN_NORM);
|
|
KickPlayer(-2, self);
|
|
|
|
angle = self.v_angle;
|
|
|
|
if (self.heat < vlen(self.velocity) * 0.1)
|
|
self.heat = vlen(self.velocity) * 0.1;
|
|
|
|
if (!(self.flags & FL_ONGROUND) && self.heat < 100)
|
|
self.heat = 100;
|
|
|
|
if (self.heat > 0) {
|
|
angle_x += (random() * 0.08 * self.heat) - (0.04 * self.heat);
|
|
angle_y += (random() * 0.08 * self.heat) - (0.04 * self.heat);
|
|
}
|
|
|
|
makevectors(angle);
|
|
src = self.origin + v_forward * 10;
|
|
src_z = self.absmin_z + self.size_z * 0.7;
|
|
|
|
dir = v_forward; //XXX is this correct?
|
|
|
|
use_this = FALSE;
|
|
traceline (src, src + dir*8092, FALSE, self);
|
|
if (trace_fraction != 1.0) {
|
|
if (trace_ent) {
|
|
if (trace_ent.classname == "player") {
|
|
use_this = TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
KickPlayer(-4, self);
|
|
|
|
if (!use_this) {
|
|
// aim, 'cause no entity in sights
|
|
dir = aim (self, 10000); // this corrects the aiming slightly, for bad players
|
|
traceline (src, src + dir*3072, FALSE, self);
|
|
}
|
|
|
|
deathmsg = DMSG_SNIPERRIFLE;
|
|
|
|
dam_mult = 0;
|
|
|
|
local string st = infokey (NIL, "sniper_dmg");
|
|
if (st)
|
|
dam_mult = stof (st);
|
|
if (!dam_mult)
|
|
dam_mult = SNIPER_DMG;
|
|
|
|
#if 0 // otr out
|
|
if (self.cutf_items & CUTF_OTR)
|
|
{
|
|
ign_mult=dam_mult*OTR_IGNFACTOR;
|
|
dam_mult=dam_mult*OTR_DMGFACTOR;
|
|
}
|
|
#endif
|
|
|
|
self.ammo_shells -= 5;
|
|
W_SetCurrentAmmo();
|
|
|
|
|
|
if (trace_ent) {
|
|
if (trace_ent.classname == "player") {
|
|
local float x;
|
|
local vector f, g, h;
|
|
|
|
f = trace_endpos - src;
|
|
|
|
g_x = trace_endpos_x;
|
|
g_y = trace_endpos_y;
|
|
g_z = 0;
|
|
|
|
h_x = trace_ent.origin_x;
|
|
h_y = trace_ent.origin_y;
|
|
h_z = 0;
|
|
|
|
x = vlen(g - h);
|
|
|
|
f = (normalize(f) * x) + trace_endpos;
|
|
|
|
zdif = f_z - trace_ent.origin_z;
|
|
deathmsg = DMSG_SNIPERRIFLE;
|
|
|
|
trace_ent.head_shot_vector = '0 0 0';
|
|
if (zdif < 0) {
|
|
// leg shot doesn't hurt very much
|
|
dam_mult *= 0.40;
|
|
|
|
if (dam_mult > SNIPER_MAXLEG)
|
|
{
|
|
dam_mult = SNIPER_MAXLEG;
|
|
}
|
|
|
|
if (trace_ent.armorvalue < 0.8)
|
|
trace_ent.leg_damage++;
|
|
|
|
if (trace_ent.leg_damage > 5)
|
|
trace_ent.leg_damage = 5;
|
|
|
|
#if 0 // OTR out
|
|
//- OfN - OTR bullets do twice damage ----------//
|
|
if (self.cutf_items & CUTF_OTR) {
|
|
trace_ent.leg_damage++;
|
|
}
|
|
#endif
|
|
|
|
TeamFortress_SetSpeed(trace_ent);
|
|
deathmsg = DMSG_SNIPERLEGSHOT;
|
|
//WK Kevlar ignored for snipers again.
|
|
//GR red armor has leg protection
|
|
|
|
if (trace_ent.armortype >= 0.7 && trace_ent.armorvalue > 0) {
|
|
TF_T_Damage (trace_ent, self, self, dam_mult, 0, TF_TD_SHOT);
|
|
} else {
|
|
TF_T_Damage (trace_ent, self, self, dam_mult, TF_TD_IGNOREARMOUR,0);
|
|
}
|
|
|
|
if (trace_ent.health > 0) {
|
|
#if 0 // OTR out
|
|
if (self.cutf_items & CUTF_OTR) {
|
|
sprint(trace_ent, PRINT_MEDIUM, "Leg injury! That was OTR stuff!\n");
|
|
sprint(self, PRINT_MEDIUM, "OTR Leg shot - that'll slow him down!\n");
|
|
} else
|
|
#endif
|
|
{ //- not OTR..
|
|
sprint(trace_ent, PRINT_MEDIUM, "Leg injury!\n");
|
|
sprint(self, PRINT_MEDIUM, "Leg shot - that'll slow him down!\n");
|
|
}
|
|
}
|
|
|
|
return;
|
|
} else if (zdif > 20) {
|
|
// head shot
|
|
dam_mult *= 2;
|
|
|
|
stuffcmd(trace_ent, "bf\n");
|
|
|
|
trace_ent.head_shot_vector = trace_ent.origin - self.origin;
|
|
deathmsg = DMSG_SNIPERHEADSHOT;
|
|
if (trace_ent.armortype > 0.5 && trace_ent.armorvalue > 0)
|
|
{
|
|
TF_T_Damage (trace_ent, self, self, dam_mult, 0, TF_TD_SHOT);
|
|
} else {
|
|
TF_T_Damage (trace_ent, self, self, dam_mult, TF_TD_IGNOREARMOUR, 0);
|
|
}
|
|
|
|
if (trace_ent.health > 0) {
|
|
#if 0 // otr out
|
|
if (self.cutf_items & CUTF_OTR) {
|
|
sprint(trace_ent, PRINT_MEDIUM, "Head injury! That was OTR stuff!\n");
|
|
sprint(self, PRINT_MEDIUM, "OTR Head shot - that's gotta hurt!\n");
|
|
} else
|
|
#endif
|
|
{
|
|
sprint(trace_ent, PRINT_MEDIUM, "Head injury!\n");
|
|
sprint(self, PRINT_MEDIUM, "Head shot - that's gotta hurt!\n");
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
ClearMultiDamage ();
|
|
|
|
if (trace_fraction == 1.0)
|
|
TraceAttack (0, dir);
|
|
else // if it hit something
|
|
TraceAttack (SNIPER_DMG * dam_mult, dir);
|
|
|
|
ApplyMultiDamage ();
|
|
Multi_Finish (FALSE);
|
|
};
|
|
|
|
#ifndef NO_AUTORIFLE
|
|
/*
|
|
===================================
|
|
TeamFortress : W_FireAutoRifle
|
|
===================================
|
|
*/
|
|
// same as sniper rifle
|
|
void() W_FireAutoRifle =
|
|
{
|
|
local vector dir;
|
|
|
|
sound (self ,CHAN_WEAPON, "weapons/sniper.wav", 1, ATTN_NORM);
|
|
|
|
KickPlayer(-1, self);
|
|
|
|
self.currentammo = self.ammo_shells = self.ammo_shells - 1;
|
|
|
|
makevectors(self.v_angle);
|
|
dir = v_forward;
|
|
deathmsg = DMSG_AUTORIFLE;
|
|
FireSniperBullet (dir, 8);
|
|
};
|
|
|
|
#endif
|
|
|
|
/*
|
|
================
|
|
TeamFortress : W_FireAssaultCannon
|
|
================
|
|
*/
|
|
void(float num) W_FireAssaultCannon =
|
|
{
|
|
local vector dir;
|
|
|
|
KickPlayer(-4, self);
|
|
|
|
self.currentammo = self.ammo_shells = self.ammo_shells - 1;
|
|
dir = aim (self, 100000);
|
|
deathmsg = DMSG_ASSAULTCANNON;
|
|
FireBullets (num, dir, '0.013 0.013 0' * num);
|
|
|
|
if (!(self.flags & FL_ONGROUND)) {
|
|
if (!self.waterlevel || (self.tf_items & NIT_SCUBA))
|
|
self.velocity -= 5 * num * dir;
|
|
else
|
|
self.velocity -= 10 * num * dir;
|
|
}
|
|
};
|
|
|
|
/*
|
|
=========================================
|
|
Custom TeamFortress : W_FireLightAssault
|
|
=========================================
|
|
*/
|
|
void() W_FireLightAssault =
|
|
{
|
|
local vector dir;
|
|
|
|
if (self.ammo_nails < 1) {
|
|
self.current_weapon = W_BestWeapon ();
|
|
W_SetCurrentAmmo ();
|
|
W_PrintWeaponMessage();
|
|
return;
|
|
}
|
|
|
|
sound (self, CHAN_WEAPON, "weapons/guncock.wav", 0.6, ATTN_NORM);
|
|
|
|
KickPlayer(-4, self);
|
|
|
|
makevectors(self.v_angle);
|
|
|
|
dir = v_forward;
|
|
deathmsg = DMSG_LIGHT_ASSAULT;
|
|
self.currentammo = self.ammo_nails = self.ammo_nails - 1;
|
|
self.reload_light_assault = self.reload_light_assault + 1;
|
|
|
|
if (CheckForReload() == TRUE)
|
|
return;
|
|
FireBullets (5, dir, '0.1 0.1 0');
|
|
Attack_Finished(0.2);
|
|
};
|
|
|
|
/*
|
|
==============================================================================
|
|
|
|
ROCKETS
|
|
|
|
==============================================================================
|
|
*/
|
|
|
|
void() s_explode1 = [0, s_explode2] {};
|
|
void() s_explode2 = [1, s_explode3] {};
|
|
void() s_explode3 = [2, s_explode4] {};
|
|
void() s_explode4 = [3, s_explode5] {};
|
|
void() s_explode5 = [4, s_explode6] {};
|
|
void() s_explode6 = [5, SUB_Remove] {};
|
|
|
|
void() BecomeExplosion =
|
|
{
|
|
dremove(self);
|
|
};
|
|
|
|
void() T_MissileTouch =
|
|
{
|
|
local float damg;
|
|
local float bonus;
|
|
|
|
if (other.classname == "player" && other.playerclass == PC_UNDEFINED)
|
|
return;
|
|
|
|
if (pointcontents(self.origin) == CONTENTS_SKY) {
|
|
dremove(self);
|
|
return;
|
|
}
|
|
|
|
// Lowered from 120
|
|
damg = 100;
|
|
|
|
if (self.owner.classname == "trap_shooter" || self.owner.classname == "trap_spikeshooter" || self.owner.classname == "trap_tf_spikeshooter" || self.owner.classname == "trap_tf_shooter")
|
|
if (self.owner.dmg != 0)
|
|
damg = self.owner.dmg;
|
|
|
|
if (self.has_tesla) //Cluster rockets do less
|
|
damg = damg - 25;
|
|
|
|
deathmsg = self.weapon;
|
|
if (other.health) {
|
|
bonus = 10 + random()*20;
|
|
if (self.has_tesla)
|
|
bonus = 0;
|
|
TF_T_Damage (other, self, self.owner, damg+bonus, 0, TF_TD_EXPLOSION);
|
|
}
|
|
|
|
// don't do radius damage to the other, because all the damage
|
|
// was done in the impact
|
|
// Lowered from 120
|
|
T_RadiusDamage (self, self.owner, damg, other);
|
|
|
|
self.origin = self.origin - 8*normalize(self.velocity);
|
|
|
|
#ifdef DEMO_STUFF
|
|
// Remove any camera's locks on this missile
|
|
if (self.enemy)
|
|
CamProjectileLockOff();
|
|
#endif
|
|
|
|
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);
|
|
|
|
if (other.classname == "force_field") //- OfN - Makes field explosion b4 removing it
|
|
FieldExplosion(other,self.origin,self);
|
|
|
|
dremove(self);
|
|
};
|
|
|
|
//CH rocket tracker (taken from shalrath.qc)
|
|
void() Rocket_Track_Dot =
|
|
{
|
|
if (time > self.has_sentry) {
|
|
self.think = SUB_Remove;
|
|
} else {
|
|
local vector dir, vtemp;
|
|
local entity tg, sight;
|
|
|
|
sight = NIL;
|
|
tg = find (NIL, classname, "timer");
|
|
while (tg) {
|
|
if (tg.owner == self.owner && tg.think == SniperSight_Update2)
|
|
sight = tg;
|
|
tg = find(tg, classname, "timer");
|
|
}
|
|
if (sight) { //Found a sight
|
|
vtemp = sight.origin;
|
|
dir = normalize(vtemp - self.origin);
|
|
// if (self.owner.cluster_mode && !(self.has_tesla)) //Slow down main rocket
|
|
// self.velocity = dir * 500;
|
|
// else
|
|
if (self.owner.cluster_mode)
|
|
self.velocity = dir * 600;
|
|
else
|
|
self.velocity = dir * 900;
|
|
self.angles = vectoangles(self.velocity);
|
|
self.think = Rocket_Track_Dot;
|
|
//WK Add in cluster munitions support
|
|
if (self.has_tesla) { //We're a cluster, so act crazy
|
|
makevectors(self.angles);
|
|
self.v_angle = sight.origin; //Save last direction for losing sight
|
|
self.velocity = self.velocity + 300 * v_right * (random() - 0.5);
|
|
self.velocity = self.velocity + 300 * v_up * (random() - 0.5);
|
|
}
|
|
} else { //Lost sight of sight
|
|
if (self.has_tesla && self.v_angle != '0 0 0') {
|
|
vtemp = self.v_angle;
|
|
dir = normalize(vtemp - self.origin);
|
|
if (self.owner.cluster_mode && !(self.has_tesla)) //Slow down main rocket
|
|
self.velocity = dir * 900;
|
|
else
|
|
self.velocity = dir * 1000;
|
|
self.angles = vectoangles(self.velocity);
|
|
self.think = Rocket_Track_Dot;
|
|
makevectors(self.angles);
|
|
self.velocity = self.velocity + 300 * v_right * (random() - 0.5);
|
|
self.velocity = self.velocity + 300 * v_up * (random() - 0.5);
|
|
}
|
|
// self.velocity = 1000 * normalize(self.v_angle);
|
|
}
|
|
}
|
|
|
|
self.nextthink = time + 0.25;
|
|
};
|
|
/*
|
|
================
|
|
W_FireRocket
|
|
================
|
|
*/
|
|
void() W_FireRocket =
|
|
{
|
|
local float loops;
|
|
local vector olorigin, dir = NIL; //XXX false +ve for dir
|
|
loops = 0;
|
|
if (self.tf_items & NIT_CLUSTER_ROCKETS && self.cluster_mode == TRUE)
|
|
loops = 5;
|
|
while (loops >= 0) {
|
|
self.currentammo = self.ammo_rockets = self.ammo_rockets - 1;
|
|
if (self.ammo_rockets < 0) {
|
|
self.currentammo = self.ammo_rockets = 0;
|
|
return;
|
|
}
|
|
if (loops == 0) {
|
|
sound (self, CHAN_WEAPON, "weapons/sgun1.wav", 1, ATTN_NORM);
|
|
if (self.classname == "player")
|
|
KickPlayer(-2, self);
|
|
}
|
|
newmis = spawn ();
|
|
newmis.owner = self;
|
|
newmis.movetype = MOVETYPE_FLYMISSILE;
|
|
newmis.solid = SOLID_BBOX;
|
|
|
|
newmis.classname = "rocket"; //- OfN - Checked on airfist
|
|
|
|
// set newmis speed
|
|
if (self.classname == "player") {
|
|
makevectors (self.v_angle);
|
|
dir = v_forward;
|
|
} else if (self.classname == "monster_army")
|
|
dir = Grunty_LeadShot();
|
|
|
|
newmis.velocity = dir;
|
|
if (!loops) { //WK Not a cluster Rocket
|
|
if (self.tf_items & NIT_CLUSTER_ROCKETS && self.cluster_mode == TRUE)
|
|
{ //Make lead rocket start slower too
|
|
if (self.tf_items & NIT_RL_LASER_SIGHT) //Cluster and sight to start
|
|
newmis.velocity = newmis.velocity * 400;
|
|
else
|
|
newmis.velocity = newmis.velocity * 600; //Cluster and no sight
|
|
} else
|
|
newmis.velocity = newmis.velocity * 900;
|
|
} else if (self.tf_items & NIT_RL_LASER_SIGHT) //Cluster and sight to start
|
|
newmis.velocity = newmis.velocity * 400;
|
|
else
|
|
newmis.velocity = newmis.velocity * 600; //Cluster and no sight
|
|
newmis.angles = vectoangles(newmis.velocity);
|
|
|
|
newmis.touch = T_MissileTouch;
|
|
|
|
// set newmis duration
|
|
if (self.tf_items & NIT_RL_LASER_SIGHT) {
|
|
newmis.nextthink = time + 0.1; //Because tracks a sight
|
|
if (loops)
|
|
newmis.nextthink = time + 0.5; //Delay one second to spread out
|
|
newmis.think = Rocket_Track_Dot;
|
|
newmis.has_sentry = time + 6;
|
|
} else {
|
|
newmis.nextthink = time + 4;
|
|
newmis.think = SUB_Remove;
|
|
}
|
|
|
|
newmis.weapon = DMSG_ROCKETL;
|
|
if (!loops)
|
|
setmodel (newmis, "progs/missile.mdl");
|
|
else
|
|
setmodel (newmis, "progs/minimissile.mdl"); //Diff model for swarm rockets
|
|
setsize (newmis, '0 0 0', '0 0 0');
|
|
setorigin (newmis, self.origin + dir*8 + '0 0 16');
|
|
if (loops) { //WK Vary the starting point of the cluster
|
|
traceline (self.origin, self.origin + v_forward*9192, FALSE, self); //Make this TRUE
|
|
newmis.v_angle = trace_endpos;
|
|
newmis.has_tesla = loops; //Tell the rocket it is a cluster
|
|
newmis.weapon = DMSG_CLUSTER_ROCKET;
|
|
olorigin = newmis.origin;
|
|
newmis.origin = newmis.origin + v_right * (random() * 120 - 60);
|
|
newmis.origin = newmis.origin + v_forward * (random() * 40);
|
|
newmis.origin = newmis.origin + v_up * (random() * 50 - 20);
|
|
olorigin = olorigin - newmis.origin; //Get vector to new point
|
|
olorigin = 200 * normalize(olorigin);
|
|
if (self.tf_items & NIT_RL_LASER_SIGHT) //Dont spread without sight
|
|
newmis.velocity = newmis.velocity + olorigin;
|
|
}
|
|
loops = loops - 1;
|
|
} //End while loops
|
|
#ifdef DEMO_STUFF
|
|
// Have we got a live camera in projectile mode?
|
|
if (live_camera)
|
|
CamProjectileLockOn();
|
|
#endif
|
|
};
|
|
|
|
/*
|
|
===============================================================================
|
|
|
|
LIGHTNING
|
|
|
|
===============================================================================
|
|
*/
|
|
|
|
void(entity from, float damage) LightningHit =
|
|
{
|
|
WriteByte (MSG_MULTICAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_MULTICAST, TE_LIGHTNINGBLOOD);
|
|
WriteCoord (MSG_MULTICAST, trace_endpos_x);
|
|
WriteCoord (MSG_MULTICAST, trace_endpos_y);
|
|
WriteCoord (MSG_MULTICAST, trace_endpos_z);
|
|
multicast (trace_endpos, MULTICAST_PVS);
|
|
|
|
TF_T_Damage (trace_ent, from, from, damage, TF_TD_NOTTEAM, TF_TD_ELECTRICITY);
|
|
};
|
|
|
|
/*
|
|
=================
|
|
LightningDamage
|
|
=================
|
|
*/
|
|
void(vector p1, vector p2, entity from, float damage) LightningDamage =
|
|
{
|
|
local entity e1, e2;
|
|
local vector f;
|
|
|
|
f = p2 - p1;
|
|
normalize (f);
|
|
f_x = 0 - f_y;
|
|
f_y = f_x;
|
|
f_z = 0;
|
|
f = f*16;
|
|
|
|
e1 = e2 = NIL;
|
|
|
|
traceline (p1, p2, FALSE, self);
|
|
|
|
//WK Sweep mines at point of impact
|
|
GuerillaMineSweep(trace_endpos);
|
|
|
|
deathmsg = DMSG_LIGHTNING;
|
|
if (trace_ent.takedamage)
|
|
LightningHit (from, damage);
|
|
e1 = trace_ent;
|
|
|
|
traceline (p1 + f, p2 + f, FALSE, self);
|
|
if (trace_ent != e1 && trace_ent.takedamage)
|
|
LightningHit (from, damage);
|
|
e2 = trace_ent;
|
|
|
|
traceline (p1 - f, p2 - f, FALSE, self);
|
|
if (trace_ent != e1 && trace_ent != e2 && trace_ent.takedamage)
|
|
LightningHit (from, damage);
|
|
};
|
|
|
|
|
|
void() W_FireLightning =
|
|
{
|
|
local vector org;
|
|
local float cells, damage;
|
|
|
|
if (self.ammo_cells < 1) {
|
|
self.current_weapon = W_BestWeapon ();
|
|
W_SetCurrentAmmo ();
|
|
W_PrintWeaponMessage();
|
|
return;
|
|
}
|
|
|
|
makevectors (self.v_angle);
|
|
|
|
// explode if under water
|
|
if (self.waterlevel > 1) {
|
|
cells = self.ammo_cells;
|
|
if (cells > 10)
|
|
cells = 10; //WK Don't allow detpacks on the fly!
|
|
self.ammo_cells = self.ammo_cells - cells;
|
|
W_SetCurrentAmmo ();
|
|
deathmsg = DMSG_LIGHTNING;
|
|
//WK T_RadiusDamage (self, self, 35*cells, NIL);
|
|
T_RadiusDamage (self, self, 20*cells, NIL);
|
|
Attack_Finished(5);
|
|
return;
|
|
}
|
|
|
|
if (self.t_width < time) {
|
|
sound (self, CHAN_WEAPON, "weapons/lhit.wav", 1, ATTN_NORM);
|
|
self.t_width = time + 0.6;
|
|
}
|
|
|
|
KickPlayer(-2, self);
|
|
|
|
if (self.ammo_cells > 300) {
|
|
damage = 25;
|
|
self.ammo_cells -= 15;
|
|
} else if (self.ammo_cells > 200) {
|
|
damage = 20;
|
|
self.ammo_cells -= 10;
|
|
} else if (self.ammo_cells > 100) {
|
|
damage = 17;
|
|
self.ammo_cells -= 4;
|
|
} else {
|
|
damage = 12;
|
|
self.ammo_cells -= 1.2;
|
|
}
|
|
|
|
self.currentammo = self.ammo_cells;
|
|
|
|
org = self.origin + '0 0 16';
|
|
|
|
// OfN - Check for force field
|
|
traceline (org, org + v_forward*600, FALSE, self);
|
|
|
|
if (trace_ent.classname == "force_field") {
|
|
FieldExplosion(trace_ent,trace_endpos,trace_ent);
|
|
PutFieldWork(trace_ent);
|
|
|
|
WriteByte (MSG_MULTICAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_MULTICAST, TE_LIGHTNING2);
|
|
WriteEntity (MSG_MULTICAST, self);
|
|
WriteCoord (MSG_MULTICAST, org_x);
|
|
WriteCoord (MSG_MULTICAST, org_y);
|
|
WriteCoord (MSG_MULTICAST, org_z);
|
|
WriteCoord (MSG_MULTICAST, trace_endpos_x);
|
|
WriteCoord (MSG_MULTICAST, trace_endpos_y);
|
|
WriteCoord (MSG_MULTICAST, trace_endpos_z);
|
|
multicast (org, MULTICAST_PHS);
|
|
|
|
return;
|
|
}
|
|
//_------------------------------------_//
|
|
|
|
traceline (org, org + v_forward*600, TRUE, self);
|
|
|
|
WriteByte (MSG_MULTICAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_MULTICAST, TE_LIGHTNING2);
|
|
WriteEntity (MSG_MULTICAST, self);
|
|
WriteCoord (MSG_MULTICAST, org_x);
|
|
WriteCoord (MSG_MULTICAST, org_y);
|
|
WriteCoord (MSG_MULTICAST, org_z);
|
|
WriteCoord (MSG_MULTICAST, trace_endpos_x);
|
|
WriteCoord (MSG_MULTICAST, trace_endpos_y);
|
|
WriteCoord (MSG_MULTICAST, trace_endpos_z);
|
|
multicast (org, MULTICAST_PHS);
|
|
|
|
LightningDamage (self.origin, trace_endpos + v_forward*4, self, damage);
|
|
};
|
|
|
|
//=============================================================================
|
|
|
|
float (float tno) num_team_pipebombs =
|
|
{
|
|
if (tno == 1)
|
|
return num_team_pipebombs_1;
|
|
if (tno == 2)
|
|
return num_team_pipebombs_2;
|
|
if (tno == 3)
|
|
return num_team_pipebombs_3;
|
|
if (tno == 4)
|
|
return num_team_pipebombs_4;
|
|
|
|
return 0;
|
|
};
|
|
|
|
void(float tno) ExplodeOldPipebomb =
|
|
{
|
|
local entity old;
|
|
local float index;
|
|
|
|
if (tno != 0) {
|
|
index = num_team_pipebombs(tno);
|
|
index = index - (MAX_WORLD_PIPEBOMBS / number_of_teams);
|
|
} else {
|
|
index = num_world_pipebombs - MAX_WORLD_PIPEBOMBS;
|
|
}
|
|
|
|
old = find(NIL, classname, "pipebomb");
|
|
while (index > 0) {
|
|
if (!old) {
|
|
RPrint("*** ERROR: ExplodeOldPipebomb. ***\n");
|
|
RPrint("*** Please report this. ***\n");
|
|
num_world_pipebombs = 0;
|
|
num_team_pipebombs_1 = 0;
|
|
num_team_pipebombs_2 = 0;
|
|
num_team_pipebombs_3 = 0;
|
|
num_team_pipebombs_4 = 0;
|
|
return;
|
|
}
|
|
|
|
if (old.owner.team_no == tno || tno == 0) {
|
|
old.nextthink = time + 0.5;
|
|
index = index - 1;
|
|
}
|
|
|
|
old = find(old, classname, "pipebomb");
|
|
}
|
|
};
|
|
|
|
void(float tno) increment_team_pipebombs =
|
|
{
|
|
if (tno == 1)
|
|
num_team_pipebombs_1 = num_team_pipebombs_1 + 1;
|
|
else if (tno == 2)
|
|
num_team_pipebombs_2 = num_team_pipebombs_2 + 1;
|
|
else if (tno == 3)
|
|
num_team_pipebombs_3 = num_team_pipebombs_3 + 1;
|
|
else if (tno == 4)
|
|
num_team_pipebombs_4 = num_team_pipebombs_4 + 1;
|
|
};
|
|
|
|
void(float tno) decrement_team_pipebombs =
|
|
{
|
|
if (tno == 1)
|
|
num_team_pipebombs_1 = num_team_pipebombs_1 - 1;
|
|
else if (tno == 2)
|
|
num_team_pipebombs_2 = num_team_pipebombs_2 - 1;
|
|
else if (tno == 3)
|
|
num_team_pipebombs_3 = num_team_pipebombs_3 - 1;
|
|
else if (tno == 4)
|
|
num_team_pipebombs_4 = num_team_pipebombs_4 - 1;
|
|
};
|
|
|
|
//=============================================================================
|
|
|
|
|
|
void() GrenadeExplode =
|
|
{
|
|
if (self.classname == "pipebomb") {
|
|
num_world_pipebombs = num_world_pipebombs - 1;
|
|
decrement_team_pipebombs(self.owner.team_no);
|
|
}
|
|
|
|
deathmsg = self.weapon;
|
|
T_RadiusDamage (self, self.owner, 120, NIL);
|
|
|
|
#ifdef DEMO_STUFF
|
|
// Remove any camera's locks on this missile
|
|
if (self.enemy)
|
|
CamProjectileLockOff();
|
|
#endif
|
|
|
|
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);
|
|
dremove(self);
|
|
};
|
|
|
|
void() GrenadeTouch =
|
|
{
|
|
if (other == self.owner)
|
|
return; // don't explode on owner
|
|
if (other.takedamage == DAMAGE_AIM) {
|
|
GrenadeExplode();
|
|
return;
|
|
}
|
|
sound (self, CHAN_WEAPON, "weapons/bounce.wav", 1, ATTN_NORM); // bounce sound
|
|
if (self.velocity == '0 0 0')
|
|
self.avelocity = '0 0 0';
|
|
};
|
|
|
|
/*
|
|
================
|
|
W_FireGrenade
|
|
================
|
|
*/
|
|
void() W_FireGrenade =
|
|
{
|
|
self.currentammo = self.ammo_rockets = self.ammo_rockets - 1;
|
|
sound (self, CHAN_WEAPON, "weapons/grenade.wav", 1, ATTN_NORM);
|
|
KickPlayer(-2, self);
|
|
newmis = spawn ();
|
|
newmis.owner = self;
|
|
newmis.movetype = MOVETYPE_BOUNCE;
|
|
newmis.solid = SOLID_BBOX;
|
|
|
|
// Set grenade type based on firing mode
|
|
if (self.weaponmode == GL_NORMAL) {
|
|
newmis.weapon = DMSG_GRENADEL;
|
|
newmis.classname = "grenade";
|
|
newmis.skin = 1;
|
|
newmis.touch = GrenadeTouch;
|
|
newmis.nextthink = time + 2.5;
|
|
} else { // if (self.weaponmode == GL_PIPEBOMB)
|
|
if (self.team_no != 0) {
|
|
increment_team_pipebombs(self.team_no);
|
|
if (num_team_pipebombs(self.team_no) > (MAX_WORLD_PIPEBOMBS / number_of_teams))
|
|
ExplodeOldPipebomb(self.team_no);
|
|
} else {
|
|
num_world_pipebombs = num_world_pipebombs + 1;
|
|
if (num_world_pipebombs > MAX_WORLD_PIPEBOMBS)
|
|
ExplodeOldPipebomb(0);
|
|
}
|
|
|
|
newmis.classname = "pipebomb";
|
|
newmis.skin = 2;
|
|
newmis.touch = PipebombTouch;
|
|
newmis.nextthink = time + 120; // Remove pipebombs older than 2 minutes
|
|
newmis.weapon = DMSG_GREN_PIPE;
|
|
}
|
|
|
|
// set newmis speed
|
|
makevectors (self.v_angle);
|
|
if (self.v_angle_x)
|
|
newmis.velocity = v_forward*600 + v_up * 200 + crandom()*v_right*10 + crandom()*v_up*10;
|
|
else {
|
|
newmis.velocity = aim(self, 10000);
|
|
newmis.velocity = newmis.velocity * 600;
|
|
newmis.velocity_z = 200;
|
|
}
|
|
newmis.avelocity = '300 300 300';
|
|
newmis.angles = vectoangles(newmis.velocity);
|
|
|
|
newmis.think = GrenadeExplode;
|
|
setmodel (newmis, "progs/grenade2.mdl");
|
|
setsize (newmis, '0 0 0', '0 0 0');
|
|
setorigin (newmis, self.origin);
|
|
|
|
#ifdef DEMO_STUFF
|
|
// Have we got a live camera in projectile mode?
|
|
if (live_camera)
|
|
CamProjectileLockOn();
|
|
#endif
|
|
};
|
|
|
|
//=============================================================================
|
|
|
|
void() spike_touch;
|
|
|
|
|
|
/*
|
|
===============
|
|
launch_spike
|
|
|
|
Used for both the player and the ogre
|
|
===============
|
|
*/
|
|
void(vector org, vector dir) launch_spike =
|
|
{
|
|
newmis = spawn ();
|
|
newmis.owner = self;
|
|
newmis.movetype = MOVETYPE_FLYMISSILE;
|
|
newmis.solid = SOLID_BBOX;
|
|
|
|
newmis.angles = vectoangles(dir);
|
|
|
|
newmis.touch = spike_touch;
|
|
newmis.weapon = DMSG_NAILGUN;
|
|
newmis.classname = "spike";
|
|
newmis.think = SUB_Remove;
|
|
newmis.nextthink = time + 6;
|
|
setmodel (newmis, "progs/spike.mdl");
|
|
setsize (newmis, VEC_ORIGIN, VEC_ORIGIN);
|
|
setorigin (newmis, org);
|
|
|
|
newmis.velocity = dir * 1600; // was 1000
|
|
|
|
#ifdef DEMO_STUFF
|
|
// Have we got a live camera in projectile mode?
|
|
if (live_camera)
|
|
CamProjectileLockOn();
|
|
#endif
|
|
};
|
|
|
|
void() superspike_touch =
|
|
{
|
|
local float ndmg;
|
|
|
|
if (other == self.owner)
|
|
return;
|
|
|
|
if (other.solid == SOLID_TRIGGER)
|
|
return; // trigger field, do nothing
|
|
|
|
if (pointcontents(self.origin) == CONTENTS_SKY)
|
|
{
|
|
dremove(self);
|
|
return;
|
|
}
|
|
|
|
// hit something that bleeds
|
|
if (other.takedamage) {
|
|
spawn_touchblood (18);
|
|
deathmsg = self.weapon;
|
|
|
|
/*
|
|
// In QW, nail grens only launch 1 nail, and it does more damage.
|
|
if (deathmsg == DMSG_GREN_NAIL)
|
|
ndmg = 40;
|
|
else
|
|
*/
|
|
|
|
ndmg = 13;
|
|
|
|
if (self.owner.classname == "grenade")
|
|
TF_T_Damage (other, self, self.owner.owner, ndmg, TF_TD_NOTTEAM, TF_TD_NAIL);
|
|
else
|
|
TF_T_Damage (other, self, self.owner, ndmg, TF_TD_NOTTEAM, TF_TD_NAIL);
|
|
} else {
|
|
if (other.classname == "force_field") //- OfN - Makes field explosion b4 removing it
|
|
FieldExplosion(other,self.origin,self);
|
|
else {
|
|
WriteByte (MSG_MULTICAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_MULTICAST, TE_SUPERSPIKE);
|
|
WriteCoord (MSG_MULTICAST, self.origin_x);
|
|
WriteCoord (MSG_MULTICAST, self.origin_y);
|
|
WriteCoord (MSG_MULTICAST, self.origin_z);
|
|
multicast (self.origin, MULTICAST_PHS);
|
|
}
|
|
}
|
|
|
|
dremove(self);
|
|
};
|
|
|
|
|
|
void() W_FireSuperSpikes =
|
|
{
|
|
local vector dir;
|
|
|
|
sound (self, CHAN_WEAPON, "weapons/spike2.wav", 1, ATTN_NORM);
|
|
Attack_Finished(0.2);
|
|
self.currentammo = self.ammo_nails = self.ammo_nails - 2;
|
|
dir = aim (self, 1000);
|
|
launch_spike (self.origin + '0 0 16', dir);
|
|
newmis.touch = superspike_touch;
|
|
newmis.weapon = DMSG_SNG;
|
|
setmodel (newmis, "progs/s_spike.mdl");
|
|
setsize (newmis, VEC_ORIGIN, VEC_ORIGIN);
|
|
|
|
KickPlayer(-2, self);
|
|
};
|
|
|
|
|
|
void(float ox) W_FireSpikes =
|
|
{
|
|
if (self.ammo_nails >= 2 && self.current_weapon == WEAP_SNG) {
|
|
W_FireSuperSpikes ();
|
|
return;
|
|
}
|
|
|
|
local vector dir;
|
|
|
|
if (self.classname == "player")
|
|
makevectors (self.v_angle);
|
|
else
|
|
makevectors (self.angles);
|
|
|
|
if (self.ammo_nails < 1)
|
|
if (self.classname == "player") {
|
|
self.current_weapon = W_BestWeapon ();
|
|
W_SetCurrentAmmo ();
|
|
W_PrintWeaponMessage();
|
|
return;
|
|
}
|
|
|
|
sound (self, CHAN_WEAPON, "weapons/rocket1i.wav", 1, ATTN_NORM);
|
|
Attack_Finished(0.2);
|
|
self.currentammo = self.ammo_nails = self.ammo_nails - 1;
|
|
if (self.classname == "player") {
|
|
dir = aim (self, 1000);
|
|
launch_spike (self.origin + '0 0 16' + v_right*ox, dir);
|
|
KickPlayer(-2, self);
|
|
} else if (self.classname == "monster_army") { //- OfN -
|
|
dir = Grunty_LeadShot();
|
|
grunty_spike(self.origin + '0 0 16' + v_right*ox, dir);
|
|
}
|
|
};
|
|
|
|
//.float hit_z;
|
|
void() spike_touch =
|
|
{
|
|
if (other.solid == SOLID_TRIGGER)
|
|
return; // trigger field, do nothing
|
|
|
|
if (pointcontents(self.origin) == CONTENTS_SKY) {
|
|
dremove(self);
|
|
return;
|
|
}
|
|
|
|
//WK Sweep mines at point of impact
|
|
GuerillaMineSweep(self.origin);
|
|
|
|
// hit something that bleeds
|
|
if (other.takedamage) {
|
|
spawn_touchblood (9);
|
|
deathmsg = self.weapon;
|
|
|
|
//- OfN - Wizard projectiles do SCRAG_DMG damage
|
|
if (self.classname == "wizspike") {
|
|
local float wizdmg;
|
|
wizdmg = SCRAG_DMG;
|
|
|
|
if (other.classname == "player") {
|
|
if (other.cutf_items & CUTF_DEMONLORE) // if we have demon lore, it does less damage to us
|
|
wizdmg = wizdmg * 0.8;
|
|
}
|
|
|
|
TF_T_Damage (other, self, self.owner, wizdmg, TF_TD_NOTTEAM, 0);
|
|
sound (self, CHAN_MISC, "effects/crunch.wav", 0.4, ATTN_NORM); // any better sound?
|
|
} else if (self.owner.classname == "grenade") // - OfN -
|
|
TF_T_Damage (other, self, self.owner.owner, 12, TF_TD_NOTTEAM, TF_TD_NAIL);
|
|
else
|
|
TF_T_Damage (other, self, self.owner, 12, TF_TD_NOTTEAM, TF_TD_NAIL);
|
|
} else {
|
|
if (other.classname == "force_field") //- OfN - Makes field explosion b4 removing it
|
|
FieldExplosion(other,self.origin,self);
|
|
else {
|
|
WriteByte (MSG_MULTICAST, SVC_TEMPENTITY);
|
|
if (self.classname == "wizspike")
|
|
WriteByte (MSG_MULTICAST, TE_WIZSPIKE);
|
|
else if (self.classname == "knightspike")
|
|
WriteByte (MSG_MULTICAST, TE_KNIGHTSPIKE);
|
|
else
|
|
WriteByte (MSG_MULTICAST, TE_SPIKE);
|
|
WriteCoord (MSG_MULTICAST, self.origin_x);
|
|
WriteCoord (MSG_MULTICAST, self.origin_y);
|
|
WriteCoord (MSG_MULTICAST, self.origin_z);
|
|
multicast (self.origin, MULTICAST_PHS);
|
|
}
|
|
}
|
|
|
|
#ifdef DEMO_STUFF
|
|
// Remove any camera's locks on this missile
|
|
if (self.enemy)
|
|
CamProjectileLockOff();
|
|
#endif
|
|
|
|
dremove(self);
|
|
};
|
|
|
|
/*
|
|
===============================================================================
|
|
|
|
PLAYER WEAPON USE
|
|
|
|
===============================================================================
|
|
*/
|
|
|
|
void() W_SetCurrentAmmo =
|
|
{
|
|
if (self.health <= 0 || self.current_weapon == 0)
|
|
return; // get out of any weapon firing states
|
|
|
|
player_run();
|
|
|
|
self.items = self.items & ~(IT_SHELLS | IT_NAILS | IT_ROCKETS | IT_CELLS);
|
|
self.weapon = 0;
|
|
|
|
//WK Set armor here... update armor picture
|
|
self.items = self.items & ~(IT_ARMOR1 | IT_ARMOR2 | IT_ARMOR3);
|
|
if (self.armortype >= 0.8)
|
|
self.items = self.items | IT_ARMOR3;
|
|
else if (self.armortype >= 0.6)
|
|
self.items = self.items | IT_ARMOR2;
|
|
else if (self.armortype >= 0.3)
|
|
self.items = self.items | IT_ARMOR1;
|
|
|
|
//WK Show nothing for custom playerclass while building
|
|
if ( self.playerclass == PC_CUSTOM
|
|
&& (self.done_custom & CUSTOM_BUILDING)) {
|
|
//We are building a class
|
|
self.currentammo = 0;
|
|
self.weaponmodel = "";
|
|
return;
|
|
}
|
|
if (self.current_weapon == WEAP_AXE) {
|
|
self.currentammo = 0;
|
|
|
|
if (self.cutf_items & CUTF_KNIFE) { //WK
|
|
if (self.job & JOB_BLOODY_KNIFE)
|
|
self.weaponmode = 1;
|
|
else
|
|
self.weaponmode = 0; //CH maybe fix bug that knife is bloody when not?
|
|
if (self.weaponmode == 0)
|
|
self.weaponmodel = "progs/v_knife.mdl"; //Nonbloody
|
|
else
|
|
self.weaponmodel = "progs/v_knife2.mdl"; //Bloody
|
|
} else
|
|
self.weaponmodel = "progs/v_axe.mdl";
|
|
|
|
self.weaponframe = 0;
|
|
} else if (self.current_weapon == WEAP_HOOK) {
|
|
self.currentammo = 0;
|
|
self.weaponmodel = "progs/v_grap.mdl";
|
|
self.weaponframe = 0;
|
|
} else if (self.current_weapon == WEAP_SPANNER) {
|
|
self.currentammo = self.ammo_cells;
|
|
self.weaponmodel = "progs/v_span.mdl";
|
|
self.weaponframe = 0;
|
|
} else if (self.current_weapon == WEAP_SHOTGUN) {
|
|
self.currentammo = self.ammo_shells;
|
|
self.items = self.items | IT_CELLS;
|
|
if (!(self.tfstate & TFSTATE_RELOADING)) {
|
|
self.weaponmodel = "progs/v_shot.mdl";
|
|
self.weaponframe = 0;
|
|
}
|
|
self.items = self.items | IT_SHELLS;
|
|
self.weapon = IT_SHOTGUN;
|
|
} else if (self.current_weapon == WEAP_SUPER_SHOTGUN) {
|
|
self.currentammo = self.ammo_shells;
|
|
if (!(self.tfstate & TFSTATE_RELOADING)) {
|
|
self.weaponmodel = "progs/v_shot2.mdl";
|
|
self.weaponframe = 0;
|
|
}
|
|
self.items = self.items | IT_SHELLS;
|
|
self.weapon = IT_SUPER_SHOTGUN;
|
|
} else if (self.current_weapon == WEAP_NAILGUN) {
|
|
self.currentammo = self.ammo_nails;
|
|
if (!(self.tfstate & TFSTATE_RELOADING)) {
|
|
self.weaponmodel = "progs/v_nail.mdl";
|
|
self.weaponframe = 0;
|
|
}
|
|
|
|
self.items = self.items | IT_NAILS;
|
|
self.weapon = IT_NAILGUN;
|
|
} else if (self.current_weapon == WEAP_LIGHT_ASSAULT) {
|
|
self.currentammo = self.ammo_nails;
|
|
if (!(self.tfstate & TFSTATE_RELOADING)) {
|
|
self.weaponmodel = "progs/v_nail2.mdl";
|
|
self.weaponframe = 0;
|
|
}
|
|
self.items = self.items | IT_NAILS;
|
|
self.weapon = IT_LIGHT_ASSAULT;
|
|
} else if (self.current_weapon == WEAP_GRENADE_LAUNCHER) {
|
|
self.currentammo = self.ammo_rockets;
|
|
if (!(self.tfstate & TFSTATE_RELOADING)) {
|
|
self.weaponmodel = "progs/v_rock.mdl";
|
|
self.weaponframe = 0;
|
|
}
|
|
|
|
self.weapon = IT_GRENADE_LAUNCHER;
|
|
self.items = self.items | IT_ROCKETS;
|
|
} else if (self.current_weapon == WEAP_ROCKET_LAUNCHER) {
|
|
self.currentammo = self.ammo_rockets;
|
|
if (!(self.tfstate & TFSTATE_RELOADING)) {
|
|
self.weaponmodel = "progs/v_rock2.mdl";
|
|
self.weaponframe = 0;
|
|
}
|
|
self.items = self.items | IT_ROCKETS;
|
|
self.weapon = IT_ROCKET_LAUNCHER;
|
|
} else if (self.current_weapon == WEAP_LIGHTNING) {
|
|
self.currentammo = self.ammo_cells;
|
|
if (!(self.tfstate & TFSTATE_RELOADING)) {
|
|
self.weaponmodel = "progs/v_light.mdl";
|
|
self.weaponframe = 0;
|
|
}
|
|
self.items = self.items | IT_CELLS;
|
|
self.weapon = IT_LIGHTNING;
|
|
} else if (self.current_weapon == WEAP_SNIPER_RIFLE) {
|
|
self.currentammo = self.ammo_shells;
|
|
if (!(self.tfstate & TFSTATE_RELOADING)) {
|
|
self.weaponmodel = "progs/v_srifle.mdl";
|
|
self.weaponframe = 0;
|
|
}
|
|
self.items = self.items | IT_SHELLS;
|
|
self.weapon = IT_SHOTGUN;
|
|
#ifndef NO_AUTORIFLE
|
|
} else if (self.current_weapon == WEAP_AUTO_RIFLE) {
|
|
self.currentammo = self.ammo_shells;
|
|
if (!(self.tfstate & TFSTATE_RELOADING)) {
|
|
self.weaponmodel = "progs/v_srifle.mdl";
|
|
self.weaponframe = 0;
|
|
}
|
|
self.items = self.items | IT_SHELLS;
|
|
self.weapon = IT_SUPER_SHOTGUN;
|
|
#endif
|
|
} else if (self.current_weapon == WEAP_ASSAULT_CANNON) {
|
|
self.currentammo = self.ammo_shells;
|
|
if (!(self.tfstate & TFSTATE_RELOADING)) {
|
|
self.weaponmodel = "progs/v_asscan.mdl";
|
|
self.weaponframe = 0;
|
|
}
|
|
self.items = self.items | IT_SHELLS;
|
|
self.weapon = IT_ROCKET_LAUNCHER;
|
|
} else if (self.current_weapon == WEAP_FLAMETHROWER) {
|
|
self.currentammo = self.ammo_cells;
|
|
if (!(self.tfstate & TFSTATE_RELOADING)) {
|
|
self.weaponmodel = "progs/v_rock.mdl";
|
|
self.weaponframe = 0;
|
|
}
|
|
self.items = self.items | IT_CELLS;
|
|
self.weapon = IT_GRENADE_LAUNCHER;
|
|
} else if (self.current_weapon == WEAP_INCENDIARY) {
|
|
self.currentammo = self.ammo_rockets;
|
|
if (!(self.tfstate & TFSTATE_RELOADING)) {
|
|
self.weaponmodel = "progs/v_rock2.mdl";
|
|
self.weaponframe = 0;
|
|
}
|
|
self.items = self.items | IT_ROCKETS;
|
|
self.weapon = IT_ROCKET_LAUNCHER;
|
|
} else if (self.current_weapon == WEAP_MEDIKIT) {
|
|
self.currentammo = 0;
|
|
self.weaponmodel = "progs/v_medi.mdl";
|
|
self.weaponframe = 0;
|
|
} else if (self.current_weapon == WEAP_TRANQ) {
|
|
self.currentammo = self.ammo_nails;
|
|
if (!(self.tfstate & TFSTATE_RELOADING)) {
|
|
self.weaponmodel = "progs/v_shot.mdl";
|
|
self.weaponframe = 0;
|
|
}
|
|
self.items = self.items | IT_NAILS;
|
|
self.weapon = IT_SHOTGUN;
|
|
} else if (self.current_weapon == WEAP_LASER) {
|
|
self.currentammo = self.ammo_nails;
|
|
|
|
if (!(self.tfstate & TFSTATE_RELOADING)) {
|
|
self.weaponmodel = "progs/v_rail.mdl";
|
|
self.weaponframe = 0;
|
|
}
|
|
|
|
self.items = self.items | IT_NAILS;
|
|
self.weapon = IT_SHOTGUN;
|
|
} else if (self.current_weapon == WEAP_DAEDALUS) {
|
|
self.currentammo = self.ammo_cells;
|
|
|
|
if (!(self.tfstate & TFSTATE_RELOADING)) {
|
|
self.weaponmodel = "progs/v_rock.mdl";
|
|
self.weaponframe = 0;
|
|
}
|
|
|
|
self.items = self.items | IT_CELLS;
|
|
self.weapon = IT_LIGHTNING;
|
|
} else if (self.current_weapon == WEAP_MAUSER) {
|
|
self.currentammo = self.ammo_nails;
|
|
|
|
if (!(self.tfstate & TFSTATE_RELOADING)) {
|
|
self.weaponmodel = "progs/v_rail.mdl";
|
|
self.weaponframe = 0;
|
|
}
|
|
|
|
self.items = self.items | IT_NAILS;
|
|
self.weapon = IT_LIGHT_ASSAULT;
|
|
} else if (self.current_weapon == WEAP_AIRF) {
|
|
self.currentammo = 0;
|
|
if (!(self.tfstate & TFSTATE_RELOADING)) {
|
|
self.weaponmodel = "progs/v_airgun.mdl";
|
|
self.weaponframe = 0;
|
|
}
|
|
} else if (self.current_weapon == WEAP_SNG) {
|
|
self.currentammo = self.ammo_nails;
|
|
if (!(self.tfstate & TFSTATE_RELOADING)) {
|
|
self.weaponmodel = "progs/v_nail2.mdl";
|
|
self.weaponframe = 0;
|
|
}
|
|
self.items = self.items | IT_NAILS;
|
|
self.weapon = IT_LIGHT_ASSAULT;//IT_SUPER_NAILGUN;
|
|
} else if (self.current_weapon == WEAP_LASERCANNON) {
|
|
self.currentammo = self.ammo_cells;
|
|
if (!(self.tfstate & TFSTATE_RELOADING)) {
|
|
self.weaponmodel = "progs/v_laserg.mdl";
|
|
self.weaponframe = 0;
|
|
}
|
|
|
|
self.items = self.items | IT_CELLS;
|
|
self.weapon = WEAP_LASERCANNON;
|
|
} else {
|
|
self.currentammo = 0;
|
|
self.weaponmodel = "";
|
|
self.weaponframe = 0;
|
|
}
|
|
};
|
|
|
|
float() W_BestWeapon =
|
|
{
|
|
local float it;
|
|
|
|
it = self.weapons_carried;
|
|
|
|
if (self.ammo_cells >= 1 && (it & WEAP_LIGHTNING) && self.waterlevel <= 1)
|
|
return WEAP_LIGHTNING;
|
|
if(self.ammo_shells >= 1 && (it & WEAP_SNIPER_RIFLE) )
|
|
return WEAP_SNIPER_RIFLE;
|
|
if(self.ammo_rockets >= 1 && (it & WEAP_ROCKET_LAUNCHER) )
|
|
return WEAP_ROCKET_LAUNCHER;
|
|
if (self.ammo_cells >= 5 && (it & WEAP_DAEDALUS) )
|
|
return WEAP_DAEDALUS;
|
|
if (self.ammo_cells >= 6 && (self.ammo_shells >= 1) && (it & WEAP_ASSAULT_CANNON))
|
|
return WEAP_ASSAULT_CANNON;
|
|
if(self.ammo_nails >= 10 && (it & WEAP_LIGHT_ASSAULT) )
|
|
return WEAP_LIGHT_ASSAULT;
|
|
if(self.ammo_rockets >= 3 && (it & WEAP_INCENDIARY) )
|
|
return WEAP_INCENDIARY;
|
|
if(self.ammo_rockets >= 1 && (it & WEAP_GRENADE_LAUNCHER) )
|
|
return WEAP_GRENADE_LAUNCHER;
|
|
if (self.ammo_cells >= 1 && (it & WEAP_LASERCANNON))
|
|
return WEAP_LASERCANNON;
|
|
if(self.ammo_nails >= 2 && (it & WEAP_SNG))
|
|
return WEAP_SNG;
|
|
if (self.ammo_cells >= 1 && (it & WEAP_FLAMETHROWER))
|
|
return WEAP_FLAMETHROWER;
|
|
if(self.ammo_shells >= 2 && (it & WEAP_SUPER_SHOTGUN) )
|
|
return WEAP_SUPER_SHOTGUN;
|
|
if (self.ammo_nails >= 1 && (it & WEAP_LASER) )
|
|
return WEAP_LASER;
|
|
if(self.ammo_nails >= 1 && (it & WEAP_NAILGUN) )
|
|
return WEAP_NAILGUN;
|
|
if(self.ammo_shells >= 1 && (it & WEAP_SHOTGUN) )
|
|
return WEAP_SHOTGUN;
|
|
if (self.ammo_nails >= 1 && (it & WEAP_MAUSER) )
|
|
return WEAP_MAUSER;
|
|
if (self.ammo_nails >= 1 && (it & WEAP_TRANQ) )
|
|
return WEAP_TRANQ;
|
|
|
|
if (it & WEAP_MEDIKIT)
|
|
return WEAP_MEDIKIT;
|
|
|
|
if (it & WEAP_AIRF)
|
|
return WEAP_AIRF;
|
|
|
|
if (it & WEAP_SPANNER)
|
|
return WEAP_SPANNER;
|
|
if (it & WEAP_AXE)
|
|
return WEAP_AXE;
|
|
|
|
return 0;
|
|
};
|
|
|
|
float() W_CheckNoAmmo =
|
|
{
|
|
if (self.current_weapon == WEAP_MEDIKIT)
|
|
return TRUE;
|
|
if (self.current_weapon == WEAP_AIRF)
|
|
return TRUE;
|
|
if (self.current_weapon == WEAP_AXE || self.current_weapon == WEAP_HOOK || self.current_weapon == WEAP_SPANNER)
|
|
return TRUE;
|
|
if (self.current_weapon == WEAP_INCENDIARY) {
|
|
if (self.currentammo >= 3)
|
|
return TRUE;
|
|
} else if (self.current_weapon == WEAP_DAEDALUS) {
|
|
if (self.currentammo >= 5)
|
|
return TRUE;
|
|
} else if (self.current_weapon == WEAP_SNIPER_RIFLE) {
|
|
if (self.currentammo >= 5)
|
|
return TRUE;
|
|
} else if (self.currentammo > 0)
|
|
return TRUE;
|
|
|
|
self.current_weapon = W_BestWeapon ();
|
|
W_SetCurrentAmmo ();
|
|
W_PrintWeaponMessage();
|
|
|
|
// drop the weapon down
|
|
return FALSE;
|
|
};
|
|
|
|
/*====================
|
|
W_Reload
|
|
Is called when weapon has finished reloading
|
|
====================*/
|
|
void() W_Reload_shotgun =
|
|
{
|
|
self.owner.tfstate = self.owner.tfstate - (self.owner.tfstate & TFSTATE_RELOADING);
|
|
self.owner.weaponmodel = "progs/v_shot.mdl";
|
|
sprint(self.owner, PRINT_LOW, "finished reloading\n");
|
|
self.owner.StatusRefreshTime = time + 0.1;
|
|
|
|
dremove(self);
|
|
};
|
|
|
|
void() W_Reload_light_assault =
|
|
{
|
|
self.owner.tfstate = self.owner.tfstate - (self.owner.tfstate & TFSTATE_RELOADING);
|
|
self.owner.weaponmodel = "progs/v_nail2.mdl";
|
|
sprint(self.owner, PRINT_LOW, "finished reloading\n");
|
|
self.owner.StatusRefreshTime = time + 0.1;
|
|
|
|
dremove(self);
|
|
};
|
|
|
|
void() W_Reload_super_shotgun =
|
|
{
|
|
self.owner.tfstate = self.owner.tfstate - (self.owner.tfstate & TFSTATE_RELOADING);
|
|
self.owner.weaponmodel = "progs/v_shot2.mdl";
|
|
sprint(self.owner, PRINT_LOW, "finished reloading\n");
|
|
self.owner.StatusRefreshTime = time + 0.1;
|
|
|
|
dremove(self);
|
|
};
|
|
|
|
void() W_Reload_grenade_launcher =
|
|
{
|
|
self.owner.tfstate = self.owner.tfstate - (self.owner.tfstate & TFSTATE_RELOADING);
|
|
self.owner.weaponmodel = "progs/v_rock.mdl";
|
|
sprint(self.owner, PRINT_LOW, "finished reloading\n");
|
|
self.owner.StatusRefreshTime = time + 0.1;
|
|
|
|
dremove(self);
|
|
};
|
|
|
|
void() W_Reload_rocket_launcher =
|
|
{
|
|
self.owner.tfstate = self.owner.tfstate - (self.owner.tfstate & TFSTATE_RELOADING);
|
|
self.owner.weaponmodel = "progs/v_rock2.mdl";
|
|
sprint(self.owner, PRINT_LOW, "finished reloading\n");
|
|
self.owner.StatusRefreshTime = time + 0.1;
|
|
|
|
dremove(self);
|
|
};
|
|
|
|
void() W_Reload_laser_cannon =
|
|
{
|
|
self.owner.tfstate = self.owner.tfstate - (self.owner.tfstate & TFSTATE_RELOADING);
|
|
self.owner.weaponmodel = "progs/v_laserg.mdl";
|
|
sprint(self.owner, PRINT_LOW, "Laser Cannon charged\n");
|
|
self.owner.StatusRefreshTime = time + 0.1;
|
|
|
|
dremove(self);
|
|
};
|
|
|
|
float() CheckForReload =
|
|
{
|
|
local entity tWeapon;
|
|
|
|
if (self.current_weapon == WEAP_SHOTGUN) {
|
|
if (self.reload_shotgun >= RE_SHOTGUN && self.ammo_shells > 0) {
|
|
self.reload_shotgun = 0;
|
|
if (self.ammo_shells < RE_SHOTGUN)
|
|
self.reload_shotgun = RE_SHOTGUN - self.ammo_shells;
|
|
|
|
sprint (self, PRINT_HIGH, "reloading...\n");
|
|
self.tfstate = (self.tfstate | TFSTATE_RELOADING);
|
|
tWeapon = spawn();
|
|
tWeapon.owner = self;
|
|
tWeapon.classname = "timer";
|
|
tWeapon.nextthink = time + RE_SHOTGUN_TIME;
|
|
tWeapon.think = W_Reload_shotgun;
|
|
|
|
self.weaponmodel = "";
|
|
self.weaponframe = 0;
|
|
|
|
return TRUE;
|
|
}
|
|
} else if (self.current_weapon == WEAP_SUPER_SHOTGUN) {
|
|
if (self.reload_super_shotgun > RE_SUPER_SHOTGUN)
|
|
self.reload_super_shotgun = RE_SUPER_SHOTGUN;
|
|
if ( self.reload_super_shotgun >= RE_SUPER_SHOTGUN
|
|
&& self.ammo_shells > 0) {
|
|
self.reload_super_shotgun = 0;
|
|
if (self.ammo_shells < RE_SUPER_SHOTGUN)
|
|
self.reload_super_shotgun =
|
|
RE_SUPER_SHOTGUN - self.ammo_shells;
|
|
|
|
sprint (self, PRINT_HIGH, "reloading...\n");
|
|
self.tfstate = (self.tfstate | TFSTATE_RELOADING);
|
|
tWeapon = spawn();
|
|
tWeapon.owner = self;
|
|
tWeapon.classname = "timer";
|
|
tWeapon.nextthink = time + RE_SUPER_SHOTGUN_TIME;
|
|
tWeapon.think = W_Reload_super_shotgun;
|
|
|
|
self.weaponmodel = "";
|
|
self.weaponframe = 0;
|
|
|
|
return TRUE;
|
|
}
|
|
} else if (self.current_weapon == WEAP_LIGHT_ASSAULT) {
|
|
if ( self.reload_light_assault >= RE_LIGHT_ASSAULT
|
|
&& self.ammo_nails > 0) {
|
|
self.reload_light_assault = 0;
|
|
if (self.ammo_nails < RE_LIGHT_ASSAULT)
|
|
self.reload_light_assault = RE_LIGHT_ASSAULT - self.ammo_nails;
|
|
|
|
sprint (self, PRINT_HIGH, "reloading...\n");
|
|
self.tfstate = (self.tfstate | TFSTATE_RELOADING);
|
|
tWeapon = spawn();
|
|
tWeapon.owner = self;
|
|
tWeapon.classname = "timer";
|
|
tWeapon.nextthink = time + RE_LIGHT_ASSAULT_TIME;
|
|
tWeapon.think = W_Reload_light_assault;
|
|
|
|
self.weaponmodel = "";
|
|
self.weaponframe = 0;
|
|
|
|
return TRUE;
|
|
}
|
|
} else if (self.current_weapon == WEAP_LASERCANNON) {
|
|
if ( self.reload_laser_cannon >= RE_LASER_CANNON
|
|
&& self.ammo_cells > 0) {
|
|
self.reload_laser_cannon = 0;
|
|
if (self.ammo_cells < RE_LASER_CANNON)
|
|
self.reload_laser_cannon = RE_LASER_CANNON - self.ammo_cells;
|
|
|
|
sprint (self, PRINT_HIGH, "Charging cannon...\n");
|
|
self.tfstate = (self.tfstate | TFSTATE_RELOADING);
|
|
tWeapon = spawn();
|
|
tWeapon.owner = self;
|
|
tWeapon.classname = "timer";
|
|
tWeapon.nextthink = time + RE_LASER_CANNON_TIME;
|
|
tWeapon.think = W_Reload_laser_cannon;
|
|
|
|
self.weaponmodel = "";
|
|
self.weaponframe = 0;
|
|
|
|
return TRUE;
|
|
}
|
|
} else if (self.current_weapon == WEAP_GRENADE_LAUNCHER) {
|
|
if ( self.reload_grenade_launcher >= RE_GRENADE_LAUNCHER
|
|
&& self.ammo_rockets > 0) {
|
|
self.reload_grenade_launcher = 0;
|
|
if (self.ammo_rockets < RE_GRENADE_LAUNCHER)
|
|
self.reload_grenade_launcher =
|
|
RE_GRENADE_LAUNCHER - self.ammo_rockets;
|
|
|
|
sprint (self, PRINT_HIGH, "reloading...\n");
|
|
self.tfstate = (self.tfstate | TFSTATE_RELOADING);
|
|
tWeapon = spawn();
|
|
tWeapon.owner = self;
|
|
tWeapon.classname = "timer";
|
|
tWeapon.nextthink = time + RE_GRENADE_LAUNCHER_TIME;
|
|
tWeapon.think = W_Reload_grenade_launcher;
|
|
|
|
self.weaponmodel = "";
|
|
self.weaponframe = 0;
|
|
|
|
return TRUE;
|
|
}
|
|
} else if (self.current_weapon == WEAP_ROCKET_LAUNCHER) {
|
|
if ( self.reload_rocket_launcher >= RE_ROCKET_LAUNCHER
|
|
&& self.ammo_rockets > 0) {
|
|
self.reload_rocket_launcher = 0;
|
|
if (self.ammo_rockets < RE_ROCKET_LAUNCHER)
|
|
self.reload_rocket_launcher = RE_ROCKET_LAUNCHER
|
|
- self.ammo_rockets;
|
|
|
|
sprint (self, PRINT_HIGH, "reloading...\n");
|
|
self.tfstate = (self.tfstate | TFSTATE_RELOADING);
|
|
tWeapon = spawn();
|
|
tWeapon.owner = self;
|
|
tWeapon.classname = "timer";
|
|
if (self.tf_items & NIT_CLUSTER_ROCKETS)
|
|
tWeapon.nextthink = time + RE_ROCKET_LAUNCHER_TIME - 1;
|
|
else
|
|
tWeapon.nextthink = time + RE_ROCKET_LAUNCHER_TIME;
|
|
tWeapon.think = W_Reload_rocket_launcher;
|
|
|
|
self.weaponmodel = "";
|
|
self.weaponframe = 0;
|
|
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
};
|
|
|
|
/*
|
|
============
|
|
W_Attack
|
|
|
|
An attack impulse can be triggered now
|
|
============
|
|
*/
|
|
void() player_axe1;
|
|
void() player_axeb1;
|
|
void() player_axec1;
|
|
void() player_axed1;
|
|
|
|
void() player_axe2;
|
|
void() player_axeb2;
|
|
void() player_axec2;
|
|
void() player_axed2;
|
|
|
|
void() player_shot1;
|
|
void() player_nail1;
|
|
|
|
void() player_snail1; // ofn SNG
|
|
|
|
void() player_light1;
|
|
void() player_light_assault1;
|
|
void() player_rocket1;
|
|
|
|
#ifndef NO_AUTORIFLE
|
|
void() player_autorifle1;
|
|
#endif
|
|
void() player_assaultcannon1;
|
|
void() player_assaultcannonup1;
|
|
void() player_assaultcannondown1;
|
|
void() player_medikit1;
|
|
void() player_medikitb1;
|
|
void() player_medikitc1;
|
|
void() player_medikitd1;
|
|
void() player_bioweapon1;
|
|
void() player_bioweaponb1;
|
|
void() player_bioweaponc1;
|
|
void() player_bioweapond1;
|
|
void() player_chain1;
|
|
void() player_chain2;
|
|
void() player_chain3;
|
|
void() player_chain4;
|
|
void() player_chain5;
|
|
|
|
|
|
void() W_Attack =
|
|
{
|
|
local float r;
|
|
|
|
if (!W_CheckNoAmmo ())
|
|
return;
|
|
|
|
if (self.playerclass == PC_CUSTOM && (self.done_custom & CUSTOM_BUILDING))
|
|
return;
|
|
|
|
if (self.tfstate & TFSTATE_RELOADING)
|
|
return;
|
|
|
|
if (self.penance_time > time)
|
|
return;
|
|
|
|
//WK Tranq does not remove disguise
|
|
//GR Spy can fire and keep their skin if not color-disguised.
|
|
if (self.undercover_team && (self.current_weapon != WEAP_TRANQ && self.current_weapon != WEAP_MAUSER))
|
|
Spy_RemoveDisguise(self);
|
|
|
|
if (self.job & JOB_THIEF && (self.job & JOB_ACTIVE || self.job & JOB_FULL_HIDE))
|
|
RevealThief(self,FALSE);
|
|
|
|
//WK When conced you randomly don't fire
|
|
if (self.tfstate & TFSTATE_CONCUSSIONED)
|
|
if (random() <= 0.3)
|
|
return;
|
|
|
|
makevectors (self.v_angle); // calculate forward angle for velocity
|
|
self.show_hostile = time + 1; // wake monsters up
|
|
|
|
if (self.current_weapon == WEAP_AXE) {
|
|
local float af = 0.5;
|
|
|
|
sound (self, CHAN_WEAPON, "weapons/ax1.wav", 1, ATTN_NORM);
|
|
r = random();
|
|
|
|
if (self.cutf_items & CUTF_KNIFE) {
|
|
find_melee (self, TRUE, 72);
|
|
|
|
if (!is_backstab (FALSE) || !trace_ent.takedamage ||
|
|
(trace_ent.classname != "player" && !IsMonster (trace_ent))) {
|
|
af -= 0.1;
|
|
|
|
if (r < 0.5)
|
|
player_axe2 ();
|
|
else
|
|
player_axec2 ();
|
|
} else {
|
|
af += 0.3;
|
|
|
|
if (r < 0.5)
|
|
player_axeb1 ();
|
|
else
|
|
player_axed1 ();
|
|
}
|
|
} else {
|
|
if (r < 0.25)
|
|
player_axe1 ();
|
|
else if (r<0.5)
|
|
player_axeb1 ();
|
|
else if (r<0.75)
|
|
player_axec1 ();
|
|
else
|
|
player_axed1 ();
|
|
}
|
|
|
|
if (self.cutf_items & CUTF_CLOSECOMBAT)
|
|
af -= 0.1;
|
|
|
|
Attack_Finished (af);
|
|
|
|
} else if (self.current_weapon == WEAP_SPANNER) {
|
|
Attack_Finished(0.35); //WK Berserk with spanner
|
|
sound (self, CHAN_WEAPON, "weapons/ax1.wav", 1, ATTN_NORM);
|
|
player_axe1 ();
|
|
} else if (self.current_weapon == WEAP_HOOK) {
|
|
if (!self.hook_out)
|
|
player_chain1 ();
|
|
|
|
Attack_Finished(1); //WK
|
|
} else if (self.current_weapon == WEAP_SHOTGUN) {
|
|
if (CheckForReload() == TRUE)
|
|
return;
|
|
|
|
player_shot1 ();
|
|
W_FireShotgun ();
|
|
|
|
self.reload_shotgun = self.reload_shotgun + 1;
|
|
self.StatusRefreshTime = time + 0.1;
|
|
|
|
CheckForReload();
|
|
Attack_Finished(0.5);
|
|
} else if (self.current_weapon == WEAP_SUPER_SHOTGUN) {
|
|
if (CheckForReload() == TRUE)
|
|
return;
|
|
|
|
player_shot1 ();
|
|
W_FireSuperShotgun ();
|
|
|
|
self.reload_super_shotgun = self.reload_super_shotgun + 2;
|
|
|
|
self.StatusRefreshTime = time + 0.1;
|
|
|
|
CheckForReload();
|
|
Attack_Finished(0.7);
|
|
} else if (self.current_weapon == WEAP_SNG) {
|
|
player_snail1 ();
|
|
} else if (self.current_weapon == WEAP_NAILGUN) {
|
|
player_nail1 ();
|
|
} else if (self.current_weapon == WEAP_LIGHT_ASSAULT) {
|
|
if (CheckForReload() == TRUE)
|
|
return;
|
|
|
|
player_light_assault1();
|
|
CheckForReload();
|
|
} else if (self.current_weapon == WEAP_GRENADE_LAUNCHER) {
|
|
if (CheckForReload() == TRUE)
|
|
return;
|
|
|
|
player_rocket1();
|
|
W_FireGrenade();
|
|
|
|
self.reload_grenade_launcher = self.reload_grenade_launcher + 1;
|
|
self.StatusRefreshTime = time + 0.1;
|
|
|
|
CheckForReload();
|
|
Attack_Finished(0.6);
|
|
} else if (self.current_weapon == WEAP_ROCKET_LAUNCHER) {
|
|
if (CheckForReload() == TRUE)
|
|
return;
|
|
|
|
player_rocket1();
|
|
W_FireRocket();
|
|
|
|
if (self.tf_items & NIT_CLUSTER_ROCKETS && self.cluster_mode == TRUE) //WK Clusters are multiple shots
|
|
self.reload_rocket_launcher = self.reload_rocket_launcher + 8;
|
|
else
|
|
self.reload_rocket_launcher = self.reload_rocket_launcher + 1;
|
|
self.StatusRefreshTime = time + 0.1;
|
|
|
|
CheckForReload();
|
|
//CH so that the dot can be removed quicker (reset after dot gone)
|
|
if (self.tf_items & NIT_RL_LASER_SIGHT)
|
|
Attack_Finished(0.1);
|
|
else
|
|
Attack_Finished(0.8);
|
|
} else if (self.current_weapon == WEAP_LIGHTNING) {
|
|
player_light1();
|
|
Attack_Finished(0.1);
|
|
sound (self, CHAN_AUTO, "weapons/lstart.wav", 1, ATTN_NORM);
|
|
} else if (self.current_weapon == WEAP_DAEDALUS) { //CHANGEME
|
|
player_rocket1();
|
|
W_FireDaedalus();
|
|
Attack_Finished(0.8);
|
|
sound (self, CHAN_AUTO, "weapons/lstart.wav", 1, ATTN_NORM);
|
|
} else if (self.current_weapon == WEAP_SNIPER_RIFLE) {
|
|
// Can't fire while jumping
|
|
// WK You can if you are a scuba commando!
|
|
if ( (self.flags & FL_ONGROUND) || (self.hook_out)
|
|
|| ((self.tf_items & NIT_SCUBA) && self.waterlevel) ) {
|
|
player_shot1();
|
|
W_FireSniperRifle();
|
|
Attack_Finished(SNIPER_RIFLE_RELOAD_TIME);
|
|
}
|
|
#ifndef NO_AUTORIFLE
|
|
} else if (self.current_weapon == WEAP_AUTO_RIFLE) {
|
|
player_autorifle1();
|
|
W_FireAutoRifle();
|
|
Attack_Finished(0.1);
|
|
#endif
|
|
} else if (self.current_weapon == WEAP_ASSAULT_CANNON) {
|
|
// Need 4 cells to power up the Assault Cannon
|
|
if (self.ammo_cells < 4) {
|
|
sprint (self, PRINT_MEDIUM, "Insufficient cells to power up the Assault Cannon.\n");
|
|
stuffcmd (self, "-attack;\n"); // to avoid flooding
|
|
} else {
|
|
self.ammo_cells = self.ammo_cells - 4;
|
|
|
|
// Can't move while firing the Assault Cannon :)
|
|
self.heat = 1;
|
|
makeImmune(self,time+2);
|
|
// self.immune_to_check = time + 2;
|
|
self.tfstate = self.tfstate | TFSTATE_ASSAULTCANNON;
|
|
TeamFortress_SetSpeed(self);
|
|
player_assaultcannonup1();
|
|
}
|
|
} else if (self.current_weapon == WEAP_FLAMETHROWER) {
|
|
player_shot1();
|
|
W_FireFlame();
|
|
if (self.waterlevel > 2)
|
|
Attack_Finished(1);
|
|
else
|
|
Attack_Finished(0.15);
|
|
} else if (self.current_weapon == WEAP_INCENDIARY) {
|
|
player_rocket1();
|
|
W_FireIncendiaryCannon();
|
|
Attack_Finished(1.2);
|
|
} else if (self.current_weapon == WEAP_MEDIKIT) {
|
|
sound (self, CHAN_WEAPON, "weapons/ax1.wav", 1, ATTN_NORM);
|
|
r = random();
|
|
if (r < 0.25)
|
|
player_medikit1 ();
|
|
else if (r < 0.5)
|
|
player_medikitb1 ();
|
|
else if (r < 0.75)
|
|
player_medikitc1 ();
|
|
else
|
|
player_medikitd1 ();
|
|
Attack_Finished(0.5);
|
|
} else if (self.current_weapon == WEAP_TRANQ) {
|
|
sound (self, CHAN_WEAPON, "weapons/dartgun.wav", 1, ATTN_NORM);
|
|
player_shot1 ();
|
|
W_FireTranq ();
|
|
Attack_Finished(1.5);
|
|
} else if (self.current_weapon == WEAP_LASER) {
|
|
sound (self, CHAN_WEAPON, "weapons/railgun.wav", 1, ATTN_NORM);
|
|
player_shot1();
|
|
W_FireLaser();
|
|
Attack_Finished(0.4);
|
|
} else if (self.current_weapon == WEAP_MAUSER) {
|
|
sound (self, CHAN_WEAPON, "weapons/dartgun.wav", 1, ATTN_NORM);
|
|
// player_shot1();
|
|
W_FireMauser();
|
|
Attack_Finished(2.0);
|
|
} else if (self.current_weapon == WEAP_AIRF) {
|
|
launch_horn();
|
|
}
|
|
if (self.current_weapon == WEAP_LASERCANNON) {
|
|
if (CheckForReload() == TRUE)
|
|
return;
|
|
|
|
player_laser1();
|
|
CheckForReload();
|
|
}
|
|
};
|
|
|
|
/*=========================
|
|
W_PrintWeaponMessage
|
|
|
|
Prints a message indicating
|
|
the current selected weapon,
|
|
if needed.
|
|
=========================*/
|
|
void() W_PrintWeaponMessage =
|
|
{
|
|
//WK Custom class friendly
|
|
if (self.current_weapon == WEAP_AXE) { // && (self.weapons_carried & WEAP_HOOK))
|
|
if (!(self.cutf_items & CUTF_KNIFE))
|
|
sprint(self, PRINT_MEDIUM, "Axe selected\n");
|
|
else {
|
|
if (self.weaponmode == 0)
|
|
sprint(self, PRINT_MEDIUM, "Knife selected\n"); //Nonbloody
|
|
else
|
|
sprint(self, PRINT_MEDIUM, "Bloody Knife selected\n"); //Bloody
|
|
}
|
|
} else if (self.current_weapon == WEAP_HOOK)
|
|
sprint(self, PRINT_MEDIUM, "Grappling Hook selected\n");
|
|
//CH because they use the same mdl, say what they are.
|
|
else if (self.current_weapon == WEAP_FLAMETHROWER)
|
|
sprint(self, PRINT_MEDIUM, "Flamethrower selected\n");
|
|
else if (self.current_weapon == WEAP_ROCKET_LAUNCHER) {
|
|
sprint(self, PRINT_MEDIUM, "Rocket Launcher selected");
|
|
|
|
if (self.cluster_mode == TRUE && self.tf_items & NIT_CLUSTER_ROCKETS) //set to fire cluster rockets
|
|
sprint(self, PRINT_MEDIUM, " Cluster Mode");
|
|
else if (self.tf_items & NIT_CLUSTER_ROCKETS)
|
|
sprint(self, PRINT_MEDIUM, " Normal Mode");
|
|
|
|
sprint(self, PRINT_MEDIUM, "\n");
|
|
} else if (self.current_weapon == WEAP_INCENDIARY)
|
|
sprint(self, PRINT_MEDIUM, "Incendiary cannon selected\n");
|
|
else if (self.current_weapon == WEAP_LIGHTNING)
|
|
sprint(self, PRINT_MEDIUM, "Lightning Gun selected\n");
|
|
else if (self.current_weapon == WEAP_SPANNER)
|
|
sprint(self, PRINT_MEDIUM, "Spanner readied\n");
|
|
|
|
else if (self.current_weapon == WEAP_GRENADE_LAUNCHER) {
|
|
if (self.weaponmode == GL_NORMAL)
|
|
sprint(self, PRINT_MEDIUM, "Normal grenade mode\n");
|
|
else if (self.weaponmode == GL_PIPEBOMB)
|
|
sprint(self, PRINT_MEDIUM, "Pipebomb mode\n");
|
|
} else if (self.current_weapon == WEAP_SNIPER_RIFLE)
|
|
sprint(self, PRINT_MEDIUM, "Sniper Rifle ready\n");
|
|
#ifndef NO_AUTORIFLE
|
|
else if (self.current_weapon == WEAP_AUTO_RIFLE)
|
|
sprint(self, PRINT_MEDIUM, "Rifle on fully auto\n");
|
|
#endif
|
|
else if (self.current_weapon == WEAP_TRANQ)
|
|
sprint(self, PRINT_MEDIUM, "Tranquiliser gun selected\n");
|
|
else if (self.current_weapon == WEAP_MEDIKIT)
|
|
sprint(self, PRINT_MEDIUM, "Medikit/Bioweapon readied\n");
|
|
else if (self.current_weapon == WEAP_MAUSER)
|
|
sprint(self, PRINT_MEDIUM, "Mauser WK-77 Assassin's Gun readied\n");
|
|
else if (self.current_weapon == WEAP_DAEDALUS)
|
|
sprint(self, PRINT_MEDIUM, "Daedalus Impulse Rifle readied\n");
|
|
else if (self.current_weapon == WEAP_LIGHTNING)
|
|
sprint(self, PRINT_MEDIUM, "Lighting gun selected\n");
|
|
else if (self.current_weapon == WEAP_AIRF)
|
|
sprint(self, PRINT_MEDIUM, "Airfist prepared\n");
|
|
else if (self.current_weapon == WEAP_LASERCANNON)
|
|
sprint(self, PRINT_MEDIUM, "Laser Cannon selected\n");
|
|
else if (self.current_weapon == WEAP_SNG)
|
|
sprint(self, PRINT_MEDIUM, "Super Nailgun selected\n");
|
|
else if (self.current_weapon == WEAP_NAILGUN)
|
|
sprint(self, PRINT_MEDIUM, "Nailgun selected\n");
|
|
else if (self.current_weapon == WEAP_LIGHT_ASSAULT)
|
|
sprint(self, PRINT_MEDIUM, "Light Assualt cannon selected\n");
|
|
};
|
|
|
|
/*
|
|
============
|
|
W_ChangeWeapon
|
|
|
|
============
|
|
*/
|
|
void() W_ChangeWeapon =
|
|
{
|
|
local float it, am, fl, loopck;
|
|
local float have_weapon, usable;
|
|
|
|
if (self.tfstate & TFSTATE_RELOADING)
|
|
return;
|
|
|
|
if (self.tfstate & TFSTATE_CANT_MOVE)
|
|
return;
|
|
|
|
it = self.weapons_carried;
|
|
fl = self.current_weapon;
|
|
am = 0;
|
|
usable = 0;
|
|
have_weapon = TRUE;
|
|
|
|
if (self.impulse == 1) {
|
|
if (!(it & (WEAP_HOOK | WEAP_MEDIKIT | WEAP_AXE | WEAP_SPANNER)))
|
|
have_weapon = FALSE;
|
|
|
|
while (!usable && have_weapon) {
|
|
if (fl == WEAP_SPANNER) {
|
|
fl = WEAP_MEDIKIT;
|
|
|
|
if (it & WEAP_MEDIKIT)
|
|
usable = 1;
|
|
} else if (fl == WEAP_MEDIKIT) {
|
|
fl = WEAP_HOOK;
|
|
|
|
if ((self.weapons_carried & WEAP_HOOK) && (fl & WEAP_HOOK))
|
|
usable = 1;
|
|
|
|
if (self.hook_out)
|
|
Reset_Grapple(self.hook);
|
|
} else if (fl == WEAP_HOOK) {
|
|
fl = WEAP_AXE;
|
|
|
|
if (it & WEAP_AXE)
|
|
usable = 1;
|
|
} else {
|
|
fl = WEAP_SPANNER;
|
|
|
|
if (it & WEAP_SPANNER)
|
|
usable = 1;
|
|
}
|
|
}
|
|
} else if ((self.weapons_carried & WEAP_HOOK) && (self.impulse == HOOK_IMP1 || self.impulse == HOOK_IMP2)) {
|
|
fl = WEAP_HOOK;
|
|
} else if (self.impulse == AXE_IMP) {
|
|
if (!(it & (WEAP_MEDIKIT | WEAP_AXE | WEAP_SPANNER)))
|
|
have_weapon = FALSE;
|
|
|
|
while (!usable && have_weapon) {
|
|
if (fl == WEAP_SPANNER) {
|
|
fl = WEAP_MEDIKIT;
|
|
|
|
if (it & WEAP_MEDIKIT)
|
|
usable = 1;
|
|
} else if (fl == WEAP_MEDIKIT) {
|
|
fl = WEAP_AXE;
|
|
|
|
if (it & WEAP_AXE)
|
|
usable = 1;
|
|
} else {
|
|
fl = WEAP_SPANNER;
|
|
|
|
if (it & WEAP_SPANNER)
|
|
usable = 1;
|
|
}
|
|
}
|
|
} else if (self.impulse == 2) {
|
|
if (!(((it & WEAP_SNIPER_RIFLE || it & WEAP_SHOTGUN || it & WEAP_TRANQ) && self.ammo_shells > 0) || (it & WEAP_LASER && self.ammo_nails > 0)|| (it & WEAP_DAEDALUS && self.ammo_cells > 0)))
|
|
have_weapon = FALSE;
|
|
loopck = 0;
|
|
while (!usable && have_weapon) {
|
|
loopck = (loopck + 1);
|
|
if (loopck >= 10)
|
|
have_weapon = FALSE;
|
|
if (fl == WEAP_SNIPER_RIFLE) {
|
|
fl = WEAP_SHOTGUN;
|
|
if (it & WEAP_SHOTGUN && self.ammo_shells > 0)
|
|
usable = 1;
|
|
} else if (fl == WEAP_SHOTGUN) {
|
|
fl = WEAP_TRANQ;
|
|
if (it & WEAP_TRANQ && self.ammo_nails > 0)
|
|
usable = 1;
|
|
} else if (fl == WEAP_TRANQ) {
|
|
fl = WEAP_LASER;
|
|
if (it & WEAP_LASER && self.ammo_nails > 0)
|
|
usable = 1;
|
|
} else {
|
|
fl = WEAP_SNIPER_RIFLE;
|
|
if (it & WEAP_SNIPER_RIFLE && self.ammo_shells > 0)
|
|
usable = 1;
|
|
}
|
|
}
|
|
|
|
if (fl == WEAP_LASER) {
|
|
if (self.ammo_nails < 1)
|
|
am = 1;
|
|
} else {
|
|
if (self.ammo_shells < 1)
|
|
am = 1;
|
|
}
|
|
} else if (self.impulse == 3) {
|
|
if (!(
|
|
#ifndef NO_AUTORIFLE
|
|
(it & WEAP_AUTO_RIFLE && self.ammo_shells > 0) ||
|
|
#endif
|
|
(it & WEAP_SUPER_SHOTGUN && self.ammo_shells >1)))
|
|
have_weapon = FALSE;
|
|
loopck = 0;
|
|
while (!usable && have_weapon) {
|
|
loopck = (loopck + 1);
|
|
if (loopck >= 10)
|
|
have_weapon = FALSE;
|
|
#ifndef NO_AUTORIFLE
|
|
if (fl == WEAP_AUTO_RIFLE) {
|
|
fl = WEAP_SUPER_SHOTGUN;
|
|
if (it & WEAP_SUPER_SHOTGUN && self.ammo_shells > 1)
|
|
usable = 1;
|
|
} else {
|
|
fl = WEAP_AUTO_RIFLE;
|
|
if (it & WEAP_AUTO_RIFLE && self.ammo_shells > 0)
|
|
usable = 1;
|
|
}
|
|
#else
|
|
fl = WEAP_SUPER_SHOTGUN;
|
|
if (it & WEAP_SUPER_SHOTGUN && self.ammo_cells > 1)
|
|
usable = 1;
|
|
else
|
|
have_weapon = FALSE;
|
|
#endif
|
|
}
|
|
|
|
if (fl == WEAP_SUPER_SHOTGUN) {
|
|
if (self.ammo_shells < 2)
|
|
am = 1;
|
|
} else {
|
|
if (self.ammo_shells < 1)
|
|
am = 1;
|
|
}
|
|
} else if (self.impulse == 4) {
|
|
/*
|
|
fl = WEAP_NAILGUN;
|
|
if (self.ammo_nails < 1)
|
|
am = 1;
|
|
*/
|
|
|
|
if (!(it & WEAP_LASERCANNON || it & WEAP_NAILGUN))
|
|
have_weapon = FALSE;
|
|
loopck = 0;
|
|
while (!usable && have_weapon) {
|
|
loopck = (loopck + 1);
|
|
if (loopck >= 10)
|
|
have_weapon = FALSE;
|
|
if (fl == WEAP_LASERCANNON) {
|
|
fl = WEAP_NAILGUN;
|
|
if (it & WEAP_NAILGUN)
|
|
usable = 1;
|
|
} else {
|
|
fl = WEAP_LASERCANNON;
|
|
if (it & WEAP_LASERCANNON)
|
|
usable = 1;
|
|
}
|
|
}
|
|
if (fl == WEAP_NAILGUN) {
|
|
if (self.ammo_nails < 1)
|
|
am = 1;
|
|
} else if (fl == WEAP_LASERCANNON) {
|
|
if (self.ammo_cells < 1)
|
|
am = 1;
|
|
}
|
|
} else if (self.impulse == 5) {
|
|
if (!(it & WEAP_LIGHT_ASSAULT || it & WEAP_MAUSER || it & WEAP_SNG))
|
|
have_weapon = FALSE;
|
|
loopck = 0;
|
|
while (!usable && have_weapon) {
|
|
loopck = (loopck + 1);
|
|
if (loopck >= 10)
|
|
have_weapon = FALSE;
|
|
if (fl == WEAP_LIGHT_ASSAULT) {
|
|
fl = WEAP_MAUSER;
|
|
if (it & WEAP_MAUSER)
|
|
usable = 1;
|
|
//-
|
|
} else if (fl == WEAP_MAUSER) {
|
|
fl = WEAP_SNG;
|
|
if (it & WEAP_SNG)
|
|
usable = 1;
|
|
//-
|
|
} else {
|
|
fl = WEAP_LIGHT_ASSAULT;
|
|
if (it & WEAP_LIGHT_ASSAULT)
|
|
usable = 1;
|
|
}
|
|
}
|
|
if (fl == WEAP_MAUSER) {
|
|
if (self.ammo_nails < 1)
|
|
am = 1;
|
|
} else if (fl == WEAP_LIGHT_ASSAULT) {
|
|
if (self.ammo_nails < 1)
|
|
am = 1;
|
|
} else if (fl == WEAP_SNG) {
|
|
if (self.ammo_nails < 2)
|
|
am = 1;
|
|
}
|
|
} else if (self.impulse == 6) {
|
|
if (!((it & WEAP_FLAMETHROWER && self.ammo_cells > 0) || (it & WEAP_GRENADE_LAUNCHER && self.ammo_rockets > 0)))
|
|
have_weapon = FALSE;
|
|
loopck = 0;
|
|
while (!usable && have_weapon) {
|
|
loopck = (loopck + 1);
|
|
if (loopck >= 10)
|
|
have_weapon = FALSE;
|
|
if (fl == WEAP_FLAMETHROWER) {
|
|
fl = WEAP_GRENADE_LAUNCHER;
|
|
self.weaponmode = GL_NORMAL;
|
|
if (it & WEAP_GRENADE_LAUNCHER && self.ammo_rockets > 0)
|
|
usable = 1;
|
|
} else {
|
|
fl = WEAP_FLAMETHROWER;
|
|
if (it & WEAP_FLAMETHROWER && self.ammo_cells > 0)
|
|
usable = 1;
|
|
}
|
|
}
|
|
if (fl == WEAP_FLAMETHROWER) {
|
|
if (self.ammo_cells < 1)
|
|
am = 1;
|
|
} else {
|
|
if (self.ammo_rockets < 1)
|
|
am = 1;
|
|
}
|
|
} else if (self.impulse == 7) {
|
|
if (!((it & WEAP_INCENDIARY && self.ammo_rockets > 2) || ((it & WEAP_ROCKET_LAUNCHER || it & WEAP_GRENADE_LAUNCHER) && self.ammo_rockets > 0) || (it & WEAP_ASSAULT_CANNON && self.ammo_shells > 0 && self.ammo_cells > 3)))
|
|
have_weapon = FALSE;
|
|
loopck = 0;
|
|
while (!usable && have_weapon) {
|
|
loopck = (loopck + 1);
|
|
if (loopck >= 10)
|
|
have_weapon = FALSE;
|
|
if (fl == WEAP_INCENDIARY) {
|
|
fl = WEAP_ROCKET_LAUNCHER;
|
|
if (it & WEAP_ROCKET_LAUNCHER && self.ammo_rockets > 0)
|
|
usable = 1;
|
|
} else if (fl == WEAP_ROCKET_LAUNCHER) {
|
|
if (self.tf_items & NIT_CLUSTER_ROCKETS) {
|
|
if (self.cluster_mode == TRUE) {
|
|
// bprint(PRINT_HIGH, "cluster mode FALSE\n");
|
|
self.cluster_mode = FALSE;
|
|
fl = WEAP_ASSAULT_CANNON;
|
|
if ( it & WEAP_ASSAULT_CANNON && self.ammo_shells > 0
|
|
&& self.ammo_cells > 3)
|
|
usable = 1;
|
|
} else {
|
|
// bprint(PRINT_HIGH, "cluster mode TRUE\n");
|
|
self.cluster_mode = TRUE;
|
|
fl = WEAP_ROCKET_LAUNCHER;
|
|
if (it & WEAP_ROCKET_LAUNCHER && self.ammo_rockets > 0)
|
|
usable = 1;
|
|
}
|
|
} else {
|
|
fl = WEAP_ASSAULT_CANNON;
|
|
if ( it & WEAP_ASSAULT_CANNON && self.ammo_shells > 0
|
|
&& self.ammo_cells > 3)
|
|
usable = 1;
|
|
}
|
|
} else if (fl == WEAP_ASSAULT_CANNON) {
|
|
fl = WEAP_GRENADE_LAUNCHER;
|
|
self.weaponmode = GL_PIPEBOMB;
|
|
if (it & WEAP_GRENADE_LAUNCHER && self.ammo_rockets > 0)
|
|
usable = 1;
|
|
} else {
|
|
fl = WEAP_INCENDIARY;
|
|
if (it & WEAP_INCENDIARY && self.ammo_rockets > 2)
|
|
usable = 1;
|
|
}
|
|
}
|
|
if (fl == WEAP_ASSAULT_CANNON) {
|
|
if (self.ammo_cells < 4 || self.ammo_shells < 1)
|
|
am = 1;
|
|
} else if (fl == WEAP_INCENDIARY) {
|
|
if (self.ammo_rockets < 3)
|
|
am = 1;
|
|
} else {
|
|
if (self.ammo_rockets < 1)
|
|
am = 1;
|
|
}
|
|
} else if (self.impulse == 8) {
|
|
if (!(it & WEAP_LIGHTNING || it & WEAP_DAEDALUS || it & WEAP_AIRF))
|
|
have_weapon = FALSE;
|
|
loopck = 0;
|
|
while (!usable && have_weapon) {
|
|
loopck = (loopck + 1);
|
|
if (loopck >= 10)
|
|
have_weapon = FALSE;
|
|
if (fl == WEAP_DAEDALUS) {
|
|
fl = WEAP_LIGHTNING;
|
|
if (it & WEAP_LIGHTNING) //WEAP_DAEDALUS
|
|
usable = 1;
|
|
} else if (fl == WEAP_LIGHTNING) {
|
|
fl = WEAP_AIRF;
|
|
if (it & WEAP_AIRF) //WEAP_DAEDALUS
|
|
usable = 1;
|
|
} else {
|
|
fl = WEAP_DAEDALUS;
|
|
if (it & WEAP_DAEDALUS)
|
|
usable = 1;
|
|
}
|
|
} if (fl == WEAP_DAEDALUS) {
|
|
if (self.ammo_cells < 5)
|
|
{
|
|
if (it & WEAP_LIGHTNING)
|
|
fl = WEAP_LIGHTNING;
|
|
else am = 1;
|
|
}
|
|
} if (fl == WEAP_LIGHTNING) {
|
|
if (self.ammo_cells < 1)
|
|
am = 1;
|
|
} if (it & WEAP_AIRF && am) {
|
|
fl = WEAP_AIRF;
|
|
am = 0;
|
|
}
|
|
} else if (self.impulse == TF_MEDIKIT) {
|
|
fl = WEAP_MEDIKIT;
|
|
|
|
if (it & WEAP_MEDIKIT)
|
|
usable = 1;
|
|
}
|
|
|
|
self.impulse = 0;
|
|
|
|
// don't have the weapon
|
|
if (!have_weapon || (!(it & fl))) {
|
|
sprint (self, PRINT_HIGH, "no weapon.\n");
|
|
return;
|
|
}
|
|
|
|
// don't have the ammo
|
|
if (am == 1) {
|
|
sprint(self, PRINT_HIGH, "not enough ammo.\n");
|
|
return;
|
|
}
|
|
|
|
// don't have the cells for the cannon
|
|
if (am == 2) {
|
|
sprint(self, PRINT_HIGH, "not enough cells to power assault cannon.\n");
|
|
return;
|
|
}
|
|
|
|
// set weapon, set ammo
|
|
self.current_weapon = fl;
|
|
W_SetCurrentAmmo ();
|
|
W_PrintWeaponMessage();
|
|
|
|
self.StatusRefreshTime = time + 0.1;
|
|
};
|
|
|
|
/*
|
|
============
|
|
CycleWeaponCommand
|
|
|
|
Go to the next weapon with ammo
|
|
============
|
|
*/
|
|
void() CycleWeaponCommand =
|
|
{
|
|
//CH reorder so that they please me
|
|
local float it, am, cont, loopcount;
|
|
|
|
// Some safety code
|
|
if (!self.weaponmodel || self.current_weapon == 0)
|
|
return;
|
|
|
|
if (self.tfstate & TFSTATE_RELOADING)
|
|
return;
|
|
|
|
if (self.tfstate & TFSTATE_CANT_MOVE)
|
|
return;
|
|
|
|
it = self.weapons_carried;
|
|
self.impulse = 0;
|
|
loopcount = 0;
|
|
|
|
while (1) {
|
|
am = 0;
|
|
cont = FALSE;
|
|
|
|
if (self.current_weapon == WEAP_AXE) {
|
|
self.current_weapon = WEAP_SPANNER;
|
|
} else if (self.current_weapon == WEAP_SPANNER) {
|
|
self.current_weapon = WEAP_SHOTGUN;
|
|
if (self.ammo_shells < 1)
|
|
am = 1;
|
|
} else if (self.current_weapon == WEAP_SHOTGUN) {
|
|
self.current_weapon = WEAP_LASER;
|
|
if (self.ammo_nails < 1)
|
|
am = 1;
|
|
} else if (self.current_weapon == WEAP_LASER) {
|
|
self.current_weapon = WEAP_TRANQ;
|
|
if (self.ammo_nails < 1)
|
|
am = 1;
|
|
} else if (self.current_weapon == WEAP_TRANQ) {
|
|
self.current_weapon = WEAP_SNIPER_RIFLE;
|
|
if (self.ammo_shells < 5)
|
|
am = 1;
|
|
} else if (self.current_weapon == WEAP_SNIPER_RIFLE) {
|
|
#ifndef NO_AUTORIFLE
|
|
self.current_weapon = WEAP_AUTO_RIFLE;
|
|
if (self.ammo_shells < 1)
|
|
am = 1;
|
|
} else if (self.current_weapon == WEAP_AUTO_RIFLE) {
|
|
#endif
|
|
self.current_weapon = WEAP_MAUSER;
|
|
if (self.ammo_nails < 1)
|
|
am = 1;
|
|
} else if (self.current_weapon == WEAP_MAUSER) {
|
|
self.current_weapon = WEAP_SUPER_SHOTGUN;
|
|
if (self.ammo_shells < 2)
|
|
am = 1;
|
|
} else if (self.current_weapon == WEAP_SUPER_SHOTGUN) {
|
|
self.current_weapon = WEAP_NAILGUN;
|
|
if (self.ammo_nails < 1)
|
|
am = 1;
|
|
//-
|
|
} else if (self.current_weapon == WEAP_NAILGUN) {
|
|
self.current_weapon = WEAP_SNG;
|
|
if (self.ammo_nails < 2)
|
|
am = 1;
|
|
//-
|
|
} else if (self.current_weapon == WEAP_SNG) {
|
|
self.current_weapon = WEAP_LASERCANNON;
|
|
if (self.ammo_cells < 1)
|
|
am = 1;
|
|
//-
|
|
} else if (self.current_weapon == WEAP_LASERCANNON) { // was WEAP_NAILGUN
|
|
self.current_weapon = WEAP_LIGHT_ASSAULT;
|
|
if (self.ammo_nails < 2)
|
|
am = 1;
|
|
} else if (self.current_weapon == WEAP_LIGHT_ASSAULT) {
|
|
self.current_weapon = WEAP_FLAMETHROWER;
|
|
if (self.ammo_cells < 1)
|
|
am = 1;
|
|
} else if (self.current_weapon == WEAP_FLAMETHROWER) {
|
|
self.current_weapon = WEAP_GRENADE_LAUNCHER;
|
|
self.weaponmode = GL_NORMAL;
|
|
if (self.ammo_rockets < 1)
|
|
am = 1;
|
|
} else if (self.current_weapon == WEAP_GRENADE_LAUNCHER && self.weaponmode == GL_NORMAL) {
|
|
self.current_weapon = WEAP_GRENADE_LAUNCHER;
|
|
self.weaponmode = GL_PIPEBOMB;
|
|
if (self.ammo_rockets < 1)
|
|
am = 1;
|
|
} else if (self.current_weapon == WEAP_GRENADE_LAUNCHER && self.weaponmode == GL_PIPEBOMB) {
|
|
self.current_weapon = WEAP_INCENDIARY;
|
|
if (self.ammo_rockets < 3)
|
|
am = 1;
|
|
} else if (self.current_weapon == WEAP_INCENDIARY) {
|
|
self.current_weapon = WEAP_DAEDALUS;
|
|
if (self.ammo_cells < 1)
|
|
am = 1;
|
|
} else if (self.current_weapon == WEAP_DAEDALUS) {
|
|
self.current_weapon = WEAP_ROCKET_LAUNCHER;
|
|
if (self.ammo_rockets < 1)
|
|
am = 1;
|
|
} else if (self.current_weapon == WEAP_ROCKET_LAUNCHER) {
|
|
if (self.tf_items & NIT_CLUSTER_ROCKETS) {
|
|
if (self.cluster_mode == TRUE) {
|
|
// bprint(PRINT_HIGH, "cluster mode FALSE\n");
|
|
self.cluster_mode = FALSE;
|
|
self.current_weapon = WEAP_ASSAULT_CANNON;
|
|
if (self.ammo_cells < 4)
|
|
am = 1;
|
|
if (self.ammo_shells < 1)
|
|
am = 1;
|
|
} else {
|
|
// bprint(PRINT_HIGH, "cluster mode TRUE\n");
|
|
self.cluster_mode = TRUE;
|
|
self.current_weapon = WEAP_ROCKET_LAUNCHER;
|
|
if (self.ammo_rockets < 1)
|
|
am = 1;
|
|
}
|
|
} else {
|
|
self.current_weapon = WEAP_ASSAULT_CANNON;
|
|
if (self.ammo_cells < 4)
|
|
am = 1;
|
|
if (self.ammo_shells < 1)
|
|
am = 1;
|
|
}
|
|
} else if (self.current_weapon == WEAP_ASSAULT_CANNON) {
|
|
self.current_weapon = WEAP_LIGHTNING;
|
|
if (self.ammo_cells < 1)
|
|
am = 1;
|
|
} else if (self.current_weapon == WEAP_LIGHTNING) {
|
|
self.current_weapon = WEAP_AIRF;
|
|
|
|
if (!(self.weapons_carried & WEAP_AIRF))
|
|
am = 1;
|
|
} else if (self.current_weapon == WEAP_AIRF) {
|
|
self.current_weapon = WEAP_HOOK;
|
|
|
|
if (!(self.weapons_carried & WEAP_HOOK))
|
|
am = 1;
|
|
} else if (self.current_weapon == WEAP_HOOK) {
|
|
self.current_weapon = WEAP_MEDIKIT;
|
|
} else if (self.current_weapon == WEAP_MEDIKIT) {
|
|
self.current_weapon = WEAP_AXE;
|
|
}
|
|
|
|
// Safety precaution
|
|
if (loopcount > 30)
|
|
return;
|
|
|
|
loopcount = loopcount + 1;
|
|
|
|
// check if player actually has the weapon
|
|
// if not, loop again
|
|
if ((self.weapons_carried & self.current_weapon) && (am == 0)) {
|
|
if (self.current_weapon != WEAP_GRENADE_LAUNCHER)
|
|
self.weaponmode = GL_NORMAL; // reset the pipebombs
|
|
|
|
W_SetCurrentAmmo ();
|
|
W_PrintWeaponMessage();
|
|
|
|
self.StatusRefreshTime = time + 0.1;
|
|
return;
|
|
}
|
|
}
|
|
};
|
|
|
|
/*
|
|
============
|
|
ImpulseCommands
|
|
|
|
============
|
|
*/
|
|
void() DeadImpulses;
|
|
|
|
float(float inp) BuildingImpulses =
|
|
{
|
|
if (inp < 10) return TRUE;
|
|
if (inp > TF_CHANGEPC && inp <= TF_CHANGEPC + PC_LASTCLASS) return TRUE;
|
|
if (inp == TF_CHANGECLASS) return TRUE;
|
|
if (inp == TF_HELP_MAP) return TRUE;
|
|
if (inp == TF_SHOWTF) return TRUE;
|
|
if (inp == TF_TEAM_CLASSES) return TRUE;
|
|
if (inp == TF_TEAM_SCORES) return TRUE;
|
|
if (inp == TF_TEAM_LIST) return TRUE;
|
|
if (inp == TF_TEAM_ENUM) return TRUE;
|
|
if (inp == TF_ID) return TRUE;
|
|
if (inp == TF_SELL) return TRUE;
|
|
if (inp == I_CHEAT_ONE) return TRUE;
|
|
if (inp == I_CHEAT_TWO) return TRUE;
|
|
if (inp == I_CHEAT_THREE) return TRUE;
|
|
return FALSE;
|
|
};
|
|
|
|
void() Alias_BitImpulse;
|
|
|
|
void() ImpulseCommands =
|
|
{
|
|
if (self.impulse >= TF_BITSTART && self.impulse <= TF_BITNONE) {
|
|
Alias_BitImpulse ();
|
|
self.impulse = 0;
|
|
return;
|
|
}
|
|
|
|
if (self.impulse == TF_ALIASNEXT) {
|
|
self.impulse = 0;
|
|
self.got_aliases_time = time;
|
|
return;
|
|
}
|
|
|
|
if ((self.impulse== TF_CHANGEPC + PC_CUSTOM || self.impulse==TF_SELL) && custom_mode==2) {
|
|
self.impulse=0;
|
|
sprint(self,PRINT_MEDIUM,"Custom player classes are disabled\n");
|
|
return;
|
|
}
|
|
|
|
if (self.impulse <= TF_CHANGEPC + PC_RANDOM && self.impulse >= TF_CHANGEPC + PC_SCOUT && stock_mode==2) {
|
|
self.impulse=0;
|
|
sprint(self,PRINT_MEDIUM,"Stock player classes are disabled\n");
|
|
return;
|
|
}
|
|
|
|
// local entity te; //- OfN - Unused!
|
|
|
|
//WK Don't allow any commands except menu ones when building a class
|
|
if ( self.playerclass == PC_CUSTOM
|
|
&& (self.done_custom & CUSTOM_BUILDING)) {
|
|
if (!BuildingImpulses(self.impulse)) {
|
|
self.impulse = 0;
|
|
return;
|
|
}
|
|
}
|
|
|
|
//WK Don't allow bastards to do any commands
|
|
if (self.penance_time > time)
|
|
return;
|
|
|
|
if (prematch >= time)
|
|
if (self.impulse > 10 || self.impulse < 1) {
|
|
self.impulse = 0;
|
|
return;
|
|
}
|
|
|
|
/*=====================
|
|
These Impulse commands rely on the use of self.last_impulse. Since they use
|
|
self.impulse for their own purposes, they _must_ be placed before the other
|
|
self.impulse tests, and they _must_ set self.impulse = 0 when they're done.
|
|
=====================*/
|
|
|
|
// TeamFortress Detpack
|
|
if (self.last_impulse == TF_DETPACK && self.impulse)
|
|
TeamFortress_SetDetpack(self.impulse);
|
|
// TeamFortress Scan
|
|
else if (self.last_impulse == TF_SCAN && self.impulse)
|
|
TeamFortress_Scan(self.impulse,FALSE);
|
|
|
|
/*=====================
|
|
The rest of these Impulse Commands don't use self.last_impulse. They _must_
|
|
be placed _after_ the Impulse Commands that do require self.last_impulse
|
|
=====================*/
|
|
|
|
// Catch the 8 key to reprint the classhelp.
|
|
// Remove this when we return the lightning gun
|
|
//CH was on impulse 8, but thats lgun
|
|
if (self.impulse == 9 && self.current_menu != MENU_CLASSHELP && (self.playerclass > 0 && self.playerclass < 11)) {
|
|
self.current_menu = MENU_CLASSHELP;
|
|
self.menu_count = MENU_REFRESH_RATE;
|
|
self.menu_displaytime = 0;
|
|
}
|
|
|
|
// uses the special skill of the player's class
|
|
if (self.impulse == TF_SPECIAL_SKILL)
|
|
UseSpecialSkill();
|
|
|
|
// uses the special skill of the player's class
|
|
if (self.impulse == TF_SKILL)
|
|
UseJobSkill();
|
|
|
|
// No weapon related impulses can be done when setting detpacks or building WK or chaplaning
|
|
if (!self.is_building && !self.is_detpacking && !self.is_feigning && !(self.job & JOB_CHAPLAN && self.job & JOB_ACTIVE)) {
|
|
if ((self.impulse >= 1 && self.impulse < 9) || (self.impulse == TF_MEDIKIT))
|
|
W_ChangeWeapon ();
|
|
|
|
// Grappling Hook
|
|
else if ((self.weapons_carried & WEAP_HOOK) && (self.impulse == HOOK_IMP1 || self.impulse == HOOK_IMP2))
|
|
W_ChangeWeapon ();
|
|
|
|
// Axe
|
|
else if (self.impulse == AXE_IMP)
|
|
W_ChangeWeapon ();
|
|
|
|
// cycle weapon reverse and cycle weapon do the same thing
|
|
// TBD: make CycleWeaponReverseCommand()
|
|
else if (self.impulse == 10 || self.impulse == 12)
|
|
CycleWeaponCommand ();
|
|
|
|
// TeamFortress Reload current weapon
|
|
else if (self.impulse == TF_RELOAD)
|
|
TeamFortress_ReloadCurrentWeapon();
|
|
|
|
// Scanning impulses
|
|
else if (self.impulse == TF_SCAN_10)
|
|
TeamFortress_Scan(10,FALSE);
|
|
else if (self.impulse == TF_SCAN_30)
|
|
TeamFortress_Scan(30,FALSE);
|
|
else if (self.impulse == TF_SCAN_100)
|
|
TeamFortress_Scan(100,FALSE);
|
|
else if (self.impulse == TF_SCAN_ENEMY) //CH
|
|
TeamFortress_Scan(TF_SCAN_ENEMY,FALSE);
|
|
else if (self.impulse == TF_SCAN_FRIENDLY)
|
|
TeamFortress_Scan(TF_SCAN_FRIENDLY,FALSE);
|
|
|
|
// TeamFortress Set Detpack Impulses
|
|
else if (self.impulse == TF_DETPACK_5)
|
|
TeamFortress_SetDetpack(5);
|
|
else if (self.impulse == TF_DETPACK_20)
|
|
TeamFortress_SetDetpack(20);
|
|
else if (self.impulse == TF_DETPACK_50)
|
|
TeamFortress_SetDetpack(50);
|
|
else if (self.impulse == TF_DROP_AMMO) {
|
|
self.current_menu = MENU_DROP;
|
|
self.menu_count = MENU_REFRESH_RATE - 5;
|
|
} else if (self.impulse == TF_DISCARD)
|
|
TeamFortress_Discard();
|
|
//CH
|
|
else if (self.impulse == TF_DROPITEMS)
|
|
TeamFortress_DropItems();
|
|
#ifdef COOP_MODE
|
|
else if (self.impulse == TF_DROPKEY && coop)
|
|
DropKey();
|
|
#endif
|
|
}
|
|
//WK We can toss grens while chaplaining. Readme says so. :p
|
|
if (!self.is_building && !self.is_detpacking && !self.is_feigning) {
|
|
// TeamFortress Prime Grenade Type 1
|
|
if (self.impulse == TF_GRENADE_1)
|
|
TeamFortress_PrimeGrenade();
|
|
// TeamFortress Prime Grenade Type 2
|
|
else if (self.impulse == TF_GRENADE_2)
|
|
TeamFortress_PrimeGrenade();
|
|
}
|
|
|
|
// TeamFortress Inventory
|
|
if (self.impulse == TF_INVENTORY)
|
|
TeamFortress_Inventory();
|
|
|
|
else if (self.impulse == TF_MEDIC_HELPME)
|
|
TeamFortress_SaveMe();
|
|
else if (self.impulse == TF_TAUNT) {
|
|
if (self.last_saveme_sound < time) {
|
|
sound(self, CHAN_WEAPON, "enforcer/sight1.wav", 1, ATTN_NORM);
|
|
self.last_saveme_sound = time + 4;
|
|
}
|
|
} else if (self.impulse == TF_TAUNT2) {
|
|
if (self.last_saveme_sound < time) {
|
|
sound(self, CHAN_WEAPON, "enforcer/sight2.wav", 1, ATTN_NORM);
|
|
self.last_saveme_sound = time + 4;
|
|
}
|
|
} else if (self.impulse == TF_TAUNT3) {
|
|
if (self.last_saveme_sound < time) {
|
|
sound(self, CHAN_WEAPON, "enforcer/sight3.wav", 1, ATTN_NORM);
|
|
self.last_saveme_sound = time + 4;
|
|
}
|
|
} else if (self.impulse == TF_TAUNT4) {
|
|
if (self.last_saveme_sound < time) {
|
|
sound(self, CHAN_WEAPON, "enforcer/sight4.wav", 1, ATTN_NORM);
|
|
self.last_saveme_sound = time + 4;
|
|
}
|
|
} else if (self.impulse == TF_TAUNT5) {
|
|
if (self.last_saveme_sound < time) {
|
|
sound(self, CHAN_WEAPON, "weapons/fith.wav", 1, ATTN_NORM);
|
|
self.last_saveme_sound = time + 4;
|
|
}
|
|
}
|
|
|
|
// TeamFortress Throw Grenade
|
|
else if (self.impulse == TF_GRENADE_T)
|
|
TeamFortress_ThrowGrenade();
|
|
|
|
//else if (self.impulse == BUILD_SENSOR && self.cutf_items & CUTF_SENSOR) // SB build sensor (not enough room on menu...)
|
|
// SBBuildSensor();
|
|
|
|
else if (self.impulse == THROW_DETPACK)
|
|
PrimeC4Det();
|
|
|
|
/*else if (self.impulse == IMPULSE_INTERFACE)
|
|
SBInitiateInterface();*/
|
|
|
|
else if (self.impulse == IMPULSE_HOLO)
|
|
ActivateHolo(self);
|
|
|
|
else if (self.impulse == TF_ID)
|
|
TeamFortress_ID(FALSE);
|
|
|
|
// TeamFortress Detonate Pipebombs (yep, can be done while setting a detpack :)
|
|
else if (self.impulse == TF_PB_DETONATE)
|
|
TeamFortress_DetonatePipebombs();
|
|
|
|
// TeamFortress Stop Setting Detpack
|
|
else if (self.impulse == TF_DETPACK_STOP)
|
|
TeamFortress_DetpackStop(FALSE);
|
|
|
|
// TeamFortress Class Impulses
|
|
//WK - Custom class friendly
|
|
else if (self.impulse == TF_SPY_SPY && (self.cutf_items & CUTF_SPY_KIT))
|
|
TeamFortress_SpyGoUndercover();
|
|
else if (self.impulse == TF_SPY_DIE && (self.cutf_items & CUTF_SPY_KIT))
|
|
TeamFortress_SpyFeignDeath(1); //CH normal feign
|
|
else if (self.impulse == TF_SPY_DIE2 && (self.cutf_items & CUTF_SPY_KIT))
|
|
TeamFortress_SpyFeignDeath(2); //CH special sfeign
|
|
else if (self.impulse == TF_ENGINEER_BUILD &&
|
|
((self.cutf_items & CUTF_SENSOR) || (self.cutf_items & CUTF_SENTRYGUN) || self.tf_items & NIT_TESLA || self.tf_items & NIT_SECURITY_CAMERA || self.tf_items & NIT_TELEPORTER || self.cutf_items & CUTF_FIELDGEN || self.cutf_items & CUTF_DISPENSER))
|
|
TeamFortress_EngineerBuild();
|
|
|
|
// CTF Support Impulses
|
|
else if (self.impulse == FLAG_INFO) {
|
|
if (CTF_Map == TRUE)
|
|
TeamFortress_CTF_FlagInfo();
|
|
else
|
|
TeamFortress_DisplayDetectionItems();
|
|
}
|
|
|
|
#ifdef DEMO_STUFF
|
|
// Camera Impulses
|
|
else if (self.playerclass == PC_UNDEFINED && self.impulse == TF_CAM_TARGET)
|
|
CamLock();
|
|
else if (self.playerclass == PC_UNDEFINED && self.impulse == TF_CAM_ZOOM)
|
|
CamDistLock();
|
|
else if (self.playerclass == PC_UNDEFINED && self.impulse == TF_CAM_VEC)
|
|
CamVecLock();
|
|
else if (self.playerclass == PC_UNDEFINED && self.impulse == TF_CAM_ANGLE)
|
|
CamAngleLock();
|
|
else if (self.playerclass == PC_UNDEFINED && self.impulse == TF_CAM_REVANGLE)
|
|
CamRevAngleLock();
|
|
else if (self.playerclass == PC_UNDEFINED && self.impulse == TF_CAM_PROJECTILE)
|
|
CamProjectileLock();
|
|
else if (self.playerclass == PC_UNDEFINED && self.impulse == TF_CAM_PROJECTILE_Z)
|
|
CamProjectileZoom();
|
|
else if (self.playerclass == PC_UNDEFINED && self.impulse == TF_CAM_OFFSET)
|
|
CamOffset();
|
|
else if (self.playerclass == PC_UNDEFINED && self.impulse == TF_CAM_DROP)
|
|
CamDrop();
|
|
else if (self.playerclass == PC_UNDEFINED && self.impulse == TF_CAM_FADETOBLACK)
|
|
fadetoblack();
|
|
else if (self.playerclass == PC_UNDEFINED && self.impulse == TF_CAM_FADEFROMBLACK)
|
|
fadefromblack();
|
|
else if (self.playerclass == PC_UNDEFINED && self.impulse == TF_CAM_FADETOWHITE)
|
|
fadetowhite();
|
|
else if (self.playerclass == PC_UNDEFINED && self.impulse == TF_CAM_FADEFROMWHITE)
|
|
fadefromwhite();
|
|
#endif
|
|
|
|
else
|
|
DeadImpulses();
|
|
|
|
/*=====================
|
|
TeamFortress Pre-Impulse Commands
|
|
=====================*/
|
|
if (self.impulse == TF_DETPACK)
|
|
self.last_impulse = self.impulse;
|
|
if (self.impulse == TF_SCAN)
|
|
self.last_impulse = self.impulse;
|
|
|
|
self.impulse = 0;
|
|
};
|
|
|
|
/*====================
|
|
DeadImpulses
|
|
|
|
Impulse commands that
|
|
are allowable if the
|
|
player is dead
|
|
====================*/
|
|
void() DeadImpulses =
|
|
{
|
|
if ((self.impulse== TF_CHANGEPC + PC_CUSTOM || self.impulse==TF_SELL) && custom_mode==2) {
|
|
self.impulse=0;
|
|
sprint(self,PRINT_MEDIUM,"Custom player classes are disabled\n");
|
|
return;
|
|
}
|
|
|
|
if (self.impulse <= TF_CHANGEPC + PC_RANDOM && self.impulse >= TF_CHANGEPC + PC_SCOUT && stock_mode==2) {
|
|
self.impulse=0;
|
|
sprint(self,PRINT_MEDIUM,"Stock player classes are disabled\n");
|
|
return;
|
|
}
|
|
#ifdef DEBUG
|
|
if (self.impulse == 69 && infokey(NIL, "allow_debug") == "yes")
|
|
{
|
|
self.demon_blood = MAX_KNIFE_BLOOD;
|
|
self.job = self.job | JOB_BLOODY_KNIFE;
|
|
self.impulse=0;
|
|
return;
|
|
}
|
|
|
|
if (self.impulse == 18 && infokey(NIL, "allow_debug") == "yes")
|
|
{
|
|
sprint(self, PRINT_HIGH, "Your origin is: '", ftos(self.origin_x), " ", ftos(self.origin_y));
|
|
sprint(self, PRINT_HIGH, " ", ftos(self.origin_z), "'\n");
|
|
self.impulse=0;
|
|
return;
|
|
}
|
|
|
|
if (self.impulse == 195 && !debug_target && infokey(NIL, "allow_debug") == "yes")
|
|
debug_target = self;
|
|
#endif
|
|
|
|
//<CH>
|
|
if (self.impulse == I_CHEAT_ONE) {
|
|
I_DID_CHEAT_ONE();
|
|
}
|
|
if (self.impulse == I_CHEAT_TWO) {
|
|
I_DID_CHEAT_TWO();
|
|
}
|
|
if (self.impulse == I_CHEAT_THREE) {
|
|
I_DID_CHEAT_THREE();
|
|
}
|
|
//</CH>
|
|
//WK - Add in TF2.6 support for changeclass
|
|
if (self.impulse == TF_CHANGECLASS) {
|
|
self.current_menu = MENU_CLASS;
|
|
Menu_Class();
|
|
}
|
|
|
|
//WK Have a normal "custom" command reset the "sell" flag
|
|
//Have to do this here, since "upgrade" corrupts self.impulse
|
|
if (self.impulse == TF_CHANGEPC + PC_CUSTOM) //
|
|
self.done_custom = self.done_custom - (self.done_custom & CUSTOM_SELLING);
|
|
|
|
//WK - CustomTF1.6 ability to sell frags
|
|
//Sets a flag saying that we wish to sell
|
|
//And then issues a command to rebuild our class.
|
|
if (self.impulse == TF_SELL) {
|
|
if (bounty) {
|
|
self.done_custom = self.done_custom | CUSTOM_SELLING;
|
|
self.impulse = TF_CHANGEPC + PC_CUSTOM;
|
|
} else {
|
|
sprint (self, PRINT_HIGH, "Sorry, the admin has bounty disabled\n");
|
|
self.impulse = 0;
|
|
return;
|
|
}
|
|
}
|
|
|
|
// TeamFortress Show Toggleflag State
|
|
if (self.impulse == TF_SHOWTF)
|
|
TeamFortress_ShowTF();
|
|
// TeamFortress Show Legal PlayerClasses
|
|
// Toggle autozoom on/off
|
|
else if (self.impulse == TF_AUTOZOOM)
|
|
TeamFortress_AutoZoomToggle();
|
|
else if (self.impulse == TF_SHOWLEGALCLASSES)
|
|
TeamFortress_DisplayLegalClasses();
|
|
|
|
// TeamFortress Change PlayerClass
|
|
else if (self.impulse > TF_CHANGEPC && self.impulse <= (TF_CHANGEPC + PC_CUSTOM) ) {
|
|
//WK Hackish bug-fix for "Custom Custom" teleporting
|
|
if (self.impulse == TF_CHANGEPC + PC_CUSTOM) {
|
|
if ((self.done_custom & (CUSTOM_FINISHED | CUSTOM_BUILDING) && !(self.done_custom & CUSTOM_ON_SPAWN)) || self.playerclass != PC_CUSTOM)
|
|
TeamFortress_ChangeClass();
|
|
} else
|
|
TeamFortress_ChangeClass();
|
|
|
|
} else if (self.impulse == TF_UNCUSTOM && self.done_custom & CUSTOM_ON_SPAWN) {
|
|
self.done_custom = self.done_custom - (self.done_custom & CUSTOM_ON_SPAWN);
|
|
sprint(self, PRINT_HIGH, "You will no longer spawn as a new playerclass.\n");
|
|
}
|
|
// TeamFortress Help Impulses
|
|
else if (self.impulse == TF_LAY)
|
|
custom_lay();
|
|
else if (self.impulse == TF_HELP_MAP)
|
|
TeamFortress_HelpMap();
|
|
else if (self.impulse == TF_STATUS_QUERY)
|
|
TeamFortress_StatusQuery();
|
|
// TeamFortress Team Impulse
|
|
else if (self.impulse == TF_TEAM_1)
|
|
TeamFortress_TeamSet(1);
|
|
else if (self.impulse == TF_TEAM_2)
|
|
TeamFortress_TeamSet(2);
|
|
else if (self.impulse == TF_TEAM_3)
|
|
TeamFortress_TeamSet(3);
|
|
else if (self.impulse == TF_TEAM_4)
|
|
TeamFortress_TeamSet(4);
|
|
else if (self.impulse == TF_TEAM_SCORES)
|
|
TeamFortress_TeamShowScores(0);
|
|
else if (self.impulse == TF_TEAM_CLASSES)
|
|
TeamFortress_TeamShowMemberClasses(self);
|
|
else if (self.impulse == TF_STATUSBAR_ON) {
|
|
self.StatusRefreshTime = time + 0.2;
|
|
self.StatusBarSize = self.StatusBarSize + 1;
|
|
if (self.StatusBarSize > 2)
|
|
self.StatusBarSize = 1;
|
|
} else if (self.impulse == TF_STATUSBAR_OFF) {
|
|
self.StatusRefreshTime = time + 60;
|
|
self.StatusBarSize = 0;
|
|
} else if (self.impulse >= TF_STATUSBAR_RES_START && self.impulse <= TF_STATUSBAR_RES_END) {
|
|
StatusRes(self.impulse - 71);
|
|
}
|
|
// TeamFortress Alias checking
|
|
else if (self.impulse == TF_ALIAS_CHECK) {
|
|
sprint (self, PRINT_HIGH, "Aliases checked.\n");
|
|
self.got_aliases = TRUE;
|
|
self.impulse = 0;
|
|
}
|
|
};
|
|
|
|
/*
|
|
============
|
|
W_WeaponFrame
|
|
|
|
Called every frame so impulse events can be handled as well as possible
|
|
============
|
|
*/
|
|
void() W_WeaponFrame =
|
|
{
|
|
local vector tv;
|
|
|
|
if (!(self.tfstate & TFSTATE_AIMING)) {
|
|
if (self.height > 29 && self.height < 90) {
|
|
self.height = self.height + 80 * frametime;
|
|
if (self.height > 90)
|
|
self.height = 90;
|
|
|
|
TF_zoom(self.height);
|
|
}
|
|
}
|
|
|
|
if (self.current_menu > 0) {
|
|
Player_Menu();
|
|
|
|
if (self.impulse > 0 && self.impulse < 11) {
|
|
Menu_Input(self.impulse);
|
|
|
|
if (self.impulse != 0) {
|
|
if (self.team_no == 0 && teamplay && (self.lives != 0)) {
|
|
Menu_Team_Input(self.impulse);
|
|
}
|
|
/* WK Disable picking normal class menu
|
|
else if (self.playerclass == PC_UNDEFINED && (self.lives != 0))
|
|
{
|
|
Menu_Class_Input(self.impulse);
|
|
}
|
|
*/
|
|
}
|
|
}
|
|
}
|
|
|
|
if (time < self.attack_finished)
|
|
return;
|
|
|
|
// Stop calling this function so much
|
|
if (self.impulse != 0)
|
|
ImpulseCommands ();
|
|
|
|
// Can't fire while setting a detpack or building something
|
|
if ((self.is_building != 0) || (self.is_detpacking != 0) || (self.is_feigning != 0) || (self.is_haxxxoring != 0) || (self.is_toffingadet != 0))
|
|
return;
|
|
|
|
//WK ...or Chaplaning
|
|
if (self.job & JOB_CHAPLAN && self.job & JOB_ACTIVE)
|
|
return;
|
|
|
|
// Check for release
|
|
if (!self.button0 && self.fire_held_down && self.current_weapon == WEAP_ASSAULT_CANNON) {
|
|
self.fire_held_down = FALSE;
|
|
|
|
// Let him/her walk again
|
|
self.tfstate &= ~TFSTATE_ASSAULTCANNON;
|
|
TeamFortress_SetSpeed(self);
|
|
|
|
player_run ();
|
|
}
|
|
|
|
// check for attack
|
|
if (self.button0 && !(self.fire_held_down)) {
|
|
if ((self.current_menu == MENU_CLASSHELP) || (self.current_menu == MENU_CLASSHELP2)) {
|
|
self.current_menu = MENU_REPEATHELP;
|
|
self.menu_count = MENU_REFRESH_RATE;
|
|
Attack_Finished(0.2);
|
|
} else if (self.current_weapon == WEAP_SNIPER_RIFLE) {
|
|
if (self.tfstate & TFSTATE_AIMING) {
|
|
if (!W_CheckNoAmmo())
|
|
self.tfstate &= ~TFSTATE_AIMING;
|
|
else if (self.height > 30) {
|
|
self.height = self.height - 5;
|
|
TF_zoom(self.height);
|
|
}
|
|
} else {
|
|
tv = self.velocity;
|
|
tv_z = 0;
|
|
|
|
if (vlen(tv) <= WEAP_SNIPER_RIFLE_MAX_MOVE && W_CheckNoAmmo()) {
|
|
// create the laser sight
|
|
SniperSight_Create(0);
|
|
self.heat = 30; // damage done
|
|
// GR no, charging sniper rifle makes no sense, idiot
|
|
#ifdef WEINER_SNIPER
|
|
self.heat = 80;
|
|
#endif
|
|
|
|
self.height = 90;
|
|
self.tfstate = self.tfstate | TFSTATE_AIMING;
|
|
|
|
TeamFortress_SetSpeed(self);
|
|
}
|
|
}
|
|
} else if (self.current_weapon == WEAP_ROCKET_LAUNCHER && self.tf_items & NIT_RL_LASER_SIGHT) {
|
|
if (!(self.tfstate & TFSTATE_RL_LASER)) {
|
|
// create the laser sight
|
|
SniperSight_Create(1);
|
|
self.tfstate = self.tfstate | TFSTATE_RL_LASER;
|
|
|
|
SuperDamageSound ();
|
|
W_Attack ();
|
|
|
|
TeamFortress_SetSpeed(self);
|
|
}
|
|
} else {
|
|
SuperDamageSound ();
|
|
W_Attack ();
|
|
}
|
|
} else if (self.playerclass == PC_UNDEFINED) {
|
|
self.weaponmode = 0;
|
|
} else if (self.tfstate & TFSTATE_AIMING) {
|
|
W_Attack();
|
|
self.tfstate = self.tfstate - TFSTATE_AIMING;
|
|
TeamFortress_SetSpeed(self);
|
|
self.heat = 0;
|
|
} else if (self.tfstate & TFSTATE_RL_LASER) {
|
|
//CH the dot should remove itself
|
|
Attack_Finished(0.5); //Finish it up
|
|
self.tfstate = self.tfstate - TFSTATE_RL_LASER;
|
|
TeamFortress_SetSpeed(self);
|
|
|
|
player_run ();
|
|
}
|
|
};
|
|
|
|
/*
|
|
================
|
|
T_DaedalusTouch
|
|
Combination conc, krac and RL
|
|
================
|
|
*/
|
|
#define BOUNCE 120
|
|
void() T_DaedalusTouch =
|
|
{
|
|
local float points = 0; //XXX false +ve
|
|
local vector org;
|
|
local entity head;
|
|
|
|
head = findradius(self.origin, BOUNCE);
|
|
while (head) {
|
|
//Conc
|
|
if (head.takedamage && !IsBuilding(head)) {
|
|
org = head.origin + (head.mins + head.maxs)*0.5;
|
|
points = 0.5*vlen (org - self.origin);
|
|
if (points < 0) points = 0;
|
|
points = BOUNCE - points;
|
|
if (points > 0) {
|
|
head.velocity = org - self.origin;
|
|
head.velocity = head.velocity * (points / 20);
|
|
if (head.cutf_items & CUTF_GYMNAST)
|
|
head.velocity = head.velocity * (points / 20);
|
|
if (head.classname != "player" && head.flags & FL_ONGROUND)
|
|
head.flags = head.flags - FL_ONGROUND;
|
|
}
|
|
}
|
|
if (head.takedamage) {
|
|
//Krac
|
|
if (IsBuilding(head))
|
|
TF_T_Damage (head, self, self.owner, 100, TF_TD_NOTTEAM, TF_TD_FIRE);
|
|
//Rocket Launcher (w/ armor stripping)
|
|
else {
|
|
if (head.armorvalue > 35) head.armorvalue = head.armorvalue - 35;
|
|
else head.armorvalue = 0;
|
|
deathmsg = DMSG_DAEDALUS;
|
|
TF_T_Damage (head, self, self.owner, points / 8, TF_TD_NOTTEAM, TF_TD_FIRE);
|
|
}
|
|
}
|
|
head = head.chain;
|
|
}
|
|
|
|
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);
|
|
|
|
if (other.classname == "force_field") //- OfN - Makes field explosion b4 removing it
|
|
FieldExplosion(other,self.origin,self);
|
|
|
|
dremove(self);
|
|
};
|
|
|
|
/*
|
|
================
|
|
W_FireDaedalus (WK)
|
|
================
|
|
*/
|
|
void() W_FireDaedalus =
|
|
{
|
|
sound (self, CHAN_WEAPON, "weapons/grenade.wav", 1, ATTN_NORM);
|
|
KickPlayer(-2, self);
|
|
|
|
self.currentammo = self.ammo_cells = self.ammo_cells - 5;
|
|
if (self.ammo_cells < 0) {
|
|
self.ammo_cells = 0;
|
|
return;
|
|
}
|
|
|
|
SuperDamageSound();
|
|
|
|
newmis = spawn ();
|
|
newmis.owner = self;
|
|
newmis.movetype = MOVETYPE_FLYMISSILE;
|
|
newmis.classname = "rocket"; // OfN - for airfist pushing
|
|
newmis.solid = SOLID_BBOX;
|
|
|
|
makevectors (self.v_angle);
|
|
newmis.velocity = v_forward;
|
|
newmis.velocity = newmis.velocity * 500;
|
|
newmis.avelocity = '600 600 0';
|
|
newmis.angles = vectoangles(newmis.velocity);
|
|
newmis.skin = 1;
|
|
//CHANGEME Replace this with s_explode sprite. :)
|
|
setmodel (newmis, "progs/flare.mdl");
|
|
setsize (newmis, '0 0 0', '0 0 0');
|
|
setorigin (newmis, self.origin + v_forward*8 + '0 0 16');
|
|
|
|
newmis.touch = T_DaedalusTouch;
|
|
newmis.nextthink = time + 4;
|
|
newmis.think = SUB_Remove;
|
|
};
|
|
|
|
/* Sponsored by SB-1 Tech
|
|
**
|
|
** W_FireMauser!
|
|
** Fires the mauser spy thing which is likely to have a name change
|
|
** so nyeh
|
|
*/
|
|
|
|
void() W_FireMauser =
|
|
{
|
|
local vector org;
|
|
local vector source;
|
|
|
|
self.currentammo = self.ammo_nails = self.ammo_nails - 1;
|
|
|
|
if (self.ammo_nails < 0) {
|
|
self.ammo_nails = 0;
|
|
return;
|
|
}
|
|
|
|
KickPlayer(-2, self);
|
|
|
|
makevectors(self.v_angle);
|
|
source = self.origin + '0 0 16';
|
|
|
|
traceline (source, source + v_forward*100, FALSE, self);
|
|
|
|
if (trace_fraction == 1.0)
|
|
return;
|
|
|
|
org = trace_endpos - v_forward*4;
|
|
|
|
if (trace_ent.takedamage) {
|
|
trace_ent.axhitme = 1;
|
|
|
|
SpawnBlood (org, 20);
|
|
deathmsg = DMSG_MAUSER;
|
|
SuperDamageSound();
|
|
TF_T_Damage(trace_ent, self, self, 20, TF_TD_NOTTEAM, TF_TD_SHOT);
|
|
if (trace_ent.classname == "player")
|
|
{
|
|
TF_T_Damage(trace_ent, self, self, 50+200*random(), TF_TD_NOTTEAM + TF_TD_IGNOREARMOUR, TF_TD_OTHER);
|
|
sprint(trace_ent, PRINT_HIGH, "Your nerves scream as poison floods your veins...\n");
|
|
}
|
|
} else {
|
|
if (trace_ent.classname == "force_field") { //- OfN - Makes field explosion
|
|
FieldExplosion(trace_ent,trace_endpos,trace_ent);
|
|
PutFieldWork(trace_ent);
|
|
}
|
|
}
|
|
|
|
newmis = spawn();
|
|
newmis.owner = self;
|
|
newmis.heat = 0;
|
|
newmis.think = MauserRecoilThink;
|
|
newmis.nextthink = time + 0.1;
|
|
};
|
|
|
|
void () MauserRecoilThink =
|
|
{
|
|
|
|
self.owner.weaponframe = self.heat = self.heat + 1;
|
|
|
|
if (self.heat >= 6) {
|
|
self.owner.weaponframe = 0;
|
|
dremove(self);
|
|
return;
|
|
}
|
|
|
|
self.nextthink = time + 0.1;
|
|
|
|
};
|
|
|
|
/*
|
|
========
|
|
SuperDamageSound
|
|
|
|
Plays sound if needed
|
|
========
|
|
*/
|
|
void() SuperDamageSound =
|
|
{
|
|
if (self.super_damage_finished > time) {
|
|
if (self.super_sound < time) {
|
|
self.super_sound = time + 1;
|
|
sound (self, CHAN_BODY, "items/damage3.wav", 1, ATTN_NORM);
|
|
}
|
|
} else if (self.aura == AURA_POWER) {
|
|
if (self.super_sound < time)
|
|
{
|
|
self.super_sound = time + 1;
|
|
sound (self, CHAN_BODY, "auras/aura2.wav", 1, ATTN_NORM);
|
|
}
|
|
} else if (self.aura == AURA_HASTE) {
|
|
if (self.super_sound < time) {
|
|
self.super_sound = time + 1;
|
|
sound (self, CHAN_BODY, "auras/aura4.wav", 1, ATTN_NORM);
|
|
}
|
|
}
|
|
return;
|
|
};
|
|
|
|
/*===========================================================
|
|
// laser cannon stuff (from hipnotic)
|
|
//==========================================================
|
|
*/
|
|
void() HIP_LaserTouch =
|
|
{
|
|
local vector org;
|
|
local vector spot1,spot2;
|
|
local vector oldvel;
|
|
// local float r;
|
|
|
|
self.owner = NIL;
|
|
self.cnt = self.cnt + 1;
|
|
if (pointcontents(self.origin) == CONTENTS_SKY) {
|
|
remove(self);
|
|
return;
|
|
}
|
|
oldvel = normalize(self.neworigin);
|
|
spot1 = self.origin - (16*oldvel);
|
|
spot2 = self.origin + (16*oldvel);
|
|
traceline (spot1, spot2, FALSE, self); // see through other monsters
|
|
self.origin = trace_endpos;
|
|
|
|
org = self.origin;
|
|
|
|
if (other.health) {
|
|
// other.deathtype = "hiplaser";
|
|
if (self.owner == other)
|
|
self.dmg = self.dmg / 2;
|
|
|
|
spawn_touchblood (self.dmg);
|
|
|
|
//- OfN laser hurts monsters a lot
|
|
if (IsMonsterNonArmy(other))
|
|
self.dmg = self.dmg * 2;
|
|
|
|
//- scrags get rapped by the laser
|
|
if (other.classname == "monster_wizard")
|
|
self.dmg = self.dmg * 2; // this is *4
|
|
|
|
deathmsg = DMSG_LASERCANNON;
|
|
TF_T_Damage (other, self, self.demon_one, self.dmg, TF_TD_NOTTEAM, TF_TD_ELECTRICITY);
|
|
} else if ((self.cnt >= 3) || (random() < 0.15)) {
|
|
WriteByte (MSG_MULTICAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_MULTICAST, TE_GUNSHOT);
|
|
WriteByte (MSG_MULTICAST, 3);
|
|
WriteCoord (MSG_MULTICAST, org_x);
|
|
WriteCoord (MSG_MULTICAST, org_y);
|
|
WriteCoord (MSG_MULTICAST, org_z);
|
|
multicast (org, MULTICAST_PHS);
|
|
} else {
|
|
// self.dmg = 0.66 * self.dmg;
|
|
self.dmg = 0.9 * self.dmg;
|
|
// self.speed = 0.95 * self.speed;
|
|
self.velocity = oldvel+(2*trace_plane_normal);
|
|
self.velocity = normalize(self.velocity);
|
|
self.velocity = self.speed * self.velocity;
|
|
self.neworigin = self.velocity;
|
|
if (self.flags & FL_ONGROUND)
|
|
self.flags = self.flags - FL_ONGROUND;
|
|
// r = random();
|
|
// self.attack_finished = time + 7; //ofn - restart its life of 7 seconds
|
|
|
|
if (other.classname == "force_field") //- OfN - Makes field explosion
|
|
FieldExplosion(other,self.origin,self);
|
|
sound (self, CHAN_WEAPON, "weapons/laserric.wav", 1, ATTN_STATIC);
|
|
return;
|
|
}
|
|
|
|
if (other.classname == "force_field") //- OfN - Makes field explosion b4 removing it
|
|
FieldExplosion(other,self.origin,self);
|
|
|
|
sound (self, CHAN_WEAPON, "enforcer/enfstop.wav", 1, ATTN_STATIC);
|
|
dremove(self);
|
|
};
|
|
|
|
void() HIP_LaserThink =
|
|
{
|
|
// local float delta; //unused?
|
|
|
|
if (time>self.attack_finished) {
|
|
remove(self);
|
|
return;
|
|
}
|
|
if (self.flags & FL_ONGROUND)
|
|
self.flags = self.flags - FL_ONGROUND;
|
|
self.velocity = self.neworigin;
|
|
self.angles = vectoangles(self.velocity);
|
|
self.nextthink = time+0.15;
|
|
};
|
|
|
|
void(vector org, vector vec, float light) HIP_LaunchLaser =
|
|
{
|
|
// sound (self ,CHAN_WEAPON, "weapons/shotgn2.wav", 1, ATTN_NORM);
|
|
sound (self ,CHAN_WEAPON, "weapons/laserg.wav", 1, ATTN_NORM);
|
|
|
|
vec = normalize(vec);
|
|
|
|
newmis = spawn();
|
|
newmis.owner = self;
|
|
newmis.classname = "hiplaser";
|
|
newmis.demon_one = self; // was "lastvictim" field of hipnotic // OfN needed because .owner is removed later
|
|
newmis.movetype = MOVETYPE_FLYMISSILE;
|
|
newmis.solid = SOLID_BBOX;
|
|
// if (light)
|
|
// newmis.effects = EF_DIMLIGHT;
|
|
|
|
setmodel (newmis, "progs/lasrspik.mdl");
|
|
setsize (newmis, '0 0 0', '0 0 0');
|
|
|
|
setorigin (newmis, org);
|
|
|
|
newmis.speed = 1000;
|
|
newmis.dmg = LASER_CANNON_DMG;
|
|
newmis.velocity = vec * newmis.speed;
|
|
newmis.neworigin = newmis.velocity; // was last_velocity field on hypnotic source code
|
|
newmis.angles = vectoangles(newmis.velocity);
|
|
newmis.avelocity = '0 0 400';
|
|
|
|
newmis.nextthink = time;
|
|
newmis.attack_finished = time + 3.5; //was 3.5//OfN- they last for 7 seconds now
|
|
newmis.think = HIP_LaserThink;
|
|
newmis.touch = HIP_LaserTouch;
|
|
newmis.count = 0;
|
|
};
|
|
|
|
/*
|
|
=================
|
|
HIP_FireLaser
|
|
=================
|
|
*/
|
|
void(float stat) HIP_FireLaser =
|
|
{
|
|
local vector org;
|
|
local vector dir;
|
|
local vector out;
|
|
local float ofs;
|
|
local float aofs;
|
|
|
|
if (!self.button0) {
|
|
player_run ();
|
|
return;
|
|
}
|
|
if (self.ammo_cells < 1) {
|
|
self.weapon = W_BestWeapon ();
|
|
W_SetCurrentAmmo ();
|
|
return;
|
|
}
|
|
|
|
SuperDamageSound();
|
|
muzzleflash();
|
|
makevectors (self.v_angle);
|
|
|
|
ofs = 6;
|
|
out = v_forward;
|
|
out_z = 0;
|
|
out = normalize(out);
|
|
org = self.origin + ((12-ofs) * v_up) + (12*out);
|
|
// org = self.origin + (1*v_forward);
|
|
dir = aim (self, 1000);
|
|
aofs = ofs * 0.707;
|
|
if (stat == 0) {
|
|
self.currentammo = self.ammo_cells = self.ammo_cells - 1;
|
|
// org = org + (aofs*v_right);
|
|
org = org - (aofs*v_up);
|
|
HIP_LaunchLaser(org, dir, 0);
|
|
// org = org - (2*aofs*v_right);
|
|
// HIP_LaunchLaser(org, dir, 0);
|
|
} else if (stat == 1) {
|
|
self.currentammo = self.ammo_cells = self.ammo_cells - 1;
|
|
org = org + (ofs*v_up);
|
|
// if (random()<0.1) {
|
|
// HIP_LaunchLaser(org, dir, 1);
|
|
// newmis.dmg = 25;
|
|
// } else
|
|
HIP_LaunchLaser(org, dir, 0);
|
|
}
|
|
msg_entity = self;
|
|
WriteByte (MSG_ONE, SVC_SMALLKICK);
|
|
|
|
self.reload_laser_cannon = self.reload_laser_cannon + 1;
|
|
|
|
if (CheckForReload() == TRUE)
|
|
return;
|
|
};
|