mirror of
https://github.com/nzp-team/quakec.git
synced 2025-01-21 17:01:16 +00:00
2565 lines
58 KiB
C++
2565 lines
58 KiB
C++
/*
|
|
server/weapons/weapon_core.qc
|
|
|
|
weapon logic
|
|
|
|
Copyright (C) 2021-2022 NZ:P Team
|
|
|
|
This program is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU General Public License
|
|
as published by the Free Software Foundation; either version 2
|
|
of the License, or (at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
See the GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to:
|
|
|
|
Free Software Foundation, Inc.
|
|
59 Temple Place - Suite 330
|
|
Boston, MA 02111-1307, USA
|
|
|
|
|
|
*/
|
|
|
|
void() W_PutOut;
|
|
void() GrenadeExplode;
|
|
|
|
void() ReturnWeaponModel =
|
|
{
|
|
self.weaponmodel = GetWeaponModel(self.weapon, 0);
|
|
|
|
if (IsDualWeapon(self.weapon)) {
|
|
self.weapon2model = GetLeftWeaponModel(self.weapon);
|
|
} else if (self.weapon == W_KAR_SCOPE || self.weapon == W_HEADCRACKER) {
|
|
self.weapon2model = "models/weapons/kar/v_karscope.mdl";
|
|
UpdateV2model(self.weapon2model, 0);
|
|
}
|
|
|
|
UpdateVmodel(self.weaponmodel, GetWepSkin(self.weapon));
|
|
UpdateV2model(self.weapon2model, GetWepSkin(self.weapon));
|
|
}
|
|
|
|
.float scopetime;
|
|
|
|
void() W_AimIn =
|
|
{
|
|
if (self.weapon == W_TESLA || self.weapon == W_DG3)
|
|
return;
|
|
|
|
if (IsDualWeapon(self.weapon) ||
|
|
self.zoom ||
|
|
self.reload_delay > time ||
|
|
self.sprinting ||
|
|
self.new_anim_stop ||
|
|
self.weapon == W_BK) {
|
|
return;
|
|
}
|
|
|
|
#ifndef PC
|
|
self.ADS_Offset = GetWeaponADSOfs_PSP(self.weapon);
|
|
#endif
|
|
|
|
if (self.weapon == W_KAR_SCOPE || self.weapon == W_PTRS ||
|
|
self.weapon == W_HEADCRACKER || self.weapon == W_PENETRATOR) {
|
|
SetUpdate(self, UT_ZOOM2, 0, 0, 0);
|
|
#ifdef NX
|
|
self.zoom = 1;
|
|
#else
|
|
self.zoom = 2;
|
|
#endif
|
|
self.scopetime = time + 0.3;
|
|
} else {
|
|
self.zoom = 1;
|
|
}
|
|
|
|
if (!self.downed)
|
|
playaim();
|
|
|
|
}
|
|
|
|
void() W_AimOut =
|
|
{
|
|
if (!self.zoom ||
|
|
self.new_anim_stop ||
|
|
self.sprinting) {
|
|
return;
|
|
}
|
|
|
|
if (self.weapon == W_KAR_SCOPE || self.weapon == W_PTRS ||
|
|
self.weapon == W_HEADCRACKER || self.weapon == W_PENETRATOR) {
|
|
ReturnWeaponModel();
|
|
}
|
|
|
|
#ifndef PC
|
|
self.zoom = 0;
|
|
#endif
|
|
|
|
if (!self.downed)
|
|
playout();
|
|
}
|
|
|
|
void() W_SprintStop =
|
|
{
|
|
if (self.sprinting)
|
|
{
|
|
self.sprint_stop_time = time;
|
|
self.sprint_duration = self.sprint_timer;
|
|
}
|
|
|
|
float startframe = GetFrame(self.weapon,SPRINT_OUT_START);
|
|
float endframe = GetFrame(self.weapon,SPRINT_OUT_END);
|
|
string modelname = GetWeaponModel(self.weapon, 0);
|
|
|
|
if (self.isBuying || !self.sprinting)
|
|
return;
|
|
|
|
//TEMPORARY
|
|
#ifndef PC
|
|
self.zoom = 0;
|
|
#endif
|
|
|
|
Set_W_Frame (startframe, endframe, 0, 0, SPRINT, SUB_Null, modelname, false, S_BOTH); // BUG IS HERE
|
|
|
|
self.sprinting = 0;
|
|
self.into_sprint = 0;
|
|
self.reload_delay2 = self.fire_delay2 = self.reload_delay = self.fire_delay = 0;
|
|
|
|
if (self.velocity)
|
|
playwalk();
|
|
}
|
|
|
|
|
|
void W_SprintStart () {
|
|
if (self.speed_penalty_time > time)
|
|
return;
|
|
|
|
self.sprint_start_time = time;
|
|
|
|
if (self.sprint_rest_time > sprint_max_time)
|
|
self.sprint_duration = 0.0;
|
|
else
|
|
self.sprint_duration -= self.sprint_rest_time;
|
|
|
|
#ifdef NX
|
|
if (!self.sprintflag) {
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
if (self.fire_delay > time ||
|
|
self.fire_delay2 > time ||
|
|
self.new_anim_stop ||
|
|
self.new_anim2_stop ||
|
|
self.isBuying ||
|
|
self.downed ||
|
|
!(self.flags & FL_ONGROUND) ||
|
|
self.sprint_delay > time) {
|
|
return;
|
|
}
|
|
|
|
float startframe = GetFrame(self.weapon,SPRINT_IN_START);
|
|
float endframe = GetFrame(self.weapon,SPRINT_IN_END);
|
|
string modelname = GetWeaponModel(self.weapon, 0);
|
|
|
|
#ifndef PC
|
|
self.zoom = 3;
|
|
#endif
|
|
|
|
if (startframe || endframe) {
|
|
Set_W_Frame (startframe, endframe, 0, 0, SPRINT, ContinueRun, modelname, false, S_BOTH);
|
|
}
|
|
|
|
self.sprint_delay = time + 1;
|
|
self.sprinting = true;
|
|
self.reload_delay2 = self.fire_delay2 = self.reload_delay = self.fire_delay = 0;
|
|
}
|
|
|
|
void() W_Switch =
|
|
{
|
|
if (self.secondaryweapon && self.secondaryweapon != 0 && !self.new_anim_stop && !other.button7)
|
|
{
|
|
float tempf,tempf1,tempf2,tempf3,tempf4;
|
|
float startframe;
|
|
float endframe;
|
|
string modelname;
|
|
|
|
// un-zoom camera
|
|
self.zoom = false;
|
|
|
|
// fix fire rate exploit
|
|
self.reload_delay2 = self.fire_delay2 = self.reload_delay = self.fire_delay = true;
|
|
|
|
// just do normal weapon swapping if we don't have mule..
|
|
if (!(self.perks & P_MULE) || !self.thirdweapon) {
|
|
tempf = self.currentammo;
|
|
self.currentammo = self.secondaryammo;
|
|
self.secondaryammo = tempf;
|
|
|
|
tempf1 = self.currentmag;
|
|
self.currentmag = self.secondarymag;
|
|
self.secondarymag = tempf1;
|
|
|
|
tempf1 = self.currentmag2;
|
|
self.currentmag2 = self.secondarymag2;
|
|
self.secondarymag2 = tempf1;
|
|
|
|
tempf2 = self.weapon;
|
|
self.weapon = self.secondaryweapon;
|
|
self.secondaryweapon = tempf2;
|
|
|
|
tempf3 = self.weaponskin;
|
|
self.weaponskin = self.secondaryweaponskin;
|
|
self.secondaryweaponskin = tempf3;
|
|
} else {
|
|
// store primary weapon
|
|
tempf = self.weapon;
|
|
tempf1 = self.currentmag;
|
|
tempf2 = self.currentmag2;
|
|
tempf3 = self.currentammo;
|
|
tempf4 = self.weaponskin;
|
|
|
|
// set primary weapon to third weapon
|
|
self.weapon = self.thirdweapon;
|
|
self.currentmag = self.thirdmag;
|
|
self.currentmag2 = self.thirdmag2;
|
|
self.currentammo = self.thirdammo;
|
|
self.weaponskin = self.thirdweaponskin;
|
|
|
|
// set third weapon to secondary weapon
|
|
self.thirdweapon = self.secondaryweapon;
|
|
self.thirdmag = self.secondarymag;
|
|
self.thirdmag2 = self.secondarymag2;
|
|
self.thirdammo = self.secondaryammo;
|
|
self.thirdweaponskin = self.secondaryweaponskin;
|
|
|
|
// set secondary weapon to primary weapon
|
|
self.secondaryweapon = tempf;
|
|
self.secondarymag = tempf1;
|
|
self.secondarymag2 = tempf2;
|
|
self.secondaryammo = tempf3;
|
|
self.secondaryweaponskin = tempf4;
|
|
}
|
|
|
|
startframe = GetFrame(self.weapon,TAKE_OUT_START);
|
|
endframe = GetFrame(self.weapon,TAKE_OUT_END);
|
|
modelname = GetWeaponModel(self.weapon, 0);
|
|
|
|
SwitchWeapon(self.weapon);
|
|
|
|
if (IsDualWeapon(self.weapon)) {
|
|
self.weapon2model = GetLeftWeaponModel(self.weapon);
|
|
} else if (self.weapon == W_KAR_SCOPE || self.weapon == W_HEADCRACKER)
|
|
self.weapon2model = "models/weapons/kar/v_karscope.mdl";
|
|
else
|
|
self.weapon2model = "";
|
|
|
|
UpdateV2model(self.weapon2model, 0);
|
|
|
|
// TEMP
|
|
if (self.weapon == W_M2 || self.weapon == W_FIW)
|
|
UpdateVmodel(modelname, GetWepSkin(self.weapon));
|
|
|
|
|
|
Set_W_Frame (startframe, endframe, 0, 0, 0, SUB_Null, modelname, false, S_BOTH);//FIXME
|
|
|
|
self.reload_delay2 = self.fire_delay2 = self.reload_delay = self.fire_delay = 0;
|
|
|
|
|
|
#ifndef PC
|
|
self.Weapon_Name = GetWeaponName(self.weapon);
|
|
self.Flash_Offset = GetWeaponFlash_Offset(self.weapon);
|
|
self.Flash_Size = GetWeaponFlash_Size(self.weapon);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
void() W_PutOut =
|
|
{
|
|
if (self.downed) {
|
|
return;
|
|
}
|
|
|
|
W_AimOut();
|
|
self.weaponnum = !self.weaponnum;
|
|
|
|
if (self.secondaryweapon && !self.new_anim_stop)
|
|
{
|
|
float startframe;
|
|
float endframe;
|
|
string modelname;
|
|
startframe = GetFrame(self.weapon,PUT_OUT_START);
|
|
endframe = GetFrame(self.weapon,PUT_OUT_END);
|
|
modelname = GetWeaponModel(self.weapon, 0);
|
|
|
|
Set_W_Frame (startframe, endframe, (endframe - startframe)/10, 0, SWITCHWEP, W_Switch, modelname, false, S_BOTH);//FIXME
|
|
self.reload_delay2 = self.fire_delay2 = self.reload_delay = self.fire_delay = 0;
|
|
}
|
|
}
|
|
|
|
void() W_TakeOut =
|
|
{
|
|
W_AimOut();
|
|
|
|
float startframe;
|
|
float endframe;
|
|
string modelname;
|
|
startframe = GetFrame(self.weapon,TAKE_OUT_START);
|
|
endframe = GetFrame(self.weapon,TAKE_OUT_END);
|
|
modelname = GetWeaponModel(self.weapon, 0);
|
|
|
|
self.isBuying = false;
|
|
|
|
Set_W_Frame (startframe, endframe, (endframe - startframe)/10, 0, SWITCHWEP, SUB_Null, modelname, false, S_BOTH);//FIXME
|
|
self.reload_delay2 = self.fire_delay2 = self.reload_delay = self.fire_delay = 0;
|
|
}
|
|
|
|
float(entity who) hasWeapon =
|
|
{
|
|
if (other.weapon || other.secondaryweapon || other.thirdweapon)
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
|
|
//RELOAD
|
|
|
|
void(float side) W_Give_Ammo =
|
|
{
|
|
float ammo_shot, max_mag, loadammo;
|
|
|
|
max_mag = getWeaponMag(self.weapon);
|
|
|
|
if (side == S_LEFT) {
|
|
ammo_shot = max_mag - self.currentmag2;
|
|
} else {
|
|
ammo_shot = max_mag - self.currentmag;
|
|
}
|
|
if (ammo_shot < self.currentammo)
|
|
{
|
|
self.currentammo = self.currentammo - ammo_shot;
|
|
|
|
loadammo = max_mag;
|
|
}
|
|
else
|
|
{
|
|
loadammo = self.currentmag + self.currentammo;
|
|
self.currentammo = 0;
|
|
}
|
|
|
|
if (side == S_LEFT) {
|
|
self.currentmag2 = loadammo;
|
|
} else {
|
|
self.currentmag = loadammo;
|
|
}
|
|
};
|
|
|
|
void () W_LoadAmmo;
|
|
void() ContinueReload = //Special reloads
|
|
{
|
|
if (self.new_anim_stop)
|
|
return;
|
|
|
|
if (self.weapon == W_GUT && self.currentmag == 10)
|
|
return;
|
|
|
|
float delay = 0;
|
|
string modelname = GetWeaponModel(self.weapon, 0);
|
|
float startframe = 0;
|
|
float endframe = 0;
|
|
void(optional float t) endanimfunc = SUB_Null;
|
|
|
|
if (self.currentmag >= getWeaponMag(self.weapon) || !self.currentammo || self.reloadinterupted) {
|
|
if (self.weapon == W_KAR_SCOPE || self.weapon == W_HEADCRACKER)
|
|
{
|
|
delay = 1;
|
|
startframe = 24;
|
|
endframe = 28;
|
|
} else if (self.weapon == W_TRENCH || self.weapon == W_GUT) {
|
|
delay = 0.5;
|
|
startframe = 24;
|
|
endframe = 26;
|
|
endanimfunc = W_LoadAmmo;
|
|
}
|
|
self.reloadinterupted = FALSE;
|
|
} else if (self.currentmag < getWeaponMag(self.weapon)) {
|
|
if (self.weapon == W_BROWNING || self.weapon == W_ACCELERATOR) {
|
|
delay = 4;
|
|
startframe = 45;
|
|
endframe = 77;
|
|
endanimfunc = W_Give_Ammo;
|
|
} else if (self.weapon == W_KAR_SCOPE || self.weapon == W_HEADCRACKER) {
|
|
self.currentmag++;
|
|
self.currentammo = self.currentammo - 1;
|
|
delay = 0.8;
|
|
startframe = 19;
|
|
endframe = 24;
|
|
endanimfunc = ContinueReload;
|
|
} else if (self.weapon == W_TRENCH || self.weapon == W_GUT) {
|
|
if (self.weapon == W_GUT && self.currentammo >= 2 && self.currentmag < 9) {
|
|
self.currentmag = self.currentmag + 2;
|
|
self.currentammo = self.currentammo - 2;
|
|
} else {
|
|
self.currentmag++;
|
|
self.currentammo = self.currentammo - 1;
|
|
}
|
|
delay = 0.5;
|
|
startframe = 18;
|
|
endframe = 23;
|
|
endanimfunc = ContinueReload;
|
|
}
|
|
}
|
|
|
|
if (delay) {
|
|
if (self.perks & P_SPEED) {
|
|
delay *= 0.5;
|
|
}
|
|
self.reload_delay = time + delay;
|
|
Set_W_Frame (startframe, endframe, delay, 0, RELOAD, endanimfunc, modelname, false, S_RIGHT);
|
|
}
|
|
}
|
|
|
|
void(float side) W_AdvanceAnim =
|
|
{
|
|
float startframe, endframe, delay, reloadcancelframe;
|
|
string modelname;
|
|
|
|
startframe = GetFrame(self.weapon, RELOAD_START);
|
|
endframe = GetFrame(self.weapon, RELOAD_END);
|
|
delay = getWeaponDelay(self.weapon, RELOAD);
|
|
reloadcancelframe = GetFrame(self.weapon, RELOAD_CANCEL);
|
|
|
|
modelname = GetWeaponModel(self.weapon, 0);
|
|
|
|
self.reload_delay = self.reload_delay2 = delay + time;
|
|
|
|
Set_W_Frame (startframe, endframe, delay, reloadcancelframe, RELOAD, W_Give_Ammo, modelname, false, side);
|
|
}
|
|
|
|
void(float side) W_Reload =
|
|
{
|
|
if (self.weapon == W_M2 || self.weapon == W_FIW)
|
|
return;
|
|
|
|
if (side == S_BOTH) {
|
|
W_Reload(S_RIGHT);
|
|
if (IsDualWeapon(self.weapon)) {
|
|
W_Reload(S_LEFT);
|
|
}
|
|
return;
|
|
}
|
|
if (side == S_RIGHT &&
|
|
(self.reload_delay > time ||
|
|
self.new_anim_stop ||
|
|
!self.currentammo ||
|
|
self.currentmag >= getWeaponMag(self.weapon))){
|
|
return;
|
|
}
|
|
if (side == S_LEFT &&
|
|
(self.reload_delay2 > time ||
|
|
self.new_anim2_stop ||
|
|
!self.currentammo ||
|
|
self.currentmag2 >= getWeaponMag(self.weapon))) {
|
|
return;
|
|
}
|
|
|
|
//TEMPORARY
|
|
#ifdef PC
|
|
self.viewzoom = 1;
|
|
#endif
|
|
self.zoom = false;
|
|
|
|
W_AimOut();
|
|
W_SprintStop();
|
|
|
|
float endframe, startframe, reloadcancelframe, delay;
|
|
string modelname = "";
|
|
|
|
if (side == S_RIGHT) {
|
|
modelname = GetWeaponModel(self.weapon, 0);
|
|
} else {
|
|
if (IsDualWeapon(self.weapon)) {
|
|
modelname = GetLeftWeaponModel(self.weapon);
|
|
}
|
|
}
|
|
|
|
if (self.currentammo)
|
|
{
|
|
playreload();
|
|
|
|
startframe = GetFrame(self.weapon,RELOAD_START);
|
|
endframe = GetFrame(self.weapon,RELOAD_END);
|
|
reloadcancelframe = GetFrame(self.weapon,RELOAD_CANCEL);
|
|
delay = getWeaponDelay(self.weapon,RELOAD);
|
|
void(optional float t) endanimfunc = SUB_Null;
|
|
|
|
// TODO: Make browning use new reload status frames
|
|
if (self.weapon == W_BROWNING || self.weapon == W_ACCELERATOR) {
|
|
if (!self.currentmag) {
|
|
startframe = 26;
|
|
endframe = 77;
|
|
reloadcancelframe = 45;
|
|
delay = 6;
|
|
endanimfunc = W_Give_Ammo;
|
|
} else if (self.currentmag > 0) {
|
|
startframe = 4;
|
|
endframe = 25;
|
|
reloadcancelframe = 0;
|
|
delay = 3.5;
|
|
endanimfunc = ContinueReload;
|
|
}
|
|
} else if (self.weapon == W_KAR_SCOPE || self.weapon == W_HEADCRACKER ){
|
|
startframe = 14;
|
|
endframe = 18;
|
|
reloadcancelframe = 0;
|
|
delay = 0.8;
|
|
endanimfunc = ContinueReload;
|
|
}
|
|
// Check for special reload type
|
|
else if (GetFrame(self.weapon, RELOAD_EMPTY_START) != 0) {
|
|
//reloadcancelframe = GetFrame(self.weapon, RELOAD_CANCEL);
|
|
if (self.currentmag > 0) {
|
|
startframe = GetFrame(self.weapon, RELOAD_PART_START);
|
|
endframe = GetFrame(self.weapon, RELOAD_PART_END);
|
|
delay = getWeaponDelay(self.weapon, RELOAD_PAR);
|
|
} else {
|
|
startframe = GetFrame(self.weapon, RELOAD_EMPTY_START);
|
|
endframe = GetFrame(self.weapon, RELOAD_EMPTY_END);
|
|
delay = getWeaponDelay(self.weapon, RELOAD_EMP);
|
|
}
|
|
endanimfunc = W_AdvanceAnim;
|
|
}
|
|
else if (self.weapon == W_TRENCH || self.weapon == W_GUT) {
|
|
if (self.currentmag == 0) {
|
|
self.NeedLoad = true;
|
|
}
|
|
startframe = 14;
|
|
endframe = 17;
|
|
reloadcancelframe = 0;
|
|
delay = 0.8;
|
|
endanimfunc = ContinueReload;
|
|
} else {
|
|
endanimfunc = W_Give_Ammo;
|
|
}
|
|
|
|
if (self.perks & P_SPEED)
|
|
delay = delay*0.5;
|
|
|
|
if (side == S_RIGHT) {
|
|
self.reload_delay = delay + time;
|
|
} else {
|
|
self.reload_delay2 = delay + time;
|
|
}
|
|
|
|
Set_W_Frame (startframe, endframe, delay, reloadcancelframe, RELOAD, endanimfunc, modelname, false, side);
|
|
}
|
|
|
|
if (self.weapon != W_TRENCH) {
|
|
self.NeedLoad = false;
|
|
}
|
|
};
|
|
|
|
void () W_LoadAmmoDone =
|
|
{
|
|
self.NeedLoad = false;
|
|
}
|
|
void () W_LoadAmmo =
|
|
{
|
|
if (!self.NeedLoad)
|
|
return;
|
|
if (!self.currentmag)
|
|
{
|
|
W_Reload(S_BOTH);
|
|
return;
|
|
}
|
|
|
|
string modelname;
|
|
float endframe, startframe, reloadcancelframe, delay;
|
|
|
|
modelname = GetWeaponModel(self.weapon, 0);
|
|
startframe = GetFrame(self.weapon,RELOAD_START);
|
|
endframe = GetFrame(self.weapon,RELOAD_END);
|
|
reloadcancelframe = GetFrame(self.weapon,RELOAD_CANCEL);
|
|
delay = getWeaponDelay(self.weapon,RELOAD);
|
|
void(optional float t) endanimfunc = SUB_Null;
|
|
|
|
if (self.weapon == W_TRENCH || self.weapon == W_GUT)
|
|
{
|
|
startframe = 4;
|
|
endframe = 14;
|
|
reloadcancelframe = 12;
|
|
delay = 1;
|
|
endanimfunc = W_LoadAmmoDone;
|
|
}
|
|
else if (self.weapon == W_KAR || self.weapon == W_ARMAGEDDON || self.weapon == W_KAR_SCOPE || self.weapon == W_HEADCRACKER)
|
|
{
|
|
startframe = 4;
|
|
endframe = 13;
|
|
reloadcancelframe = 9;
|
|
delay = 1.2;
|
|
endanimfunc = W_LoadAmmoDone;
|
|
}
|
|
|
|
if (delay) {
|
|
|
|
if (self.perks & P_DOUBLE)
|
|
delay *= 0.66;
|
|
|
|
Set_W_Frame (startframe, endframe, delay, reloadcancelframe, FIRE, W_LoadAmmoDone, modelname, false, S_RIGHT);
|
|
self.fire_delay = delay + time;
|
|
}
|
|
}
|
|
|
|
void () CheckReload =
|
|
{
|
|
if (!self.currentmag2 && IsDualWeapon(self.weapon)) {
|
|
W_Reload(S_LEFT);
|
|
}
|
|
if (!self.currentmag && self.currentammo) {
|
|
W_Reload(S_RIGHT);
|
|
}
|
|
}
|
|
|
|
/*
|
|
================
|
|
SpawnBlood
|
|
================
|
|
*/
|
|
void(vector org, vector vel, float damage) SpawnBlood =
|
|
{
|
|
local entity f;
|
|
|
|
#ifdef PC
|
|
WriteByte(MSG_MULTICAST, SVC_CGAMEPACKET);
|
|
WriteByte(MSG_MULTICAST, EVENT_BLOOD);
|
|
WriteCoord(MSG_MULTICAST, org_x);
|
|
WriteCoord(MSG_MULTICAST, org_y);
|
|
WriteCoord(MSG_MULTICAST, org_z);
|
|
msg_entity = self;
|
|
multicast('0 0 0', MULTICAST_ONE);
|
|
#else
|
|
particle (org, vel*0.1, 73, damage*2);
|
|
#endif
|
|
|
|
f = find (world, classname, "player");
|
|
f.hitcount = f.hitcount + 1;
|
|
};
|
|
|
|
void(vector where) spawn_gibs = {
|
|
#ifndef PC
|
|
particle(where, where*0.1, 225, 1);
|
|
#else
|
|
WriteByte(MSG_MULTICAST, SVC_CGAMEPACKET);
|
|
WriteByte(MSG_MULTICAST, EVENT_BLOOD);
|
|
WriteCoord(MSG_MULTICAST, where_x);
|
|
WriteCoord(MSG_MULTICAST, where_y);
|
|
WriteCoord(MSG_MULTICAST, where_z);
|
|
msg_entity = self;
|
|
multicast('0 0 0', MULTICAST_ONE);
|
|
#endif
|
|
}
|
|
|
|
|
|
void Parse_Damage () = // DO NOT TOUCH
|
|
{
|
|
entity ent;
|
|
float total_dmg;
|
|
|
|
entity body_ent;
|
|
float head_hit;
|
|
|
|
//warn
|
|
head_hit = 0;
|
|
|
|
ent = findfloat (world, washit, 1);
|
|
|
|
while (ent) {
|
|
if (ent.classname != "radio" && ent.classname != "explosive_barrel") {
|
|
|
|
if (ent.classname == "ai_zombie_head")
|
|
head_hit = 1;
|
|
|
|
if (ent.classname != "ai_zombie" && ent.classname != "ai_dog") //limb
|
|
body_ent = ent.owner;
|
|
else
|
|
body_ent = ent;
|
|
|
|
total_dmg = body_ent.hitamount;
|
|
|
|
if (body_ent.head.washit) {
|
|
total_dmg += body_ent.head.hitamount;
|
|
body_ent.head.health -= body_ent.head.hitamount;
|
|
head_hit = 1;
|
|
|
|
// Head = Gone!
|
|
if (body_ent.head.health <= 0) {
|
|
// Head has already been blown off, bro
|
|
if (!body_ent.head.deadflag)
|
|
return;
|
|
|
|
vector where;
|
|
|
|
// If our weapon is capable, blow the head off
|
|
if (getWeaponDamage(self.weapon) >= 100) {
|
|
makevectors(body_ent.head.owner.angles);
|
|
where = body_ent.head.owner.origin + (body_ent.head.view_ofs_x * v_right) + (body_ent.head.view_ofs_y * v_forward) + (body_ent.head.view_ofs_z * v_up);
|
|
spawn_gibs(where);
|
|
|
|
// Should the Zombie try and walk on?
|
|
if (random() <= 0.01) {
|
|
body_ent.bleedingtime = time + 2;
|
|
} else {
|
|
// Nah, he dead.
|
|
total_dmg *= 5000;
|
|
}
|
|
|
|
body_ent.head.deadflag = false;
|
|
body_ent.head.solid = SOLID_NOT;
|
|
setmodel(body_ent.head,"");
|
|
body_ent.head.frame = 0;
|
|
|
|
body_ent.usedent = self;
|
|
body_ent.bleedingtime = time + 2;
|
|
#ifdef HANDHELD
|
|
updateLimb (body_ent.head.owner, 0, world);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
body_ent.head.washit = 0;
|
|
body_ent.head.hitamount = 0;
|
|
}
|
|
if (body_ent.larm.washit) {
|
|
total_dmg = total_dmg + body_ent.larm.hitamount;
|
|
|
|
body_ent.larm.health = body_ent.larm.health - body_ent.larm.hitamount;
|
|
if (body_ent.larm.health <= 0 && self.weapon != W_COLT && body_ent.larm.owner.rarm.deadflag && body_ent.larm.deadflag)
|
|
{
|
|
|
|
makevectors(body_ent.larm.owner.angles);
|
|
where = body_ent.larm.owner.origin + (body_ent.larm.view_ofs_x * v_right) + (body_ent.larm.view_ofs_y * v_forward) + (body_ent.larm.view_ofs_z * v_up);
|
|
spawn_gibs(where);
|
|
|
|
body_ent.larm.deadflag = false;
|
|
body_ent.larm.solid = SOLID_NOT;
|
|
setmodel(body_ent.larm,"");
|
|
body_ent.larm.frame = 0;
|
|
#ifdef HANDHELD
|
|
updateLimb (body_ent.larm.owner, 1, world);
|
|
#endif
|
|
}
|
|
|
|
body_ent.larm.washit = 0;
|
|
body_ent.larm.hitamount = 0;
|
|
}
|
|
if (body_ent.rarm.washit) {
|
|
total_dmg = total_dmg + body_ent.rarm.hitamount;
|
|
|
|
body_ent.rarm.health = body_ent.rarm.health - body_ent.rarm.hitamount;
|
|
if (body_ent.rarm.health <= 0 && self.weapon != W_COLT && body_ent.rarm.owner.larm.deadflag && body_ent.rarm.deadflag) {
|
|
|
|
makevectors(body_ent.rarm.owner.angles);
|
|
where = body_ent.rarm.owner.origin + (body_ent.rarm.view_ofs_x * v_right) + (body_ent.rarm.view_ofs_y * v_forward) + (body_ent.rarm.view_ofs_z * v_up);
|
|
spawn_gibs(where);
|
|
|
|
body_ent.rarm.deadflag = false;
|
|
body_ent.rarm.solid = SOLID_NOT;
|
|
setmodel(body_ent.rarm,"");
|
|
body_ent.rarm.frame = 0;
|
|
#ifdef HANDHELD
|
|
updateLimb (body_ent.rarm.owner, 2, world);
|
|
#endif
|
|
}
|
|
|
|
body_ent.rarm.washit = 0;
|
|
body_ent.rarm.hitamount = 0;
|
|
}
|
|
|
|
if (instakill_finished) {
|
|
if(body_ent.head.deadflag) {
|
|
body_ent.head.deadflag = false;
|
|
body_ent.head.solid = SOLID_NOT;
|
|
setmodel(body_ent.head,"");
|
|
body_ent.head.frame = 0;
|
|
}
|
|
}
|
|
|
|
if (head_hit)
|
|
DamageHandler(body_ent,self, total_dmg, S_HEADSHOT);
|
|
else
|
|
DamageHandler(body_ent,self, total_dmg, S_NORMAL);
|
|
|
|
#ifndef PC
|
|
body_ent.washit = 0;
|
|
body_ent.hitamount = 0;
|
|
#else
|
|
ent.washit = 0;
|
|
ent.hitamount = 0;
|
|
#endif
|
|
|
|
}
|
|
ent = findfloat (ent, washit, 1);
|
|
}
|
|
}
|
|
|
|
void() LungeKnifeHit =
|
|
{
|
|
vector source,org;
|
|
float f_damage;
|
|
entity hit_ent;
|
|
|
|
makevectors (self.v_angle);
|
|
source = self.origin + self.view_ofs;
|
|
|
|
if (self.weapon == W_BK)
|
|
traceline (source, source + v_forward*80*1.2, 0, self); // naievil -- added some extra length
|
|
else
|
|
traceline (source, source + v_forward*64*1.2, 0, self); // naievil -- ^^
|
|
|
|
org = trace_endpos - v_forward*3; // naievil -- changed to factor of 3 from 4
|
|
|
|
|
|
if (trace_ent.takedamage)
|
|
{
|
|
if(trace_ent.classname == "item_radio" || trace_ent.classname == "teddy_spawn")
|
|
{
|
|
entity oldself;
|
|
oldself = self;
|
|
self = trace_ent;
|
|
self.th_die();
|
|
self = oldself;
|
|
return;
|
|
}
|
|
sound (self, CHAN_WEAPON, "sounds/weapons/knife/knife_hitbod.wav", 1, ATTN_NORM);
|
|
|
|
if (self.bowie && self.weapon == W_BK)
|
|
f_damage = 1200;
|
|
else if (self.bowie)
|
|
f_damage = 1000;
|
|
else if (self.weapon == W_BK)
|
|
f_damage = 500;
|
|
else
|
|
f_damage = 150;
|
|
|
|
if (trace_ent.classname == "ai_zombie_head")
|
|
{
|
|
hit_ent = trace_ent.owner;
|
|
f_damage = f_damage*1.5;
|
|
}
|
|
else if (trace_ent.classname == "ai_zombie_larm")
|
|
{
|
|
hit_ent = trace_ent.owner;
|
|
}
|
|
else if (trace_ent.classname == "ai_zombie_rarm")
|
|
{
|
|
hit_ent = trace_ent.owner;
|
|
}
|
|
else
|
|
{
|
|
hit_ent = trace_ent;
|
|
}
|
|
|
|
if (hit_ent.takedamage)
|
|
{
|
|
org = source + (v_forward * 20);
|
|
SpawnBlood (org, org, 50);
|
|
//hit = 1;
|
|
}
|
|
|
|
DamageHandler (hit_ent, self, f_damage, S_KNIFE);
|
|
}
|
|
}
|
|
|
|
void(float damage, vector dir, vector org, vector plane, entity hit_ent, float side) TraceAttack =
|
|
{
|
|
vector vel;
|
|
float f_damage = 0;
|
|
|
|
vel = normalize(dir);
|
|
vel = vel + 2*plane;
|
|
vel = vel * 200;
|
|
|
|
if (hit_ent.takedamage) {
|
|
if (trace_fraction >= 1) {
|
|
return;
|
|
}
|
|
if(hit_ent.classname == "item_radio" || hit_ent.classname == "teddy_spawn")
|
|
{
|
|
entity oldself;
|
|
oldself = self;
|
|
self = hit_ent;
|
|
self.th_die();
|
|
self = oldself;
|
|
return;
|
|
} else if (hit_ent.classname == "explosive_barrel") {
|
|
hit_ent.enemy = self;
|
|
oldself = self;
|
|
self = hit_ent;
|
|
self.health = self.health - damage;
|
|
|
|
Barrel_Hit();
|
|
|
|
self = oldself;
|
|
return;
|
|
} else if (hit_ent.classname == "door" || hit_ent.classname == "door_nzp") {
|
|
local entity old;
|
|
old = self;
|
|
self = hit_ent;
|
|
self.health -= damage;
|
|
|
|
if (self.health < 0)
|
|
door_use();
|
|
self = old;
|
|
return;
|
|
}
|
|
|
|
SpawnBlood (org, vel, damage*10);
|
|
|
|
switch(hit_ent.classname) {
|
|
case "ai_zombie_head":
|
|
f_damage = damage*getWeaponMultiplier(self.weapon, HEAD_X);
|
|
break;
|
|
case "ai_zombie_larm":
|
|
case "ai_zombie_rarm":
|
|
f_damage = damage*getWeaponMultiplier(self.weapon, LIMBS_X);
|
|
break;
|
|
case "ai_zombie":
|
|
if (trace_endpos_z < hit_ent.origin_z)
|
|
f_damage = damage*getWeaponMultiplier(self.weapon, LOWER_TORSO_X);
|
|
else
|
|
f_damage = damage*getWeaponMultiplier(self.weapon, UPPER_TORSO_X);
|
|
break;
|
|
default:
|
|
f_damage = damage;
|
|
break;
|
|
}
|
|
|
|
hit_ent.washit = 1;
|
|
hit_ent.hitamount = hit_ent.hitamount + f_damage;
|
|
|
|
SetUpdate(self, UT_HM, 0, 0, 0);
|
|
|
|
#ifndef PC
|
|
msg_entity = self;
|
|
WriteByte(MSG_ONE, SVC_HITMARK);
|
|
#endif
|
|
} else {
|
|
|
|
#ifndef PC
|
|
WriteByte (MSG_BROADCAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_BROADCAST, TE_GUNSHOT);
|
|
WriteCoord (MSG_BROADCAST, org_x);
|
|
WriteCoord (MSG_BROADCAST, org_y);
|
|
WriteCoord (MSG_BROADCAST, org_z);
|
|
#endif
|
|
}
|
|
|
|
#ifdef PC
|
|
WriteByte(MSG_MULTICAST, SVC_CGAMEPACKET);
|
|
WriteByte(MSG_MULTICAST, EVENT_PISTOLFIRE);
|
|
WriteEntity(MSG_MULTICAST, self);
|
|
WriteFloat(MSG_MULTICAST, side);
|
|
WriteCoord(MSG_MULTICAST, org_x);
|
|
WriteCoord(MSG_MULTICAST, org_y);
|
|
WriteCoord(MSG_MULTICAST, org_z);
|
|
WriteCoord(MSG_MULTICAST, plane_x);
|
|
WriteCoord(MSG_MULTICAST, plane_y);
|
|
WriteCoord(MSG_MULTICAST, plane_z);
|
|
WriteEntity(MSG_MULTICAST, hit_ent);
|
|
multicast(trace_endpos, MULTICAST_PHS);
|
|
#endif
|
|
|
|
};
|
|
|
|
void (float shotcount, float sprd, float Damage, float side) FireTrace =
|
|
{
|
|
float count, r, ru, penetration_times, do_break;
|
|
vector dir, src, trace_start_org;
|
|
entity hitent;
|
|
|
|
count = 0;
|
|
|
|
makevectors(self.v_angle);
|
|
|
|
src = self.origin + self.view_ofs;
|
|
|
|
#ifndef PC
|
|
CL_SendWeaponFire();
|
|
#endif
|
|
|
|
while (count < shotcount)
|
|
{
|
|
dir = aim (self,0);
|
|
r = random() * sprd*10;
|
|
if (random() < 0.5)
|
|
r = r*-1;
|
|
ru = random() * sprd*10;
|
|
if (random() < 0.5)
|
|
ru = ru*-1;
|
|
|
|
dir = dir*2048;
|
|
dir = dir + v_right*r + v_up*ru;
|
|
trace_start_org = src;
|
|
trace_ent = self;
|
|
do_break = 0;
|
|
penetration_times = 0;
|
|
hitent = world;
|
|
while (!do_break && random() < getWeaponPenetration(self.weapon, penetration_times))
|
|
{
|
|
while (1)
|
|
{
|
|
traceline (trace_start_org, trace_start_org + dir, MOVE_HITMODEL, trace_ent);
|
|
|
|
if (hitent == world)
|
|
break;
|
|
/*if (hitent.larm == trace_ent || hitent.rarm == trace_ent || hitent.head == trace_ent)
|
|
{
|
|
hitent = trace_ent;
|
|
trace_start_org = trace_endpos + normalize(dir)*10;
|
|
//break;
|
|
}*/
|
|
else
|
|
break;
|
|
}
|
|
|
|
if (trace_fraction != 1.0)
|
|
TraceAttack (Damage, dir, trace_endpos, trace_plane_normal, trace_ent, side);
|
|
trace_start_org = trace_endpos;
|
|
penetration_times++;
|
|
if (!trace_ent.takedamage)
|
|
do_break = 1;
|
|
|
|
if (trace_ent.classname == "ai_zombie_head" || trace_ent.classname == "ai_zombie_larm" || trace_ent.classname == "ai_zombie_rarm")
|
|
hitent = trace_ent.owner;
|
|
else
|
|
hitent = trace_ent;
|
|
|
|
}
|
|
hitent = world;
|
|
|
|
count++;
|
|
}
|
|
if (self.weapon != W_TESLA)
|
|
Parse_Damage();
|
|
|
|
}
|
|
|
|
void(float side) W_FireGrenade =
|
|
{
|
|
entity nade = spawn();
|
|
nade.owner = self;
|
|
nade.movetype = MOVETYPE_BOUNCE;
|
|
nade.solid = SOLID_BBOX;
|
|
nade.classname = "explosiveNade";
|
|
|
|
makevectors (self.v_angle);
|
|
|
|
nade.velocity = v_forward*1000;
|
|
nade.avelocity = '50 -50 50';
|
|
|
|
//centerprint(self, vtos(nade.angles));
|
|
|
|
nade.touch = GrenadeExplode;
|
|
setsize(nade, '0 0 0', '0 0 0');
|
|
nade.origin = self.origin + self.view_ofs;
|
|
|
|
setorigin(nade, nade.origin);
|
|
}
|
|
|
|
//
|
|
// RAY GUN - modified from patch <3
|
|
//
|
|
|
|
void() RayBulletExplode =
|
|
{
|
|
DamgageExplode (self, self.owner, 1800, 1800, 64);//we want A SMALL radius of explosion
|
|
|
|
// MOTO - FTE explosions are.. large, let's flash instead.
|
|
|
|
#ifdef PC
|
|
te_smallflash(self.origin);
|
|
#else
|
|
WriteByte (MSG_BROADCAST, SVC_TEMPENTITY);
|
|
#ifdef HANDHELD
|
|
if (self.effects & EF_RAYGREEN)
|
|
WriteByte (MSG_BROADCAST, TE_RAYSPLASHGREEN);
|
|
else
|
|
WriteByte (MSG_BROADCAST, TE_RAYSPLASHRED);
|
|
#else
|
|
WriteByte (MSG_BROADCAST, TE_EXPLOSION);
|
|
#endif
|
|
WriteCoord (MSG_BROADCAST, self.origin_x);
|
|
WriteCoord (MSG_BROADCAST, self.origin_y);
|
|
WriteCoord (MSG_BROADCAST, self.origin_z);
|
|
#endif
|
|
|
|
SUB_Remove ();
|
|
};
|
|
|
|
void() velocity_increase_ray =
|
|
{
|
|
if (!other.solid || other.solid == SOLID_TRIGGER)
|
|
if (other != world)
|
|
return;
|
|
|
|
if (other == self.owner)
|
|
return;
|
|
|
|
if (other.solid == SOLID_NOT) {
|
|
RayBulletExplode();
|
|
}
|
|
|
|
self.velocity = self.velocity*1.5;
|
|
|
|
RayBulletExplode();
|
|
};
|
|
|
|
void() W_FireRay =
|
|
{
|
|
local entity porter;
|
|
|
|
porter = spawn();
|
|
porter.owner = self;
|
|
porter.movetype = MOVETYPE_FLY;
|
|
porter.solid = SOLID_BBOX;
|
|
|
|
makevectors(self.v_angle);
|
|
|
|
porter.velocity = v_forward*2000;
|
|
porter.avelocity = '0 0 0';
|
|
|
|
porter.angles = vectoangles(porter.velocity);
|
|
porter.angles_z += (porter.angles_z + 180 < 360)? 180 : -180;
|
|
porter.angles = vectoangles(v_forward);
|
|
|
|
porter.v_angle = '0 0 200';
|
|
|
|
porter.touch = velocity_increase_ray;
|
|
setmodel(porter, "models/misc/raybeam.mdl");
|
|
setsize(porter, '0 0 0', '0 0 0');
|
|
|
|
|
|
if (self.weapon == W_PORTER) {
|
|
#ifdef HANDHELD
|
|
porter.effects = EF_RAYRED;
|
|
#else
|
|
porter.effects = EF_RED;
|
|
#endif
|
|
} else {
|
|
#ifdef HANDHELD
|
|
porter.effects = EF_RAYGREEN;
|
|
#else
|
|
porter.effects = EF_GREEN;
|
|
#endif
|
|
}
|
|
|
|
|
|
porter.origin = self.origin + self.view_ofs;
|
|
porter.origin += v_forward * 0;
|
|
|
|
setorigin(porter, porter.origin);
|
|
|
|
self.animend = ReturnWeaponModel;
|
|
self.callfuncat = 0;
|
|
}
|
|
|
|
// Remove zombie, add money, request clone spawn.
|
|
void(entity hit_ent) LightningHit =
|
|
{
|
|
local entity tempe;
|
|
|
|
// 50 points for waffe kills
|
|
addmoney(self, 50, true);
|
|
|
|
// set lib models to null
|
|
// (if we remove them RelinkZombies will cry)
|
|
if (hit_ent.head)
|
|
setmodel(hit_ent.head, "");
|
|
if (hit_ent.larm)
|
|
setmodel(hit_ent.larm, "");
|
|
if (hit_ent.rarm)
|
|
setmodel(hit_ent.rarm, "");
|
|
|
|
// window hop fix
|
|
if (self.state == 1 || self.hop_step != 0) {
|
|
self.state = self.hop_step = 0;
|
|
}
|
|
|
|
// change classname so it can't be re-targted
|
|
hit_ent.classname = "wunder";
|
|
|
|
// own it!
|
|
hit_ent.owner = self;
|
|
|
|
// we're the start of a chain
|
|
hit_ent.teslacount = 1;
|
|
|
|
// play our wunder-ful anim
|
|
tempe = self;
|
|
self = hit_ent;
|
|
self.th_diewunder();
|
|
self = tempe;
|
|
|
|
// kill
|
|
self.kills += 1;
|
|
};
|
|
|
|
// Initial Zombie impact, creds to Naievil/UP
|
|
void(vector p1, vector p2, entity from, float damage) LightningDamage =
|
|
{
|
|
local entity e1, e2;
|
|
local vector f;
|
|
entity hit_ent;
|
|
|
|
f = p2 - p1;
|
|
normalize (f);
|
|
f_x = 0 - f_y;
|
|
f_y = f_x;
|
|
f_z = 0;
|
|
f = f*16;
|
|
|
|
e1 = e2 = world;
|
|
|
|
//traceline (p1, p2, false, self);
|
|
|
|
hit_ent = trace_ent;
|
|
|
|
if (hit_ent == world)
|
|
return;
|
|
|
|
if (hit_ent.classname == "ai_zombie_head" || hit_ent.classname == "ai_zombie_larm" || hit_ent.classname == "ai_zombie_rarm")
|
|
hit_ent = hit_ent.owner;
|
|
|
|
if (trace_ent.takedamage)
|
|
{
|
|
LightningHit (hit_ent);
|
|
}
|
|
e1 = trace_ent;
|
|
|
|
traceline (p1 + f, p2 + f, FALSE, self);
|
|
if (trace_ent != e1 && trace_ent.takedamage)
|
|
{
|
|
LightningHit (hit_ent);
|
|
}
|
|
e2 = trace_ent;
|
|
|
|
|
|
traceline (p1 - f, p2 - f, FALSE, self);
|
|
if (trace_ent != e1 && trace_ent != e2 && trace_ent.takedamage)
|
|
{
|
|
LightningHit (hit_ent);
|
|
}
|
|
};
|
|
|
|
// Fire lightning out of the gun
|
|
void() W_FireTesla =
|
|
{
|
|
local vector source;
|
|
|
|
makevectors (self.v_angle);
|
|
source = self.origin + self.view_ofs;
|
|
|
|
FireTrace(1, 0, 0, 0);
|
|
|
|
#ifdef PC
|
|
te_lightning2(self, source, trace_endpos);
|
|
#endif
|
|
#ifdef HANDHELD
|
|
WriteByte (MSG_BROADCAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_BROADCAST, TE_LIGHTNING2);
|
|
WriteEntity (MSG_BROADCAST, self);
|
|
WriteCoord (MSG_BROADCAST, source_x);
|
|
WriteCoord (MSG_BROADCAST, source_y);
|
|
WriteCoord (MSG_BROADCAST, source_z);
|
|
WriteCoord (MSG_BROADCAST, trace_endpos_x);
|
|
WriteCoord (MSG_BROADCAST, trace_endpos_y);
|
|
WriteCoord (MSG_BROADCAST, trace_endpos_z);
|
|
#endif
|
|
|
|
LightningDamage (self.origin, trace_endpos + v_forward*4, self, 30);
|
|
}
|
|
|
|
void() Flame_LingerThink =
|
|
{
|
|
#ifdef PC
|
|
te_flamejet(self.origin, v_up*8, 5);
|
|
#else
|
|
particle (self.origin, v_up*8, 0, 0);
|
|
#endif
|
|
|
|
if (self.ltime < time)
|
|
remove(self);
|
|
|
|
self.nextthink = time + 0.1;
|
|
}
|
|
|
|
// spawn linger effect/entity
|
|
void(vector org, entity whodunit) Flame_Linger =
|
|
{
|
|
local entity linger = spawn();
|
|
|
|
setorigin(linger, org);
|
|
|
|
linger.owner = whodunit.owner;
|
|
linger.think = Flame_LingerThink;
|
|
linger.nextthink = time + 0.1;
|
|
linger.ltime = time + 5;
|
|
}
|
|
|
|
void() Flame_Touch =
|
|
{
|
|
if (other == world) {
|
|
Flame_Linger(self.origin, self);
|
|
remove(self);
|
|
} else {
|
|
if (other.onfire || other.owner.onfire)
|
|
return;
|
|
if (other.classname == "ai_zombie_head" || other.classname == "ai_zombie_larm"
|
|
|| other.classname == "ai_zombie_rarm") {
|
|
other.owner.onfire = true;
|
|
other.owner.ltime = time + 2;
|
|
addmoney(self.owner, 10, true);
|
|
} else if (other.classname == "ai_zombie" || other.classname == "ai_dog") {
|
|
other.onfire = true;
|
|
other.firer = self.owner;
|
|
other.ltime = time + 2;
|
|
addmoney(self.owner, 10, true);
|
|
}
|
|
}
|
|
}
|
|
|
|
// fallen flame touches something
|
|
void() Flame_DropCatch =
|
|
{
|
|
if (other == world) {
|
|
Flame_Linger(self.origin + '0 0 5', self);
|
|
remove(self);
|
|
}
|
|
}
|
|
|
|
// drop didn't make it to the ground in time to spread
|
|
void() Flame_DropDissapate =
|
|
{
|
|
remove(self);
|
|
}
|
|
|
|
// rare chance some falls
|
|
void() Flame_Drop =
|
|
{
|
|
local entity drop = spawn();
|
|
|
|
drop.owner = self.owner;
|
|
|
|
setmodel(drop, "models/sprites/flame.spr");
|
|
setsize(drop, '0 0 0', '0 0 0');
|
|
setorigin(drop, self.origin);
|
|
|
|
drop.touch = Flame_DropCatch;
|
|
drop.think = Flame_DropDissapate;
|
|
drop.nextthink = time + 0.8;
|
|
drop.solid = SOLID_TRIGGER;
|
|
drop.movetype = MOVETYPE_BOUNCE;
|
|
|
|
drop.velocity = v_up*-55;
|
|
}
|
|
|
|
// increase frames, remove/drop
|
|
void() Flame_Buildup =
|
|
{
|
|
self.frame++;
|
|
|
|
if (self.frame > 11) {
|
|
if (random() > 0.75)
|
|
Flame_Drop();
|
|
remove(self);
|
|
}
|
|
|
|
self.nextthink = time + 0.05;
|
|
}
|
|
|
|
void() W_FireM2 =
|
|
{
|
|
|
|
local entity flamespr;
|
|
local vector rand_angle;
|
|
|
|
flamespr = spawn();
|
|
flamespr.owner = self;
|
|
flamespr.movetype = MOVETYPE_FLYMISSILE;
|
|
flamespr.solid = SOLID_TRIGGER;
|
|
flamespr.classname = "flamethrower_sprite";
|
|
flamespr.state = 0;
|
|
flamespr.finaldest = self.v_angle;
|
|
|
|
// calc some randomness
|
|
rand_angle = self.v_angle;
|
|
rand_angle_x += random()*10;
|
|
rand_angle_y += random()*10;
|
|
|
|
makevectors(rand_angle);
|
|
flamespr.velocity = aim(self, 1000);
|
|
flamespr.velocity *= 600;
|
|
|
|
flamespr.owner = self;
|
|
|
|
flamespr.think = Flame_Buildup;
|
|
flamespr.nextthink = time + 0.01;
|
|
flamespr.touch = Flame_Touch;
|
|
|
|
setmodel(flamespr, "models/sprites/flame.spr");
|
|
setorigin(flamespr, self.origin + self.view_ofs - '0 0 10');
|
|
setsize(flamespr, '0 0 0', '0 0 0');
|
|
}
|
|
|
|
void(float side) W_Fire =
|
|
{
|
|
if (self.weapon == W_M2 || self.weapon == W_FIW && self.cooldown)
|
|
return;
|
|
|
|
if (self.semiswitch == true)
|
|
return;
|
|
|
|
//First check that we can actualy fire
|
|
if (side == S_RIGHT &&
|
|
(time < self.fire_delay ||
|
|
self.new_anim_stop ||
|
|
self.reload_delay > time || (!self.currentammo && !self.currentmag))) {
|
|
return;
|
|
}
|
|
|
|
if (side == S_LEFT &&
|
|
(time < self.fire_delay2 ||
|
|
self.new_anim2_stop ||
|
|
self.reload_delay2 > time)) {
|
|
return;
|
|
}
|
|
|
|
if (self.sprinting) {
|
|
W_SprintStop();
|
|
return;
|
|
}
|
|
|
|
float startframe;
|
|
float endframe;
|
|
float firetype;
|
|
float damage;
|
|
float shotcount;
|
|
string modelname = "";
|
|
string soundname;
|
|
float spread;
|
|
float sprdstance = 0;
|
|
float delay;
|
|
|
|
//Update the basic vectors
|
|
makevectors(self.v_angle);
|
|
|
|
//make sure magazine is loading
|
|
if (!self.currentmag && side == S_RIGHT)
|
|
{
|
|
W_Reload(S_RIGHT);
|
|
return;
|
|
} else if (!self.currentmag2 && side == S_LEFT)
|
|
{
|
|
W_Reload(S_LEFT);
|
|
return;
|
|
}
|
|
|
|
//Dont fire if the gun has to cycle
|
|
if (self.NeedLoad && (self.weapon == W_TRENCH || self.weapon == W_GUT || self.weapon == W_KAR_SCOPE || self.weapon == W_KAR || self.weapon == W_ARMAGEDDON || self.weapon == W_HEADCRACKER))
|
|
{
|
|
W_LoadAmmo();
|
|
return;
|
|
}
|
|
|
|
//get some basic info
|
|
damage = getWeaponDamage(self.weapon);
|
|
firetype = GetFiretype (self.weapon);
|
|
if (side == S_RIGHT) {
|
|
modelname = GetWeaponModel(self.weapon, 0);
|
|
} else {
|
|
if (IsDualWeapon(self.weapon)) {
|
|
modelname = GetLeftWeaponModel(self.weapon);
|
|
}
|
|
}
|
|
shotcount = GetWeaponShotcount(self.weapon);
|
|
soundname = GetWeaponSound(self.weapon);
|
|
delay = getWeaponDelay(self.weapon, FIRE);
|
|
|
|
// Double Tap 2.0
|
|
if (self.perks & P_DOUBLE)
|
|
shotcount *= 2;
|
|
|
|
switch(self.stance) {
|
|
case 2:
|
|
sprdstance = 0;
|
|
break;
|
|
case 1:
|
|
sprdstance = 2;
|
|
break;
|
|
case 0:
|
|
sprdstance = 4;
|
|
break;
|
|
default:
|
|
sprdstance = 0;
|
|
break;
|
|
}
|
|
|
|
spread = (1.5*GetWeaponSpread(self.weapon))-sprdstance;
|
|
|
|
if (self.perks & P_DEAD) {
|
|
spread *= 0.65;
|
|
}
|
|
|
|
if (self.zoom == 1) {
|
|
if (self.weapon == W_TRENCH ||
|
|
self.weapon == W_DB ||
|
|
self.weapon == W_SAWNOFF ||
|
|
self.weapon == W_GUT ||
|
|
self.weapon == W_BORE) {
|
|
spread = spread * 0.75;
|
|
} else {
|
|
spread = spread * 0.1;
|
|
}
|
|
} else if (self.zoom == 2) {
|
|
spread = spread * 0.05;
|
|
} else if (self.velocity || !(self.flags & FL_ONGROUND))
|
|
spread *= 2;
|
|
|
|
if (self.downed)
|
|
playdownfire();
|
|
|
|
// Check if weapon is semi-automatic and if it is, if it's ready to be fired.
|
|
if (Util_WeaponIsSemiAutomatic(self.weapon)) {
|
|
if (side == S_RIGHT) {
|
|
if (self.semi) return;
|
|
self.semi = true;
|
|
} else if (side == S_LEFT) {
|
|
if (self.semi2) return;
|
|
self.semi2 == true;
|
|
}
|
|
}
|
|
|
|
// Weapon Projectile/Trace Logic.
|
|
if (Util_WeaponFiresTraceshot(self.weapon)) {
|
|
FireTrace(shotcount, spread, damage, side);
|
|
}
|
|
|
|
switch(firetype) {
|
|
case FIRETYPE_GRENADE:
|
|
W_FireGrenade(side);
|
|
break;
|
|
case FIRETYPE_RAYBEAM:
|
|
W_FireRay();
|
|
break;
|
|
case FIRETYPE_TESLA:
|
|
W_FireTesla();
|
|
break;
|
|
case FIRETYPE_FLAME:
|
|
W_FireM2();
|
|
break;
|
|
default: break;
|
|
}
|
|
|
|
//Play weapon animation and sound
|
|
startframe = GetFrame(self.weapon,FIRE_START);
|
|
endframe = GetFrame(self.weapon,FIRE_END);
|
|
|
|
// Increment the amount of shots fired while downed
|
|
if (self.downed == true)
|
|
self.teslacount++;
|
|
|
|
if (self.perks & P_DOUBLE) {
|
|
delay *= 0.66;
|
|
}
|
|
|
|
if (self.weapon == W_GUT || self.weapon == W_KAR_SCOPE || self.weapon == W_TRENCH || self.weapon == W_KAR || self.weapon == W_ARMAGEDDON || self.weapon == W_HEADCRACKER)
|
|
{
|
|
Set_W_Frame (startframe, endframe, delay, 0, FIRE, W_LoadAmmo, modelname, FALSE, side);
|
|
self.NeedLoad = true;
|
|
} else {
|
|
Set_W_Frame (startframe, endframe, delay, 0, FIRE, CheckReload, modelname, FALSE, side);
|
|
}
|
|
|
|
|
|
sound (self, CHAN_WEAPON, soundname, 1, ATTN_NORM);
|
|
|
|
if (side == S_RIGHT) {
|
|
self.currentmag = self.currentmag - 1;
|
|
self.fire_delay = getWeaponDelay(self.weapon, FIRE) + time;
|
|
} else {
|
|
self.currentmag2 = self.currentmag2 - 1;
|
|
self.fire_delay2 = getWeaponDelay(self.weapon, FIRE) + time;
|
|
}
|
|
|
|
if (IsPapWeapon(self.weapon)) {
|
|
sound (self, 0, "sounds/weapons/papfire.wav", 1, ATTN_NORM);
|
|
}
|
|
|
|
SetUpdate(self, UT_HUD, 6, 0, 0);
|
|
SetUpdate(self, UT_CROSSHAIR, 5, 0, 0);
|
|
|
|
|
|
// REFORMAT THIS
|
|
#ifndef PC
|
|
self.effects = self.effects | EF_MUZZLEFLASH;
|
|
#else
|
|
// psp takes care of this in engine
|
|
if (self.perks & P_DEAD)
|
|
return;
|
|
|
|
vector Wep_Recoil;
|
|
Wep_Recoil = GetWeaponRecoil(self.weapon)*0.5;
|
|
|
|
self.punchangle_x = Wep_Recoil_x;
|
|
self.punchangle_y = Wep_Recoil_y;
|
|
self.punchangle_z = Wep_Recoil_z;
|
|
#endif
|
|
}
|
|
|
|
|
|
/******************************
|
|
* W_Knife *
|
|
******************************/
|
|
|
|
void () W_Knife =
|
|
{
|
|
if (time < self.knife_delay || self.new_anim_stop ||
|
|
self.zoom || self.health == 10 ||
|
|
(!(self.flags & FL_ONGROUND)))
|
|
return;
|
|
|
|
vector source;
|
|
entity hit_ent = world;
|
|
//entity oldself;
|
|
float r, backupSkin;
|
|
vector org;
|
|
float hit = false;
|
|
|
|
if (self.sprinting) {
|
|
W_SprintStop();
|
|
}
|
|
|
|
backupSkin = self.weaponskin;
|
|
self.weaponskin = 0;
|
|
sound (self, CHAN_WEAPON, "sounds/weapons/knife/knife.wav", 1, ATTN_NORM);
|
|
makevectors (self.v_angle);
|
|
source = self.origin + self.view_ofs;
|
|
if (self.weapon == W_BK)
|
|
traceline (source, source + v_forward*128, 0, self);
|
|
else
|
|
traceline (source, source + v_forward*100, 0, self);
|
|
|
|
org = trace_endpos - v_forward*4;
|
|
|
|
r = fabs(vlen(source - org));
|
|
|
|
if (trace_ent.takedamage)
|
|
{
|
|
if (trace_ent.classname == "ai_zombie_head")
|
|
hit_ent = trace_ent.owner;
|
|
else if (trace_ent.classname == "ai_zombie_larm")
|
|
hit_ent = trace_ent.owner;
|
|
else if (trace_ent.classname == "ai_zombie_rarm")
|
|
hit_ent = trace_ent.owner;
|
|
else
|
|
hit_ent = trace_ent;
|
|
if (hit_ent.classname == "ai_zombie")
|
|
hit = true;
|
|
|
|
// if zombie has a head and it's instakill, gib for a e s t h e t i c
|
|
if (hit_ent.head && instakill_finished >= time) {
|
|
spawn_gibs(hit_ent.head.origin);
|
|
setmodel(hit_ent.head, "");
|
|
}
|
|
}
|
|
else if (trace_fraction < 1.0)
|
|
{ // hit wall
|
|
sound (self, CHAN_WEAPON, "sounds/weapons/knife/knife_hit.wav", 1, ATTN_NORM);
|
|
|
|
/* naievil (FIXME) add weapon decal here */
|
|
|
|
#ifdef PC
|
|
if (self.weapon != W_RAY && self.weapon != W_PORTER) {
|
|
WriteByte(MSG_MULTICAST, SVC_CGAMEPACKET);
|
|
WriteByte(MSG_MULTICAST, EVENT_PISTOLFIRE);
|
|
WriteEntity(MSG_MULTICAST, self);
|
|
WriteFloat(MSG_MULTICAST, 0);
|
|
WriteCoord(MSG_MULTICAST, org_x);
|
|
WriteCoord(MSG_MULTICAST, org_y);
|
|
WriteCoord(MSG_MULTICAST, org_z);
|
|
WriteCoord(MSG_MULTICAST, trace_plane_normal_x);
|
|
WriteCoord(MSG_MULTICAST, trace_plane_normal_y);
|
|
WriteCoord(MSG_MULTICAST, trace_plane_normal_z);
|
|
WriteEntity(MSG_MULTICAST, hit_ent);
|
|
multicast(trace_endpos, MULTICAST_PHS);
|
|
}
|
|
#else
|
|
WriteByte (MSG_BROADCAST, SVC_TEMPENTITY);
|
|
WriteByte (MSG_BROADCAST, TE_GUNSHOT);
|
|
WriteCoord (MSG_BROADCAST, org_x);
|
|
WriteCoord (MSG_BROADCAST, org_y);
|
|
WriteCoord (MSG_BROADCAST, org_z);
|
|
#endif
|
|
}
|
|
|
|
if (r > 40 && hit)//lunge
|
|
{
|
|
if (self.weapon == W_BK)
|
|
{
|
|
Set_W_Frame (14, 24, 1.4, 0, KNIFE, SUB_Null, "models/weapons/bk/v_bk.mdl", false, S_RIGHT);
|
|
self.fire_delay = time + 1.5;
|
|
self.knife_delay = time + 1.75;
|
|
}
|
|
else if (self.bowie)
|
|
{
|
|
Set_W_Frame (4, 10, 0.7, 0, KNIFE, ReturnWeaponModel, "models/weapons/knife/v_bowie.mdl", false, S_RIGHT);
|
|
self.reload_delay2 = self.fire_delay2 = self.reload_delay = self.fire_delay = time + 0.8;
|
|
self.knife_delay = time + 1.3;
|
|
}
|
|
else
|
|
{
|
|
self.punchangle_x = -5;
|
|
self.punchangle_y = -10;
|
|
Set_W_Frame (4, 12, 0.7, 0, KNIFE, ReturnWeaponModel, "models/weapons/knife/v_knife.mdl", false, S_RIGHT);
|
|
self.reload_delay2 = self.fire_delay2 = self.reload_delay = self.fire_delay = time + 0.8;
|
|
self.knife_delay = time + 1;
|
|
}
|
|
self.velocity = v_forward * 200;
|
|
|
|
}
|
|
else
|
|
{
|
|
if (self.weapon == W_BK)//slash
|
|
{
|
|
Set_W_Frame (8, 13, 0, 0.4, KNIFE, SUB_Null, "models/weapons/bk/v_bk.mdl", false, S_RIGHT);
|
|
self.fire_delay = time + 0.5;
|
|
self.knife_delay = time + 0.8;
|
|
}
|
|
else if (self.bowie)
|
|
{
|
|
Set_W_Frame (0, 3, 0.3, 0, KNIFE, ReturnWeaponModel, "models/weapons/knife/v_bowie.mdl", false, S_RIGHT);
|
|
self.reload_delay2 = self.fire_delay2 = self.reload_delay = self.fire_delay = time + 0.4;
|
|
self.knife_delay = time + 0.9;
|
|
}
|
|
else
|
|
{
|
|
self.punchangle_x = -5;
|
|
self.punchangle_y = -10;
|
|
//self.punchangle_z = 5;
|
|
Set_W_Frame (0, 3, 0.3, 0, KNIFE, ReturnWeaponModel, "models/weapons/knife/v_knife.mdl", false, S_RIGHT);
|
|
self.reload_delay2 = self.fire_delay2 = self.reload_delay = self.fire_delay = time + 0.4;
|
|
self.knife_delay = time + 0.7;
|
|
}
|
|
}
|
|
self.reload_delay2 = self.reload_delay = 0;
|
|
self.weaponskin = backupSkin;
|
|
|
|
LungeKnifeHit();
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/******************************
|
|
* W_Grenade *
|
|
******************************/
|
|
|
|
void() switch_nade =
|
|
{
|
|
if (self.downed || self.isBuying || !(self.grenades & 2) || self.seminade)
|
|
return;
|
|
|
|
if (self.pri_grenade_state == 0)
|
|
{
|
|
centerprint (self, "Bouncing Betties Selected");
|
|
self.bk_nade = 1;
|
|
}
|
|
if (self.pri_grenade_state == 1)
|
|
{
|
|
centerprint (self, "Frag Grenades Selected");
|
|
self.bk_nade = 0;
|
|
}
|
|
|
|
self.pri_grenade_state = self.bk_nade;
|
|
|
|
};
|
|
|
|
void() GrenadeExplode =
|
|
{
|
|
sound (self, CHAN_WEAPON, "sounds/weapons/grenade/explode.wav", 1, ATTN_NORM);
|
|
|
|
if (self.classname == "grenade")
|
|
DamgageExplode (self, self.owner, 225, 75, 128);//was 256 inch radius, kicked it down to 128 because 21 feet exploding radius is... a bit overkill...
|
|
else
|
|
DamgageExplode (self, self.owner, 1200, 1000, 128); //explosive weapons are more powerful :)
|
|
|
|
#ifdef PC
|
|
te_customflash(self.origin, 178, 300, '1 1 1');
|
|
#endif
|
|
|
|
CallExplosion(self.origin);
|
|
|
|
SUB_Remove ();
|
|
};
|
|
void() NadeStraighten =
|
|
{
|
|
self.angles_x = (self.angles_x > 0) ? 90 : -90;
|
|
|
|
if (self.pri_grenade_state == 1)
|
|
return;
|
|
};
|
|
void() Velocity_reduce =
|
|
{
|
|
if (!other.solid || other.solid == SOLID_TRIGGER)
|
|
if (other != world)
|
|
return;
|
|
|
|
self.velocity = self.velocity*0.5;
|
|
|
|
NadeStraighten();
|
|
};
|
|
|
|
void() betty_touch =
|
|
{
|
|
//centerprint (other, "wow\n"); <- THIS
|
|
|
|
if (other.classname == "grenade")
|
|
return;
|
|
|
|
if (other == self.owner || other.solid == SOLID_TRIGGER)
|
|
if (other != world)
|
|
return;
|
|
|
|
GrenadeExplode();
|
|
}
|
|
|
|
void() W_ThrowBetty =
|
|
{
|
|
local entity betty;
|
|
betty = spawn ();
|
|
betty.owner = self;
|
|
betty.grenade_delay = betty.owner.grenade_delay;
|
|
betty.owner.grenade_delay = 0;
|
|
betty.movetype = MOVETYPE_NONE;
|
|
betty.solid = SOLID_TRIGGER;
|
|
betty.classname = "betty";
|
|
|
|
// set betty speed
|
|
betty.velocity = v_forward*0;
|
|
|
|
betty.think = GrenadeExplode;
|
|
betty.touch = betty_touch;
|
|
|
|
setmodel (betty, "models/weapons/grenade/g_betty.mdl");
|
|
setsize (betty, '-16 -16 -4', '16 16 1');
|
|
|
|
betty.origin = betty.owner.origin - self.view_ofs + '0 0 1';
|
|
//if player isn't on ground, drop the betty to the floor.
|
|
if (!(self.flags & FL_ONGROUND)) {
|
|
local entity oldself;
|
|
oldself = self;
|
|
self = betty;
|
|
|
|
#ifdef PC
|
|
droptofloor();
|
|
#else
|
|
droptofloor(0, 0);
|
|
#endif
|
|
self = oldself;
|
|
}
|
|
betty.origin += v_forward * 0;
|
|
setorigin (betty, betty.origin);
|
|
|
|
self.animend = ReturnWeaponModel;
|
|
self.callfuncat = 0;
|
|
self.isBuying = false;
|
|
}
|
|
|
|
void() W_ThrowGrenade =
|
|
{
|
|
string modelname;
|
|
float startframe;
|
|
float endframe;
|
|
local entity nade;
|
|
|
|
if (self.pri_grenade_state == 0) {
|
|
nade = spawn ();
|
|
nade.owner = self;
|
|
nade.grenade_delay = nade.owner.grenade_delay;
|
|
nade.owner.grenade_delay = 0;
|
|
nade.movetype = MOVETYPE_BOUNCE;
|
|
nade.solid = SOLID_BBOX;
|
|
nade.classname = "grenade";
|
|
|
|
// set nade speed
|
|
|
|
makevectors (self.v_angle);
|
|
|
|
nade.velocity = v_forward*800;
|
|
|
|
nade.avelocity = '400 -400 400';
|
|
|
|
//nade.angles = vectoangles(nade.velocity);
|
|
//nade.angles_z += (nade.angles_z + 180 < 360)? 180 : -180;
|
|
nade.angles = vectoangles(v_forward);
|
|
nade.angles_z -= 15;
|
|
// set nade duration
|
|
nade.nextthink = nade.grenade_delay;
|
|
nade.think = GrenadeExplode;
|
|
|
|
nade.touch = Velocity_reduce;
|
|
setmodel (nade, "models/weapons/grenade/g_grenade.mdl");
|
|
setsize (nade, '0 0 0', '0 0 0');
|
|
nade.origin = self.origin + self.view_ofs;
|
|
nade.origin += v_forward * 12;
|
|
setorigin (nade, nade.origin);
|
|
|
|
self.animend = ReturnWeaponModel;
|
|
self.callfuncat = 0;
|
|
self.isBuying = false;
|
|
} else if (self.pri_grenade_state == 1) {
|
|
W_ThrowBetty();
|
|
} else {
|
|
centerprint (other, "No grenadetype defined...\n");
|
|
}
|
|
|
|
//if (!(self.flags & FL_ONGROUND)) {
|
|
// self.secondary_grenades = self.secondary_grenades + 1;
|
|
//}
|
|
|
|
startframe = GetFrame(self.weapon,TAKE_OUT_START);
|
|
endframe = GetFrame(self.weapon,TAKE_OUT_END);
|
|
modelname = GetWeaponModel(self.weapon, 0);
|
|
Set_W_Frame (startframe, endframe, 0, 0, 0, SUB_Null, modelname, false, S_BOTH);
|
|
|
|
SetUpdate(self, UT_HUD, 6, 0, 0);
|
|
}
|
|
|
|
|
|
void() checkHoldB =
|
|
{
|
|
if (!self.button3)
|
|
{
|
|
if(self.grenade_delay < time)
|
|
self.grenade_delay = time + 0.05;
|
|
self.isBuying = true;
|
|
Set_W_Frame (18, 23, 0, 5, GRENADE, W_ThrowBetty, "models/weapons/grenade/v_betty.mdl", true, S_RIGHT);
|
|
sound (self, CHAN_WEAPON, "sounds/weapons/grenade/throw.wav", 1, ATTN_NORM);
|
|
self.reload_delay2 = self.fire_delay2 = self.reload_delay = self.fire_delay = time + 0.4;
|
|
self.throw_delay = time + 0.9;
|
|
}
|
|
else
|
|
{
|
|
self.isBuying = true;
|
|
Set_W_Frame (18, 18, 0, 0, GRENADE, checkHoldB, "models/weapons/grenade/v_betty.mdl", true, S_RIGHT);
|
|
}
|
|
}
|
|
|
|
void() checkHold =
|
|
{
|
|
if (self.pri_grenade_state == 1) {
|
|
checkHoldB();
|
|
return;
|
|
}
|
|
|
|
if (!self.button3 || self.grenade_delay < time)
|
|
{
|
|
if(self.grenade_delay < time)
|
|
self.grenade_delay = time + 0.05;
|
|
|
|
self.isBuying = true;
|
|
Set_W_Frame (3, 6, 0, 5, GRENADE, W_ThrowGrenade, "models/weapons/grenade/v_grenade.mdl", true, S_RIGHT);
|
|
sound (self, CHAN_WEAPON, "sounds/weapons/grenade/throw.wav", 1, ATTN_NORM);
|
|
self.reload_delay2 = self.fire_delay2 = self.reload_delay = self.fire_delay = time + 0.4;
|
|
self.throw_delay = time + 0.9;
|
|
}
|
|
else
|
|
{
|
|
self.isBuying = true;
|
|
Set_W_Frame (2, 2, 0, 0, GRENADE, checkHold, "models/weapons/grenade/v_grenade.mdl", true, S_RIGHT);
|
|
}
|
|
}
|
|
|
|
|
|
void() W_Betty = {
|
|
if (self.throw_delay > time || self.zoom || self.downed || self.secondary_grenades < 1 || self.isBuying)
|
|
return;
|
|
|
|
// Prevent the Player from Sprinting and also avoid issues with
|
|
// the equipment being completely cancelled..
|
|
if (self.sprinting)
|
|
W_SprintStop();
|
|
|
|
Set_W_Frame (0, 18, 0, 0, GRENADE, checkHoldB, "models/weapons/grenade/v_betty.mdl", true, S_RIGHT);
|
|
sound (self, CHAN_WEAPON, "sounds/weapons/grenade/prime.wav", 1, ATTN_NORM);
|
|
self.secondary_grenades -= 1;
|
|
self.reload_delay2 = self.fire_delay2 = self.throw_delay = self.reload_delay = self.fire_delay = time + 6;
|
|
self.seminade = true;
|
|
}
|
|
|
|
void() W_Grenade =
|
|
{
|
|
if (self.throw_delay > time || self.zoom || self.downed || self.primary_grenades < 1 || self.isBuying)
|
|
return;
|
|
|
|
// Prevent the Player from Sprinting and also avoid issues with
|
|
// the equipment being completely cancelled..
|
|
if (self.sprinting)
|
|
W_SprintStop();
|
|
|
|
Set_W_Frame (0, 2, 0, 0, GRENADE, checkHold, "models/weapons/grenade/v_grenade.mdl", true, S_RIGHT);
|
|
sound (self, CHAN_WEAPON, "sounds/weapons/grenade/prime.wav", 1, ATTN_NORM);
|
|
self.primary_grenades -= 1;
|
|
self.reload_delay2 = self.fire_delay2 = self.throw_delay = self.reload_delay = self.fire_delay = time + 6;
|
|
self.grenade_delay = time + 5;
|
|
self.seminade = true;
|
|
}
|
|
|
|
void(float wepnum) CheckButton = {
|
|
W_PutOut();
|
|
}
|
|
|
|
void() Change_Stance = {
|
|
|
|
if (self.downed || !(self.flags & FL_ONGROUND) || self.changestance == true || self.new_ofs_z != self.view_ofs_z)
|
|
return;
|
|
|
|
switch(self.stance) {
|
|
case 2:
|
|
self.new_ofs_z = self.view_ofs_z - 24;
|
|
self.stance = 1;
|
|
break;
|
|
case 1:
|
|
if (self.isBuying) { //don't want to prone while buying a perk..
|
|
self.stance = 2;
|
|
self.new_ofs_z = self.view_ofs_z + 24;
|
|
} else if (!self.stancereset) {
|
|
self.new_ofs_z = self.view_ofs_z - 18;
|
|
self.stance = 0;
|
|
} else {
|
|
self.new_ofs_z = self.view_ofs_z + 24;
|
|
self.stance = 2;
|
|
self.stancereset = 0;
|
|
}
|
|
break;
|
|
case 0:
|
|
self.new_ofs_z = self.view_ofs_z + 18;
|
|
self.stance = self.stancereset = 1;
|
|
break;
|
|
default: break;
|
|
}
|
|
|
|
}
|
|
|
|
void() dolphin_dive = //naievil
|
|
{
|
|
if (self.stance != 2 || self.view_ofs_z != 32)
|
|
return;
|
|
if (self.flags & FL_WATERJUMP)
|
|
return;
|
|
|
|
if (self.waterlevel >= 2)
|
|
{
|
|
if (self.watertype == CONTENT_WATER)
|
|
self.velocity_z = 100;
|
|
else if (self.watertype == CONTENT_SLIME)
|
|
self.velocity_z = 80;
|
|
else
|
|
self.velocity_z = 50;
|
|
}
|
|
|
|
if (!(self.flags & FL_ONGROUND))
|
|
return;
|
|
|
|
if ( !(self.flags & FL_JUMPRELEASED) )
|
|
return; // don't pogo stick
|
|
|
|
|
|
self.flags = self.flags - (self.flags & FL_JUMPRELEASED);
|
|
self.flags = self.flags - FL_ONGROUND; // don't stairwalk
|
|
makevectors (self.v_angle);
|
|
self.velocity = self.velocity * 1.2;
|
|
self.velocity_z = self.velocity_z + 270;
|
|
|
|
if (self.button2)
|
|
self.button2 = 0;
|
|
|
|
self.dive = 1;
|
|
W_SprintStop();
|
|
|
|
self.oldz = self.origin_z;
|
|
self.new_ofs_z = self.view_ofs_z - 42;
|
|
self.stance = 0;
|
|
}
|
|
|
|
|
|
void () Impulse_Functions =
|
|
{
|
|
if (!self.impulse)
|
|
return;
|
|
|
|
switch (self.impulse)
|
|
{
|
|
case 10:
|
|
#ifdef PC
|
|
// FTE handles jumping on PC, which we don't want.. so we use impulses to override.
|
|
JumpCheck(1);
|
|
#endif
|
|
break;
|
|
case 22:
|
|
addmoney(self, 10000, 0);
|
|
rounds += 4;
|
|
break;
|
|
case 23:
|
|
#ifdef PC
|
|
if (!checkMovement(forward))
|
|
return;
|
|
#endif
|
|
if (self.dive || self.downed)
|
|
return;
|
|
switch(self.stance) {
|
|
case 0:
|
|
self.new_ofs_z = self.view_ofs_z + 42;
|
|
self.stance = 2;
|
|
break;
|
|
case 1:
|
|
self.new_ofs_z = self.view_ofs_z + 24;
|
|
self.stance = 2;
|
|
break;
|
|
default: break;
|
|
}
|
|
W_SprintStart();
|
|
break;
|
|
case 24:
|
|
W_SprintStop();
|
|
break;
|
|
case 25:
|
|
#ifdef HANDHELD
|
|
switch_nade();
|
|
#else
|
|
W_Betty();
|
|
#endif
|
|
break;
|
|
case 26:
|
|
case 30:
|
|
if (self.sprinting)
|
|
dolphin_dive();
|
|
else
|
|
Change_Stance();
|
|
break;
|
|
#ifdef NX
|
|
case 31:
|
|
self.sprintflag = 0;
|
|
break;
|
|
case 32:
|
|
self.sprintflag = 1;
|
|
break;
|
|
#endif
|
|
case 110:
|
|
case 111:
|
|
W_PutOut();
|
|
self.semiswitch = true;
|
|
break;
|
|
case 100:
|
|
cvar_set("waypoint_mode", "1");
|
|
|
|
local entity d;
|
|
d = find(world,classname,"door_nzp_cost");
|
|
while(d != world)
|
|
{
|
|
d.classname = "door_nzp";
|
|
d = find(d,classname,"door_nzp_cost");
|
|
}
|
|
d = find(world,classname,"door_open");
|
|
while(d != world)
|
|
{
|
|
d.classname = "door_nzp";
|
|
d = find(d,classname,"door_open");
|
|
}
|
|
|
|
waypoint_mode = 1;
|
|
break;
|
|
default:
|
|
#ifdef PC
|
|
bprint(PRINT_HIGH, "Unknown impulse: ", ftos(self.impulse));
|
|
#endif
|
|
break;
|
|
}
|
|
|
|
self.impulse = 0;
|
|
};
|
|
|
|
void() CheckImpulses =
|
|
{
|
|
self.impulse = 0;
|
|
}
|
|
|
|
void(entity ent) CheckPowerups =
|
|
{
|
|
if (instakill_finished >= time)
|
|
{
|
|
if (instakill_finished < 4 + time)
|
|
{
|
|
if (insta_blink< time - 0.25)
|
|
insta_blink = time + 0.25;
|
|
}
|
|
else if (instakill_finished < 7 + time)
|
|
{
|
|
if (insta_blink< time - 0.5)
|
|
insta_blink = time + 0.5;
|
|
}
|
|
else if (instakill_finished < 10 + time)
|
|
{
|
|
if (insta_blink< time - 1)
|
|
insta_blink = time + 1;
|
|
}
|
|
|
|
if (insta_blink < time)
|
|
ent.insta_icon = true;
|
|
else
|
|
ent.insta_icon = false;
|
|
}
|
|
else if(instakill_finished)
|
|
{
|
|
instakill_finished = 0;
|
|
|
|
ent.insta_icon = false;
|
|
}
|
|
|
|
if (x2_finished > time)
|
|
{
|
|
if (x2_finished < 4 + time)
|
|
{
|
|
if (x2_blink< time - 0.25)
|
|
x2_blink = time + 0.25;
|
|
}
|
|
else if (x2_finished < 7 + time)
|
|
{
|
|
if (x2_blink< time - 0.5)
|
|
x2_blink = time + 0.5;
|
|
}
|
|
else if (x2_finished < 10 + time)
|
|
{
|
|
if (x2_blink< time - 1)
|
|
x2_blink = time + 1;
|
|
}
|
|
|
|
if (x2_blink < time)
|
|
ent.x2_icon = true;
|
|
else
|
|
ent.x2_icon = false;
|
|
}
|
|
else if(x2_finished)
|
|
{
|
|
x2_finished = 0;
|
|
|
|
ent.x2_icon = false;
|
|
}
|
|
|
|
}
|
|
|
|
void() CheckPlayer =
|
|
{
|
|
CheckPowerups(self);
|
|
CheckRevive(self);
|
|
|
|
#ifdef NX
|
|
if (!self.sprintflag && self.sprinting) {
|
|
W_SprintStop();
|
|
}
|
|
#endif
|
|
|
|
// FIXME - can't do frame independent stance changing.. ofs starts spazzing..
|
|
if (self.view_ofs_z > self.new_ofs_z) {
|
|
self.changestance = true;
|
|
self.view_ofs_z = self.view_ofs_z - 1.5;
|
|
} else if (self.view_ofs_z < self.new_ofs_z) {
|
|
self.changestance = true;
|
|
self.view_ofs_z = self.view_ofs_z + 1.5;
|
|
} else {
|
|
self.changestance = false;
|
|
}
|
|
|
|
if (self.view_ofs_z > 32) {
|
|
self.view_ofs_z = 32;
|
|
self.changestance = false;
|
|
}
|
|
|
|
/* Check player push start */
|
|
#ifdef PC
|
|
entity ent;
|
|
ent = findradius(self.origin, 70);
|
|
|
|
while(ent)
|
|
{
|
|
if(ent.classname == "player" && ent != self && ent.downed && !self.downed)
|
|
{
|
|
useprint (self, 13, 0, 0);
|
|
|
|
if (self.button7 && !ent.invoke_revive) {
|
|
centerprint(ent, "Another player is reviving you...");
|
|
ent.beingrevived = true;
|
|
|
|
if (!self.progress_bar_percent) {
|
|
self.movetype = MOVETYPE_NONE;
|
|
Set_W_Frame (0, 21, 0, 0, SPRINT, SUB_Null, "models/v_morphine.mdl", false, S_BOTH);
|
|
|
|
if !(self.perks & P_REVIVE)
|
|
self.progress_bar_time = 4;
|
|
else
|
|
self.progress_bar_time = 2;
|
|
|
|
self.progress_bar = self.progress_bar_time + time;
|
|
self.progress_bar_percent = 1;
|
|
self.reviving = true;
|
|
}
|
|
|
|
if (self.revived) {
|
|
self.movetype = MOVETYPE_WALK;
|
|
W_TakeOut();
|
|
ent.invoke_revive = 1;
|
|
ent.beingrevived = false;
|
|
self.reviving = 0;
|
|
self.progress_bar = 0;
|
|
self.progress_bar_time = 0;
|
|
self.progress_bar_percent = 0;
|
|
self.revived = 0;
|
|
addmoney(self, ent.requirespower, false);
|
|
}
|
|
}
|
|
else if (!self.button7 && self.reviving) {
|
|
self.movetype = MOVETYPE_WALK;
|
|
ent.beingrevived = false;
|
|
W_TakeOut();
|
|
self.progress_bar = 0;
|
|
self.progress_bar_time = 0;
|
|
self.progress_bar_percent = 0;
|
|
self.revived = 0;
|
|
self.reviving = 0;
|
|
}
|
|
}
|
|
else if(ent.classname == "player" && ent != self && !ent.downed) // small player push
|
|
{
|
|
vector push;
|
|
push = ent.origin - self.origin;
|
|
push_z = 0;
|
|
push = normalize(push) * 0.5;
|
|
|
|
ent.velocity += push;
|
|
}
|
|
ent = ent.chain;
|
|
}
|
|
#endif
|
|
/* Check for player push end */
|
|
|
|
if (!self.button2 && self.flags & FL_ONGROUND) {
|
|
if (self.dive) {
|
|
if (self.perks & P_FLOP && fabs(self.oldz - self.origin_z) > 90) {
|
|
sound (self, CHAN_WEAPON, "sounds/weapons/grenade/explode.wav", 1, ATTN_NORM);
|
|
DamgageExplode (self, self, 225, 75, 128);
|
|
#ifdef PC
|
|
te_customflash(self.origin, 128, 300, '1 1 1');
|
|
#endif
|
|
|
|
CallExplosion(self.origin);
|
|
}
|
|
self.dive = 0;
|
|
}
|
|
|
|
if (fabs(self.oldz - self.origin_z) > 90 && !(self.perks & P_FLOP)) {
|
|
DamageHandler (self, other, 50, S_ZOMBIE);
|
|
|
|
if (self.health <= 10)
|
|
GiveAchievement(7, self);
|
|
}
|
|
|
|
self.oldz = self.origin_z;
|
|
}
|
|
}
|
|
|
|
void () Weapon_Logic =
|
|
{
|
|
W_Frame_Update ();
|
|
Impulse_Functions();
|
|
|
|
#ifndef PC
|
|
// For HANDHELD only to tell our engine to display the scope
|
|
if ((self.scopetime < time) && self.scopetime) {
|
|
self.scopetime = 0;
|
|
self.zoom = 2;
|
|
self.weapon2model = "";
|
|
self.weaponmodel = "";
|
|
}
|
|
#else
|
|
UpdatePunchangle(self);
|
|
#endif
|
|
|
|
//
|
|
// Fire Button Logic
|
|
// NOTE: Since the right weapon is the "normal" side there's some
|
|
// flip flop logic on PC so that LMB will fire the left side instead.
|
|
//
|
|
|
|
// Right Trigger/LMB/etc.
|
|
if (self.button0) {
|
|
#ifdef PC
|
|
if (IsDualWeapon(self.weapon)) { W_Fire(S_LEFT); } else { W_Fire(S_RIGHT); }
|
|
#else
|
|
W_Fire(S_RIGHT);
|
|
#endif // PC
|
|
} else {
|
|
// Check for release for semi automatic weapons.
|
|
#ifdef PC
|
|
if (IsDualWeapon(self.weapon)) { self.semi2 = false; } else { self.semi = false; }
|
|
#else
|
|
self.semi = false;
|
|
#endif // PC
|
|
}
|
|
|
|
// Left Trigger/RMB/etc.
|
|
if (self.button8) {
|
|
if (IsDualWeapon(self.weapon)) {
|
|
#ifdef PC
|
|
W_Fire(S_RIGHT);
|
|
#else
|
|
W_Fire(S_LEFT);
|
|
#endif // PC
|
|
} else {
|
|
// Toggle ADS back and forth
|
|
if (cvar("cl_adsmode")) {
|
|
self.ads_toggle = !self.ads_toggle;
|
|
|
|
if (self.ads_toggle == false) {
|
|
W_AimOut();
|
|
self.zoom = 0;
|
|
|
|
#ifdef PC
|
|
self.viewzoom = 1;
|
|
UpdateVmodel(self.weaponmodel, GetWepSkin(self.weapon));
|
|
UpdateV2model(self.weapon2model, 0);
|
|
#endif // PC
|
|
} else {
|
|
W_AimIn();
|
|
self.zoom = 1;
|
|
|
|
#ifdef PC
|
|
self.viewzoom = 0.9;
|
|
#endif // PC
|
|
}
|
|
} else {
|
|
if (!self.zoom) {
|
|
W_AimIn();
|
|
}
|
|
#ifdef PC
|
|
else {
|
|
|
|
if (self.weapon == W_KAR_SCOPE || self.weapon == W_PTRS ||
|
|
self.weapon == W_HEADCRACKER || self.weapon == W_PENETRATOR) {
|
|
if (self.viewzoom > 0.5)
|
|
self.viewzoom -= 0.05;
|
|
else
|
|
self.viewzoom = 0.5;
|
|
} else {
|
|
if (self.viewzoom > 0.9)
|
|
self.viewzoom -= 0.01;
|
|
else
|
|
self.viewzoom = 0.89999;
|
|
}
|
|
}
|
|
#endif // PC
|
|
}
|
|
}
|
|
} else if (!self.button8 && self.zoom) {
|
|
#ifdef PC
|
|
if (self.weapon == W_KAR_SCOPE || self.weapon == W_PTRS ||
|
|
self.weapon == W_HEADCRACKER || self.weapon == W_PENETRATOR) {
|
|
if (self.viewzoom == 0.5) {
|
|
self.viewzoom += 0.05;
|
|
ReturnWeaponModel();
|
|
} else {
|
|
UpdateVmodel(self.weaponmodel, GetWepSkin(self.weapon));
|
|
UpdateV2model(self.weapon2model, 0);
|
|
self.viewzoom += 0.05;
|
|
}
|
|
} else {
|
|
if (self.viewzoom == 0.9) {
|
|
W_AimOut();
|
|
} else {
|
|
self.viewzoom += 0.015;
|
|
}
|
|
}
|
|
|
|
if (self.viewzoom >= 1) {
|
|
self.viewzoom = 1;
|
|
self.zoom = 0;
|
|
}
|
|
#else
|
|
W_AimOut();
|
|
#endif // PC
|
|
} else {
|
|
// Check for release for semi automatic weapons.
|
|
if (IsDualWeapon(self.weapon)) {
|
|
#ifdef PC
|
|
self.semi = false;
|
|
#else
|
|
self.semi2 = false;
|
|
#endif // PC
|
|
}
|
|
}
|
|
|
|
if (!self.button7) {
|
|
self.semiuse = false;
|
|
}
|
|
|
|
if (self.button4 && !self.semiswitch && self.secondaryweapon && self.secondaryweapon !=0 && !self.zoom)
|
|
{
|
|
W_PutOut();
|
|
self.semiswitch = true;
|
|
} else if (!self.button4 && self.semiswitch && self.impulse != 110 && self.impulse != 111) {
|
|
self.semiswitch = false;
|
|
}
|
|
|
|
if (self.button3 && self.isBuying != true && self.seminade != true)
|
|
{
|
|
if (self.pri_grenade_state == 0)
|
|
W_Grenade();
|
|
else
|
|
W_Betty();
|
|
self.seminade = true;
|
|
} else if (!self.button3 && self.seminade) {
|
|
self.seminade = false;
|
|
}
|
|
|
|
if (self.button5 && !self.semireload)
|
|
{
|
|
W_Reload(S_BOTH);
|
|
self.semireload = TRUE;
|
|
}
|
|
else if (!self.button5) {
|
|
self.semireload = FALSE;
|
|
}
|
|
|
|
if (self.button6 && !self.semiknife)
|
|
{
|
|
#ifdef HANDHELD
|
|
if (self.sprinting) {
|
|
dolphin_dive();
|
|
return;
|
|
}
|
|
#endif
|
|
W_Knife();
|
|
self.semiknife = true;
|
|
}
|
|
else if (!self.button6)
|
|
self.semiknife = false;
|
|
|
|
CheckImpulses();
|
|
CheckPlayer();
|
|
}
|