2205 lines
44 KiB
C++
2205 lines
44 KiB
C++
void() W_SetCurrentAmmo;
|
|
/* ALL LIGHTS SHOULD BE 0 1 0 IN COLOR ALL OTHER ITEMS SHOULD
|
|
BE .8 .3 .4 IN COLOR */
|
|
|
|
void(entity to, float iid, float count) AddStackable;
|
|
void (entity jeb, float time) spawn_excla;
|
|
void() AmmoTouch;
|
|
void() shop_face;
|
|
|
|
void() SUB_regen =
|
|
{
|
|
self.model = self.mdl; // restore original model
|
|
self.solid = SOLID_TRIGGER; // allow it to be touched again
|
|
sound (self, CHAN_VOICE, "items/itembk2.wav", 1, ATTN_NORM); // play respawn sound
|
|
setorigin (self, self.origin);
|
|
};
|
|
|
|
float() crandom;
|
|
|
|
void () Levelshake =
|
|
{
|
|
local entity te;
|
|
|
|
te = findradius (self.origin, 1400);
|
|
while (te)
|
|
{
|
|
if (te.classname == "player")
|
|
stuffcmd (te, "v_idlescale 4\n");
|
|
|
|
if (te.classname == "monster")
|
|
te.alert = 120;
|
|
|
|
te = te.chain;
|
|
}
|
|
};
|
|
|
|
void(vector pos) throw_debris =
|
|
{
|
|
local entity tank, oself;
|
|
local float oldz;
|
|
|
|
oself = self;
|
|
tank = spawn();
|
|
self = tank;
|
|
|
|
self.origin = pos;
|
|
|
|
self.solid = SOLID_NOT;
|
|
self.movetype = MOVETYPE_BOUNCE;
|
|
self.takedamage = DAMAGE_NO;
|
|
|
|
if (random()<0.5)
|
|
setmodel (self, "progs/junk1.mdl");
|
|
else
|
|
setmodel (self, "progs/junk2.mdl");
|
|
|
|
setsize (self, '0 0 0', '0 0 0');
|
|
|
|
self.velocity_y = random()*900 - random()*900;
|
|
self.velocity_x = random()*900 - random()*900;
|
|
self.velocity_z = 400 + random()*400;
|
|
|
|
self.avelocity_x = random()*600;
|
|
self.avelocity_y = random()*600;
|
|
self.avelocity_z = random()*600;
|
|
|
|
self.think = SUB_Remove;
|
|
self.nextthink = time + 2+random()*2;
|
|
|
|
self = oself;
|
|
|
|
};
|
|
|
|
void() fueltank_explode =
|
|
{
|
|
throw_debris(self.origin);
|
|
throw_debris(self.origin);
|
|
throw_debris(self.origin);
|
|
throw_debris(self.origin);
|
|
throw_debris(self.origin);
|
|
|
|
bprint(2, "an enemy fuel tank has been destroyed!\n");
|
|
self.takedamage = DAMAGE_NO;
|
|
self.classname = "explo_box";
|
|
Levelshake();
|
|
T_RadiusDamage2 (self, self.owner, 50+random()*100, other, 200);
|
|
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+32);
|
|
multicast (self.origin, MULTICAST_PHS);
|
|
supplies = supplies - 1;
|
|
remove (self);
|
|
};
|
|
|
|
void(vector pos) FuelTankSmall =
|
|
{
|
|
local entity item, oself;
|
|
precache_model ("maps/b_exbox2.bsp");
|
|
|
|
item = spawn();
|
|
oself = self;
|
|
self = item;
|
|
setorigin (self, pos);
|
|
setmodel (self, "maps/b_exbox2.bsp");
|
|
supplies += 1;
|
|
self.rtime = 1;
|
|
self.solid = SOLID_SLIDEBOX;
|
|
setsize (self, '-16 -16 -16', '16 16 24');
|
|
self.health = 50;
|
|
self.th_die = fueltank_explode;
|
|
self.takedamage = DAMAGE_AIM;
|
|
self.classname = "fueltank";
|
|
self.netname = "fuel tank";
|
|
self.movetype = MOVETYPE_STEP;
|
|
self = oself;
|
|
};
|
|
|
|
void() FuelTankLarge =
|
|
{
|
|
local entity item, oself;
|
|
precache_model ("maps/b_explob.bsp");
|
|
|
|
item = spawn();
|
|
oself = self;
|
|
self = item;
|
|
setmodel (self, "maps/b_explob.bsp");
|
|
supplies += 1;
|
|
self.rtime = 1;
|
|
self.solid = SOLID_SLIDEBOX;
|
|
setsize (self, '-16 -16 -16', '16 16 48');
|
|
self.health = 50;
|
|
self.th_die = fueltank_explode;
|
|
self.takedamage = DAMAGE_AIM;
|
|
self.classname = "fueltank";
|
|
self.netname = "fuel tank";
|
|
self.movetype = MOVETYPE_STEP;
|
|
self = oself;
|
|
};
|
|
|
|
|
|
void(vector pos) obj_fueltank =
|
|
{
|
|
FuelTankLarge();
|
|
|
|
if (random()<0.25)
|
|
FuelTankSmall(pos + '0 64 0');
|
|
if (random()<0.25)
|
|
FuelTankSmall(pos + '0 -64 0');
|
|
if (random()<0.25)
|
|
FuelTankSmall(pos + '64 0 0');
|
|
if (random()<0.25)
|
|
FuelTankSmall(pos + '-64 0 0');
|
|
};
|
|
|
|
|
|
void () food_touch =
|
|
{
|
|
if (other.classname != "player")
|
|
return;
|
|
|
|
if (self.rtime > 0)
|
|
{
|
|
self.rtime = 0;
|
|
bprint(2, "food crate has been secured.\n");
|
|
|
|
other.score = other.score + 10;
|
|
|
|
if (random()<0.25)
|
|
sound (self, CHAN_BODY, "effects/radio1.wav", 1, ATTN_NONE);
|
|
else if (random()<0.25)
|
|
sound (self, CHAN_BODY, "effects/radio2.wav", 1, ATTN_NONE);
|
|
else if (random()<0.25)
|
|
sound (self, CHAN_BODY, "effects/radio3.wav", 1, ATTN_NONE);
|
|
else
|
|
sound (self, CHAN_BODY, "effects/radio4.wav", 1, ATTN_NONE);
|
|
|
|
supplies = supplies - 1;
|
|
spawn_excla(self, 3000);
|
|
}
|
|
|
|
};
|
|
|
|
void() FoodCrate =
|
|
{
|
|
setmodel (self, "maps/crate.bsp");
|
|
self.touch = food_touch;
|
|
supplies += 1;
|
|
self.rtime = 1;
|
|
self.solid = SOLID_SLIDEBOX;
|
|
setsize (self, '-16 -16 0', '16 16 16');
|
|
self.takedamage = DAMAGE_NO;
|
|
self.touch = food_touch;
|
|
self.movetype = MOVETYPE_STEP;
|
|
self.classname = "foodcrate";
|
|
};
|
|
|
|
void() treasure_touch =
|
|
{
|
|
if (other.classname != "player")
|
|
return;
|
|
if (self.picking > 0)
|
|
return;
|
|
|
|
if (self.attack_finished < time)
|
|
{
|
|
sound (self, CHAN_BODY, "items/locked.wav", 1, ATTN_NORM);
|
|
sprint(other, 2, "this metal box is locked.\n");
|
|
if (other.skill_sneak >= 1)
|
|
{
|
|
sprint(other, 2, "you may attempt to pick this\n");
|
|
sprint(other, 2, "by pressing your 'action' key\n");
|
|
}
|
|
}
|
|
self.attack_finished = time + 2;
|
|
return;
|
|
};
|
|
|
|
|
|
void() bomb_touch =
|
|
{
|
|
if (other.classname != "player")
|
|
return;
|
|
if (self.picking > 0)
|
|
return;
|
|
|
|
if (self.attack_finished < time)
|
|
{
|
|
sound (self, CHAN_BODY, "items/locked.wav", 1, ATTN_NORM);
|
|
sprint(other, 2, "this is a high-tech bomb.\n");
|
|
if (other.class == 4)
|
|
{
|
|
sprint(other, 2, "you may attempt to deactivate this\n");
|
|
sprint(other, 2, "bomb by pressing your 'action' key\n");
|
|
}
|
|
}
|
|
self.attack_finished = time + 2;
|
|
return;
|
|
};
|
|
|
|
void() Explosive =
|
|
{
|
|
if (random()*100<=25)
|
|
{
|
|
remove(self);
|
|
return;
|
|
}
|
|
|
|
setmodel (self, "progs/bomb.mdl");
|
|
self.touch = bomb_touch;
|
|
self.solid = SOLID_SLIDEBOX;
|
|
setsize (self, '-16 -16 0', '16 16 16');
|
|
self.takedamage = DAMAGE_NO;
|
|
self.movetype = MOVETYPE_STEP;
|
|
self.classname = "bomb";
|
|
droptofloor();
|
|
};
|
|
|
|
void() TreasureChest =
|
|
{
|
|
setorigin(self, self.origin + '0 0 64');
|
|
self.movetype = MOVETYPE_BOUNCE;
|
|
self.velocity_x = crandom()*200;
|
|
self.velocity_y = crandom()*200;
|
|
self.velocity_z = -200;
|
|
|
|
self.avelocity_y = crandom()*200;
|
|
|
|
setmodel (self, "progs/ammobox2.mdl");
|
|
self.touch = treasure_touch;
|
|
self.solid = SOLID_BBOX;
|
|
setsize (self, '-12 -12 0', '12 12 12');
|
|
self.takedamage = DAMAGE_NO;
|
|
self.classname = "treasure_chest";
|
|
};
|
|
|
|
void() car_rock =
|
|
{
|
|
self.frame = self.frame + 1;
|
|
|
|
if (self.frame >= 11)
|
|
self.frame = 0;
|
|
|
|
self.nextthink = time + 0.1 + (self.cnt*0.025);
|
|
self.think = car_rock;
|
|
self.cnt = self.cnt + 1;
|
|
};
|
|
|
|
void() car_rock2 =
|
|
{
|
|
if (random()<0.5)
|
|
sound (self, CHAN_BODY, "effects/metal-1.wav", 1, ATTN_NORM);
|
|
else
|
|
sound (self, CHAN_BODY, "effects/metal-2.wav", 1, ATTN_NORM);
|
|
};
|
|
|
|
void() car_touch =
|
|
{
|
|
if (other.classname != "player")
|
|
return;
|
|
|
|
if (self.attack_finished < time)
|
|
{
|
|
self.think = car_rock;
|
|
self.nextthink = time + 0.1;
|
|
self.cnt = 1;
|
|
}
|
|
self.attack_finished = time + 4;
|
|
return;
|
|
};
|
|
|
|
void() BlownUpCar =
|
|
{
|
|
setmodel (self, "progs/BlownUp.mdl");
|
|
self.origin = self.origin + '0 0 32';
|
|
self.touch = car_touch;
|
|
self.solid = SOLID_SLIDEBOX;
|
|
self.helmet = 2;
|
|
setsize (self, '-32 -32 0', '32 32 32');
|
|
self.islot3 = SlotVal(IID_ARM_METAL, 1);
|
|
self.takedamage = DAMAGE_AIM;
|
|
self.health = 90000;
|
|
self.th_pain = car_rock2;
|
|
self.movetype = MOVETYPE_BOUNCE;
|
|
self.classname = "car";
|
|
};
|
|
|
|
/*QUAKED noclass (0 0 0) (-8 -8 -8) (8 8 8)
|
|
prints a warning message when spawned
|
|
376 5857
|
|
354 8501
|
|
*/
|
|
void() noclass =
|
|
{
|
|
dprint ("noclass spawned at");
|
|
dprint (vtos(self.origin));
|
|
dprint ("\n");
|
|
remove (self);
|
|
};
|
|
|
|
void() q_touch;
|
|
|
|
void() q_touch =
|
|
{
|
|
local string s;
|
|
|
|
if (other.classname != "player")
|
|
return;
|
|
if (other.health <= 0)
|
|
return;
|
|
|
|
self.mdl = self.model;
|
|
|
|
sound (other, CHAN_VOICE, self.noise, 1, ATTN_NORM);
|
|
stuffcmd (other, "bf\n");
|
|
self.solid = SOLID_NOT;
|
|
other.items = other.items | IT_QUAD;
|
|
self.model = string_null;
|
|
if (deathmatch == 4)
|
|
{
|
|
other.armortype = 0;
|
|
other.armorvalue = 0 * 0.01;
|
|
other.ammo_cells = 0;
|
|
}
|
|
|
|
// do the apropriate action
|
|
other.super_time = 1;
|
|
other.super_damage_finished = self.cnt;
|
|
|
|
s=ftos(rint(other.super_damage_finished - time));
|
|
|
|
bprint (PRINT_LOW, other.netname);
|
|
if (deathmatch == 4)
|
|
bprint (PRINT_LOW, " recovered an OctaPower with ");
|
|
else
|
|
bprint (PRINT_LOW, " recovered a Quad with ");
|
|
bprint (PRINT_LOW, s);
|
|
bprint (PRINT_LOW, " seconds remaining!\n");
|
|
|
|
activator = other;
|
|
SUB_UseTargets(); // fire all targets / killtargets
|
|
};
|
|
|
|
|
|
void(float timeleft) DropQuad =
|
|
{
|
|
local entity item;
|
|
|
|
item = spawn();
|
|
item.origin = self.origin;
|
|
|
|
item.velocity_z = 300;
|
|
item.velocity_x = -100 + (random() * 200);
|
|
item.velocity_y = -100 + (random() * 200);
|
|
|
|
item.flags = FL_ITEM;
|
|
item.solid = SOLID_TRIGGER;
|
|
item.movetype = MOVETYPE_TOSS;
|
|
item.noise = "items/damage.wav";
|
|
setmodel (item, "progs/quaddama.mdl");
|
|
setsize (item, '-16 -16 -24', '16 16 32');
|
|
item.cnt = time + timeleft;
|
|
item.touch = q_touch;
|
|
item.nextthink = time + timeleft; // remove it with the time left on it
|
|
item.think = SUB_Remove;
|
|
};
|
|
|
|
|
|
void() r_touch;
|
|
|
|
void () BombBeep =
|
|
{
|
|
local float r;
|
|
|
|
r = range (self.enemy);
|
|
|
|
if (r != RANGE_MELEE || self.enemy.health <= 0)
|
|
{
|
|
self.owner.picking = 0;
|
|
self.enemy.picking = 0;
|
|
self.enemy.currentmenu = "none";
|
|
remove(self.tumbler1);
|
|
remove(self.tumbler2);
|
|
remove(self);
|
|
return;
|
|
}
|
|
|
|
self.inplace = 0;
|
|
self.tumbler1.inplace = 0;
|
|
self.tumbler2.inplace = 0;
|
|
self.model = "";
|
|
self.tumbler1.model = "";
|
|
self.tumbler2.model = "";
|
|
|
|
if (random() < 0.75)
|
|
{
|
|
self.inplace = 1;
|
|
self.model = "progs/radio.spr";
|
|
}
|
|
if (random() < 0.75)
|
|
{
|
|
self.tumbler1.inplace = 1;
|
|
self.tumbler1.model = "progs/radio.spr";
|
|
}
|
|
if (random() < 0.75)
|
|
{
|
|
self.tumbler2.inplace = 1;
|
|
self.tumbler2.model = "progs/radio.spr";
|
|
}
|
|
|
|
|
|
self.think = BombBeep;
|
|
self.nextthink = time + 0.3+random()*0.6;
|
|
self.enemy.chest = self;
|
|
self.enemy.currentmenu = "menu_defuse";
|
|
|
|
if (random() < 0.5)
|
|
sound (self, CHAN_BODY, "items/lockpick1.wav", 1, ATTN_NORM);
|
|
else
|
|
sound (self, CHAN_BODY, "items/lockpick2.wav", 1, ATTN_NORM);
|
|
|
|
};
|
|
|
|
|
|
void (entity portal, entity toucher) SpawnBomb =
|
|
{
|
|
local entity open, t1, t2;
|
|
|
|
makevectors (toucher.v_angle);
|
|
|
|
open = spawn();
|
|
setmodel(open, "progs/radio.spr");
|
|
setorigin (open, toucher.origin + '0 0 48' + v_forward*64);
|
|
setsize (open, VEC_ORIGIN, VEC_ORIGIN);
|
|
open.think = BombBeep;
|
|
open.nextthink = time + random()*0.2;
|
|
open.enemy = toucher;
|
|
open.owner = portal;
|
|
open.owner.rtime = 3;
|
|
|
|
t1 = spawn();
|
|
setmodel(t1, "progs/radio.spr");
|
|
setorigin (t1, toucher.origin + '0 0 48' + v_forward*64 + v_right*32);
|
|
setsize (t1, VEC_ORIGIN, VEC_ORIGIN);
|
|
t1.enemy = toucher;
|
|
t1.owner = portal;
|
|
|
|
t2 = spawn();
|
|
setmodel(t2, "progs/radio.spr");
|
|
setorigin (t2, toucher.origin + '0 0 48' + v_forward*64 - v_right*32);
|
|
setsize (t2, VEC_ORIGIN, VEC_ORIGIN);
|
|
t2.enemy = toucher;
|
|
t2.owner = portal;
|
|
|
|
open.tumbler1 = t1;
|
|
open.tumbler2 = t2;
|
|
};
|
|
|
|
void () OpenChestBeep =
|
|
{
|
|
local float r;
|
|
|
|
r = range (self.enemy);
|
|
|
|
if (r != RANGE_MELEE || self.enemy.health <= 0)
|
|
{
|
|
self.owner.picking = 0;
|
|
self.enemy.picking = 0;
|
|
self.enemy.currentmenu = "none";
|
|
remove(self.tumbler1);
|
|
remove(self.tumbler2);
|
|
remove(self);
|
|
return;
|
|
}
|
|
|
|
self.inplace = 0;
|
|
self.tumbler1.inplace = 0;
|
|
self.tumbler2.inplace = 0;
|
|
self.model = "";
|
|
self.tumbler1.model = "";
|
|
self.tumbler2.model = "";
|
|
|
|
if (random() < 0.5)
|
|
{
|
|
self.inplace = 1;
|
|
self.model = "progs/radio.spr";
|
|
}
|
|
if (random() < 0.5)
|
|
{
|
|
self.tumbler1.inplace = 1;
|
|
self.tumbler1.model = "progs/radio.spr";
|
|
}
|
|
if (random() < 0.5)
|
|
{
|
|
self.tumbler2.inplace = 1;
|
|
self.tumbler2.model = "progs/radio.spr";
|
|
}
|
|
|
|
|
|
self.think = OpenChestBeep;
|
|
self.nextthink = time + 0.1+random()*0.4 + (0.02*self.enemy.skill_sneak);
|
|
self.enemy.chest = self;
|
|
self.enemy.currentmenu = "menu_lockpick";
|
|
|
|
if (random() < 0.5)
|
|
sound (self, CHAN_BODY, "items/lockpick1.wav", 1, ATTN_NORM);
|
|
else
|
|
sound (self, CHAN_BODY, "items/lockpick2.wav", 1, ATTN_NORM);
|
|
|
|
};
|
|
|
|
|
|
void (entity portal, entity toucher) SpawnOpenChest =
|
|
{
|
|
local entity open, t1, t2;
|
|
|
|
makevectors (toucher.v_angle);
|
|
|
|
open = spawn();
|
|
setmodel(open, "progs/radio.spr");
|
|
setorigin (open, toucher.origin + '0 0 48' + v_forward*64);
|
|
setsize (open, VEC_ORIGIN, VEC_ORIGIN);
|
|
open.think = OpenChestBeep;
|
|
open.nextthink = time + random()*0.2;
|
|
open.enemy = toucher;
|
|
open.owner = portal;
|
|
|
|
t1 = spawn();
|
|
setmodel(t1, "progs/radio.spr");
|
|
setorigin (t1, toucher.origin + '0 0 48' + v_forward*64 + v_right*32);
|
|
setsize (t1, VEC_ORIGIN, VEC_ORIGIN);
|
|
t1.enemy = toucher;
|
|
t1.owner = portal;
|
|
|
|
t2 = spawn();
|
|
setmodel(t2, "progs/radio.spr");
|
|
setorigin (t2, toucher.origin + '0 0 48' + v_forward*64 - v_right*32);
|
|
setsize (t2, VEC_ORIGIN, VEC_ORIGIN);
|
|
t2.enemy = toucher;
|
|
t2.owner = portal;
|
|
|
|
open.tumbler1 = t1;
|
|
open.tumbler2 = t2;
|
|
};
|
|
|
|
|
|
void() r_touch =
|
|
{
|
|
local string s;
|
|
|
|
if (other.classname != "player")
|
|
return;
|
|
if (other.health <= 0)
|
|
return;
|
|
|
|
self.mdl = self.model;
|
|
|
|
sound (other, CHAN_VOICE, self.noise, 1, ATTN_NORM);
|
|
stuffcmd (other, "bf\n");
|
|
self.solid = SOLID_NOT;
|
|
other.items = other.items | IT_INVISIBILITY;
|
|
self.model = string_null;
|
|
|
|
// do the apropriate action
|
|
other.invisible_time = 1;
|
|
other.invisible_finished = self.cnt;
|
|
s=ftos(rint(other.invisible_finished - time));
|
|
bprint (PRINT_LOW, other.netname);
|
|
bprint (PRINT_LOW, " recovered a Ring with ");
|
|
bprint (PRINT_LOW, s);
|
|
bprint (PRINT_LOW, " seconds remaining!\n");
|
|
|
|
|
|
activator = other;
|
|
SUB_UseTargets(); // fire all targets / killtargets
|
|
};
|
|
|
|
|
|
void(float timeleft) DropRing =
|
|
{
|
|
local entity item;
|
|
|
|
item = spawn();
|
|
item.origin = self.origin;
|
|
|
|
item.velocity_z = 300;
|
|
item.velocity_x = -100 + (random() * 200);
|
|
item.velocity_y = -100 + (random() * 200);
|
|
|
|
item.flags = FL_ITEM;
|
|
item.solid = SOLID_TRIGGER;
|
|
item.movetype = MOVETYPE_TOSS;
|
|
item.noise = "items/inv1.wav";
|
|
setmodel (item, "progs/invisibl.mdl");
|
|
setsize (item, '-16 -16 -24', '16 16 32');
|
|
item.cnt = time + timeleft;
|
|
item.touch = r_touch;
|
|
item.nextthink = time + timeleft; // remove after 30 seconds
|
|
item.think = SUB_Remove;
|
|
};
|
|
|
|
/*
|
|
============
|
|
PlaceItem
|
|
|
|
plants the object on the floor
|
|
============
|
|
*/
|
|
void() PlaceItem =
|
|
{
|
|
local float oldz;
|
|
|
|
self.mdl = self.model; // so it can be restored on respawn
|
|
self.flags = FL_ITEM; // make extra wide
|
|
self.solid = SOLID_TRIGGER;
|
|
self.movetype = MOVETYPE_TOSS;
|
|
self.velocity = '0 0 0';
|
|
self.origin_z = self.origin_z + 6;
|
|
oldz = self.origin_z;
|
|
if (!droptofloor())
|
|
{
|
|
dprint ("Bonus item fell out of level at ");
|
|
dprint (vtos(self.origin));
|
|
dprint ("\n");
|
|
remove(self);
|
|
return;
|
|
}
|
|
};
|
|
|
|
/*
|
|
============
|
|
StartItem
|
|
|
|
Sets the clipping size and plants the object on the floor
|
|
============
|
|
*/
|
|
void() StartItem =
|
|
{
|
|
self.nextthink = time + 0.2; // items start after other solids
|
|
self.think = PlaceItem;
|
|
};
|
|
|
|
/*
|
|
=========================================================================
|
|
|
|
HEALTH BOX
|
|
|
|
=========================================================================
|
|
*/
|
|
//
|
|
// T_Heal: add health to an entity, limiting health to max_health
|
|
// "ignore" will ignore max_health limit
|
|
//
|
|
float (entity e, float healamount, float ignore) T_Heal =
|
|
{
|
|
if (e.health <= 0)
|
|
return 0;
|
|
if ((!ignore) && (e.health >= other.max_health))
|
|
return 0;
|
|
healamount = ceil(healamount);
|
|
|
|
e.health = e.health + healamount;
|
|
if ((!ignore) && (e.health >= other.max_health))
|
|
e.health = other.max_health;
|
|
|
|
if (e.health > 250)
|
|
e.health = 250;
|
|
return 1;
|
|
};
|
|
|
|
/*QUAKED item_health (.3 .3 1) (0 0 0) (32 32 32) rotten megahealth
|
|
Health box. Normally gives 25 points.
|
|
Rotten box heals 5-10 points,
|
|
megahealth will add 100 health, then
|
|
rot you down to your maximum health limit,
|
|
one point per second.
|
|
*/
|
|
|
|
float H_ROTTEN = 1;
|
|
float H_MEGA = 2;
|
|
.float healamount, healtype;
|
|
void() health_touch;
|
|
void() item_megahealth_rot;
|
|
|
|
void() item_health =
|
|
{
|
|
|
|
if (self.spawnflags & H_MEGA)
|
|
TreasureChest();
|
|
else
|
|
{
|
|
if (random()*100<=75)
|
|
{
|
|
remove(self);
|
|
return;
|
|
}
|
|
|
|
self.touch = health_touch;
|
|
precache_model("maps/b_bh25.bsp");
|
|
precache_sound("items/health1.wav");
|
|
setmodel(self, "maps/b_bh25.bsp");
|
|
self.noise = "items/health1.wav";
|
|
self.healamount = 25;
|
|
self.healtype = 1;
|
|
setsize (self, '0 0 0', '32 32 56');
|
|
StartItem ();
|
|
}
|
|
};
|
|
|
|
|
|
float (entity to, float iid, float quant) TryGiveStackable =
|
|
{
|
|
local float slot;
|
|
local float item;
|
|
local float x;
|
|
|
|
x = 300;
|
|
|
|
if (ToIID(self.islot4) == IID_EQUIP_BACKPACK)
|
|
x = 600;
|
|
|
|
slot = SlotOfItem(to, iid);
|
|
if (slot > 4)
|
|
{
|
|
item = ItemInSlot(to, slot);
|
|
if (ToStatus(item) + quant > x) //300 is our maxstack value here
|
|
{
|
|
quant = quant + ToStatus(item) - x;
|
|
SetItemSlot(to, slot, SlotVal(iid, x));
|
|
slot = 0;
|
|
return false;
|
|
}
|
|
else
|
|
quant = quant + ToStatus(item);
|
|
}
|
|
if (quant == 0)
|
|
return true;
|
|
if (slot == 0)
|
|
{
|
|
slot = FindSuitableEmptySlot(to, iid);
|
|
if (slot == 0)
|
|
return false;
|
|
}
|
|
|
|
SetItemSlot(to, slot, SlotVal(iid, quant));
|
|
|
|
return true;
|
|
};
|
|
|
|
|
|
void () health_touch =
|
|
{
|
|
if (!triggercantouch(self, other))
|
|
return;
|
|
|
|
if (other.classname != "player" || other.ghost != 0)
|
|
return;
|
|
|
|
if (other.classname == "player")
|
|
{
|
|
if (self.healtype == H_MEGA)
|
|
return;
|
|
|
|
if (other.ghost != 0)
|
|
return;
|
|
|
|
if (!TryGiveStackable(other, IID_CHEM_STIMPACK, 1))
|
|
return;
|
|
|
|
sprint (other, PRINT_HIGH, "picked up a stimpack.\n");
|
|
|
|
if (random()*4 <= 2)
|
|
sound (other, CHAN_ITEM, "misc/item1.wav", 1, ATTN_NORM);
|
|
else
|
|
sound (other, CHAN_ITEM, "misc/item2.wav", 1, ATTN_NORM);
|
|
|
|
self.model = string_null;
|
|
self.solid = SOLID_NOT;
|
|
return;
|
|
}
|
|
|
|
activator = other;
|
|
SUB_UseTargets ();
|
|
};
|
|
|
|
void() item_megahealth_rot =
|
|
{
|
|
other = self.owner;
|
|
|
|
if (other.health > other.max_health)
|
|
{
|
|
other.health = other.health - 1;
|
|
self.nextthink = time + 1;
|
|
return;
|
|
}
|
|
|
|
// it is possible for a player to die and respawn between rots, so don't
|
|
// just blindly subtract the flag off
|
|
other.items = other.items - (other.items & IT_SUPERHEALTH);
|
|
|
|
if (deathmatch != 2) // deathmatch 2 is silly old rules
|
|
{
|
|
self.nextthink = time + 20;
|
|
self.think = SUB_regen;
|
|
}
|
|
};
|
|
|
|
/*
|
|
===============================================================================
|
|
|
|
ARMOR
|
|
|
|
===============================================================================
|
|
*/
|
|
|
|
void() armor_touch;
|
|
|
|
void() armor_touch =
|
|
{
|
|
local float type, value, bit;
|
|
|
|
if (!triggercantouch(self, other))
|
|
return;
|
|
|
|
if (other.health <= 0)
|
|
return;
|
|
if (other.classname != "player")
|
|
return;
|
|
|
|
if (deathmatch == 4)
|
|
if (other.invincible_time > 0)
|
|
return;
|
|
|
|
if (self.classname == "item_armor1")
|
|
{
|
|
type = 0.3;
|
|
value = 100;
|
|
bit = IT_ARMOR1;
|
|
}
|
|
if (self.classname == "item_armor2")
|
|
{
|
|
type = 0.6;
|
|
value = 150;
|
|
bit = IT_ARMOR2;
|
|
}
|
|
if (self.classname == "item_armorInv")
|
|
{
|
|
type = 0.8;
|
|
value = 200;
|
|
bit = IT_ARMOR3;
|
|
}
|
|
if (other.armortype*other.armorvalue >= type*value)
|
|
return;
|
|
|
|
other.armortype = type;
|
|
other.armorvalue = value;
|
|
other.items = other.items - (other.items & (IT_ARMOR1 | IT_ARMOR2 | IT_ARMOR3)) + bit;
|
|
|
|
self.solid = SOLID_NOT;
|
|
self.model = string_null;
|
|
if (deathmatch != 2)
|
|
self.nextthink = time + 20;
|
|
self.think = SUB_regen;
|
|
|
|
sprint(other, PRINT_LOW, "You got armor\n");
|
|
// armor touch sound
|
|
sound(other, CHAN_ITEM, "items/armor1.wav", 1, ATTN_NORM);
|
|
stuffcmd (other, "bf\n");
|
|
|
|
activator = other;
|
|
SUB_UseTargets(); // fire all targets / killtargets
|
|
};
|
|
|
|
|
|
/*QUAKED item_armor1 (0 .5 .8) (-16 -16 0) (16 16 32)
|
|
*/
|
|
|
|
void() buyzone1 =
|
|
{
|
|
SUB_UseTargets ();
|
|
|
|
|
|
self.touch = SUB_Null;
|
|
self.solid = SOLID_BBOX;
|
|
self.movetype = MOVETYPE_STEP;
|
|
setmodel (self, "progs/enforcer.mdl");
|
|
self.skin = 0;
|
|
setsize (self, '-16 -16 -24', '16 16 32');
|
|
setorigin(self, self.origin + '0 0 24');
|
|
self.classname = "merchant";
|
|
|
|
};
|
|
|
|
void() buyzone2 =
|
|
{
|
|
SUB_UseTargets ();
|
|
|
|
|
|
self.touch = SUB_Null;
|
|
self.solid = SOLID_BBOX;
|
|
setmodel (self, "progs/enforcer.mdl");
|
|
self.skin = 0;
|
|
setsize (self, '-16 -16 -24', '16 16 32');
|
|
setorigin(self, self.origin + '0 0 24');
|
|
self.classname = "merchant";
|
|
};
|
|
|
|
void() item_armor1_finish =
|
|
{
|
|
SUB_UseTargets ();
|
|
|
|
|
|
self.touch = SUB_Null;
|
|
self.solid = SOLID_BBOX;
|
|
self.movetype = MOVETYPE_STEP;
|
|
setmodel (self, "progs/enforcer.mdl");
|
|
self.skin = 0;
|
|
setsize (self, '-16 -16 -24', '16 16 32');
|
|
setorigin(self, self.origin + '0 0 24');
|
|
self.classname = "merchant";
|
|
self.think = shop_face;
|
|
self.nextthink = time + 0.5;
|
|
};
|
|
void() item_armor1 =
|
|
{
|
|
if (random()<0.50)
|
|
{
|
|
precache_model ("progs/enforcer.mdl");
|
|
self.think = item_armor1_finish;
|
|
self.nextthink = time + 0.1;
|
|
}
|
|
};
|
|
|
|
/*QUAKED item_armor2 (0 .5 .8) (-16 -16 0) (16 16 32)
|
|
*/
|
|
|
|
void() item_armor2 =
|
|
{
|
|
if (random()<0.50)
|
|
{
|
|
precache_model ("progs/enforcer.mdl");
|
|
self.touch = SUB_Null;
|
|
self.solid = SOLID_BBOX;
|
|
setmodel (self, "progs/enforcer.mdl");
|
|
self.skin = 0;
|
|
setsize (self, '-16 -16 -24', '16 16 32');
|
|
setorigin(self, self.origin + '0 0 24');
|
|
self.classname = "merchant";
|
|
self.think = shop_face;
|
|
self.nextthink = time + 0.5;
|
|
}
|
|
};
|
|
|
|
/*QUAKED item_armorInv (0 .5 .8) (-16 -16 0) (16 16 32)
|
|
*/
|
|
|
|
void() item_armorInv =
|
|
{
|
|
if (random()<0.50)
|
|
{
|
|
precache_model ("progs/enforcer.mdl");
|
|
self.touch = SUB_Null;
|
|
self.solid = SOLID_BBOX;
|
|
setmodel (self, "progs/enforcer.mdl");
|
|
self.skin = 0;
|
|
setsize (self, '-16 -16 -24', '16 16 32');
|
|
setorigin(self, self.origin + '0 0 24');
|
|
self.classname = "merchant";
|
|
self.think = shop_face;
|
|
self.nextthink = time + 0.5;
|
|
}
|
|
};
|
|
|
|
/*
|
|
===============================================================================
|
|
|
|
WEAPONS
|
|
|
|
===============================================================================
|
|
*/
|
|
|
|
void() bound_other_ammo =
|
|
{
|
|
|
|
};
|
|
|
|
|
|
float(float w) RankForWeapon =
|
|
{
|
|
if (w == IT_LIGHTNING)
|
|
return 1;
|
|
if (w == IT_ROCKET_LAUNCHER)
|
|
return 2;
|
|
if (w == IT_SUPER_NAILGUN)
|
|
return 3;
|
|
if (w == IT_GRENADE_LAUNCHER)
|
|
return 4;
|
|
if (w == IT_SUPER_SHOTGUN)
|
|
return 5;
|
|
if (w == IT_NAILGUN)
|
|
return 6;
|
|
return 7;
|
|
};
|
|
|
|
float (float w) WeaponCode =
|
|
{
|
|
if (w == IT_SUPER_SHOTGUN)
|
|
return 3;
|
|
if (w == IT_NAILGUN)
|
|
return 4;
|
|
if (w == IT_SUPER_NAILGUN)
|
|
return 5;
|
|
if (w == IT_GRENADE_LAUNCHER)
|
|
return 6;
|
|
if (w == IT_ROCKET_LAUNCHER)
|
|
return 7;
|
|
if (w == IT_LIGHTNING)
|
|
return 8;
|
|
return 1;
|
|
};
|
|
|
|
/*
|
|
=============
|
|
Deathmatch_Weapon
|
|
|
|
Deathmatch weapon change rules for picking up a weapon
|
|
|
|
.float ammo_shells, ammo_nails, ammo_rockets, ammo_cells;
|
|
=============
|
|
*/
|
|
void(float old, float new) Deathmatch_Weapon =
|
|
{
|
|
local float or, nr;
|
|
|
|
// change self.weapon if desired
|
|
or = RankForWeapon (self.weapon);
|
|
nr = RankForWeapon (new);
|
|
if ( nr < or )
|
|
self.weapon = new;
|
|
};
|
|
|
|
/*
|
|
=============
|
|
weapon_touch
|
|
=============
|
|
*/
|
|
float() W_BestWeapon;
|
|
|
|
void() weapon_touch =
|
|
{
|
|
local float hadammo, best, new, old;
|
|
local entity stemp;
|
|
local float leave;
|
|
|
|
// For client weapon_switch
|
|
local float w_switch;
|
|
|
|
if (!triggercantouch(self, other))
|
|
return;
|
|
|
|
if (!(other.flags & FL_CLIENT))
|
|
return;
|
|
|
|
if ((stof(infokey(other,"w_switch"))) == 0)
|
|
w_switch = 8;
|
|
else
|
|
w_switch = stof(infokey(other,"w_switch"));
|
|
|
|
// if the player was using his best weapon, change up to the new one if better
|
|
stemp = self;
|
|
self = other;
|
|
best = W_BestWeapon();
|
|
self = stemp;
|
|
|
|
if (deathmatch == 2 || deathmatch == 3 || deathmatch == 5)
|
|
leave = 1;
|
|
else
|
|
leave = 0;
|
|
|
|
if (self.classname == "weapon_nailgun")
|
|
{
|
|
if (leave && (other.items & IT_NAILGUN) )
|
|
return;
|
|
hadammo = other.ammo_nails;
|
|
new = IT_NAILGUN;
|
|
other.ammo_nails = other.ammo_nails + 30;
|
|
}
|
|
else if (self.classname == "weapon_supernailgun")
|
|
{
|
|
if (leave && (other.items & IT_SUPER_NAILGUN) )
|
|
return;
|
|
hadammo = other.ammo_rockets;
|
|
new = IT_SUPER_NAILGUN;
|
|
other.ammo_nails = other.ammo_nails + 30;
|
|
}
|
|
else if (self.classname == "weapon_supershotgun")
|
|
{
|
|
if (leave && (other.items & IT_SUPER_SHOTGUN) )
|
|
return;
|
|
hadammo = other.ammo_rockets;
|
|
new = IT_SUPER_SHOTGUN;
|
|
other.ammo_shells = other.ammo_shells + 5;
|
|
}
|
|
else if (self.classname == "weapon_rocketlauncher")
|
|
{
|
|
if (leave && (other.items & IT_ROCKET_LAUNCHER) )
|
|
return;
|
|
hadammo = other.ammo_rockets;
|
|
new = IT_ROCKET_LAUNCHER;
|
|
other.ammo_rockets = other.ammo_rockets + 5;
|
|
}
|
|
else if (self.classname == "weapon_grenadelauncher")
|
|
{
|
|
if (leave && (other.items & IT_GRENADE_LAUNCHER) )
|
|
return;
|
|
hadammo = other.ammo_rockets;
|
|
new = IT_GRENADE_LAUNCHER;
|
|
other.ammo_rockets = other.ammo_rockets + 5;
|
|
}
|
|
else if (self.classname == "weapon_lightning")
|
|
{
|
|
if (leave && (other.items & IT_LIGHTNING))
|
|
return;
|
|
hadammo = other.ammo_rockets;
|
|
new = IT_LIGHTNING;
|
|
other.ammo_cells = other.ammo_cells + 15;
|
|
}
|
|
else
|
|
objerror ("weapon_touch: unknown classname");
|
|
|
|
sprint (other, PRINT_LOW, "You got the ");
|
|
sprint (other, PRINT_LOW, self.netname);
|
|
sprint (other, PRINT_LOW, "\n");
|
|
// weapon touch sound
|
|
sound (other, CHAN_ITEM, "weapons/pkup.wav", 1, ATTN_NORM);
|
|
stuffcmd (other, "bf\n");
|
|
|
|
bound_other_ammo ();
|
|
|
|
// change to the weapon
|
|
old = other.items;
|
|
other.items = other.items | new;
|
|
|
|
stemp = self;
|
|
self = other;
|
|
|
|
if ( WeaponCode(new) <= w_switch )
|
|
{
|
|
if (self.flags & FL_INWATER)
|
|
{
|
|
if (new != IT_LIGHTNING)
|
|
{
|
|
Deathmatch_Weapon (old, new);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Deathmatch_Weapon (old, new);
|
|
}
|
|
}
|
|
|
|
W_SetCurrentAmmo();
|
|
|
|
self = stemp;
|
|
|
|
if (leave)
|
|
return;
|
|
|
|
if (deathmatch!=3 || deathmatch !=5)
|
|
{
|
|
// remove it in single player, or setup for respawning in deathmatch
|
|
self.model = string_null;
|
|
self.solid = SOLID_NOT;
|
|
if (deathmatch != 2)
|
|
self.nextthink = time + 30;
|
|
self.think = SUB_regen;
|
|
}
|
|
activator = other;
|
|
SUB_UseTargets(); // fire all targets / killtargets
|
|
};
|
|
|
|
|
|
void(vector place) hostage_new;
|
|
|
|
/*QUAKED weapon_supershotgun (0 .5 .8) (-16 -16 0) (16 16 32)
|
|
*/
|
|
|
|
|
|
|
|
void() weapon_supershotgun =
|
|
{
|
|
//weapon spawns are now either treasure chests
|
|
//or hostages that need to be rescued
|
|
|
|
if (random()<0.25)
|
|
TreasureChest();
|
|
};
|
|
|
|
/*QUAKED weapon_nailgun (0 .5 .8) (-16 -16 0) (16 16 32)
|
|
*/
|
|
|
|
void() weapon_nailgun =
|
|
{
|
|
if (world.map_obj == 2)
|
|
FoodCrate();
|
|
else if (world.map_obj == 5)
|
|
obj_fueltank(self.origin);
|
|
else if (world.map_obj == 6)
|
|
Explosive();
|
|
else
|
|
TreasureChest();
|
|
};
|
|
|
|
/*QUAKED weapon_supernailgun (0 .5 .8) (-16 -16 0) (16 16 32)
|
|
*/
|
|
|
|
void() weapon_supernailgun =
|
|
{
|
|
if (world.map_obj == 2)
|
|
FoodCrate();
|
|
else if (world.map_obj == 5)
|
|
obj_fueltank(self.origin);
|
|
else if (world.map_obj == 6)
|
|
Explosive();
|
|
else if (random()*100 <= 40)
|
|
TreasureChest();
|
|
};
|
|
|
|
/*QUAKED weapon_grenadelauncher (0 .5 .8) (-16 -16 0) (16 16 32)
|
|
*/
|
|
|
|
void() weapon_grenadelauncher =
|
|
{
|
|
|
|
if (world.map_obj == 2)
|
|
FoodCrate();
|
|
else if (world.map_obj == 5)
|
|
obj_fueltank(self.origin);
|
|
else if (world.map_obj == 6)
|
|
Explosive();
|
|
else if (random()*100 <= 40)
|
|
TreasureChest();
|
|
|
|
};
|
|
|
|
/*QUAKED weapon_rocketlauncher (0 .5 .8) (-16 -16 0) (16 16 32)
|
|
*/
|
|
|
|
void() weapon_rocketlauncher =
|
|
{
|
|
if (world.map_obj == 2)
|
|
FoodCrate();
|
|
else if (world.map_obj == 5)
|
|
obj_fueltank(self.origin);
|
|
else if (world.map_obj == 6)
|
|
Explosive();
|
|
else if (random()*100 <= 40)
|
|
TreasureChest();
|
|
|
|
};
|
|
|
|
|
|
/*QUAKED weapon_lightning (0 .5 .8) (-16 -16 0) (16 16 32)
|
|
*/
|
|
|
|
void() weapon_lightning =
|
|
{
|
|
|
|
if (world.map_obj == 2)
|
|
FoodCrate();
|
|
else if (world.map_obj == 5)
|
|
obj_fueltank(self.origin);
|
|
else if (world.map_obj == 6)
|
|
Explosive();
|
|
else if (random()*100 <= 40)
|
|
TreasureChest();
|
|
|
|
};
|
|
|
|
|
|
/*
|
|
===============================================================================
|
|
|
|
AMMO
|
|
|
|
===============================================================================
|
|
*/
|
|
|
|
float() GetRandomAmmo =
|
|
{
|
|
local float r;
|
|
|
|
r = random()*42;
|
|
|
|
if (r <= 5)
|
|
return IID_AM_10MM;
|
|
else if (r <= 10)
|
|
return IID_AM_45ACP;
|
|
else if (r <= 15)
|
|
return IID_AM_12GAUGESHELLS;
|
|
else if (r <= 20)
|
|
return IID_AM_44MAGNUM;
|
|
else if (r <= 24)
|
|
return IID_AM_556MM;
|
|
else if (r <= 28)
|
|
return IID_AM_5MMHIGHVEL;
|
|
else if (r <= 32)
|
|
return IID_AM_762MM;
|
|
else if (r <= 36)
|
|
return IID_AM_NEEDLER;
|
|
else if (r <= 38)
|
|
return IID_AM_ENERGYCELL;
|
|
else if (r <= 40)
|
|
return IID_AM_2MMEC;
|
|
else
|
|
return IID_AM_CASELESS;
|
|
};
|
|
|
|
float(float ammotype) GetAmmoCount =
|
|
{
|
|
local float ammocount;
|
|
|
|
if (ammotype == IID_AM_NEEDLER)
|
|
ammocount = 2 + random()*7;
|
|
if (ammotype == IID_AM_10MM)
|
|
ammocount = 5 + random()*5;
|
|
if (ammotype == IID_AM_556MM)
|
|
ammocount = 3 + random()*8;
|
|
if (ammotype == IID_AM_5MMHIGHVEL)
|
|
ammocount = 5 + random()*8;
|
|
if (ammotype == IID_AM_12GAUGESHELLS)
|
|
ammocount = 2 + random()*6;
|
|
if (ammotype == IID_AM_ENERGYCELL)
|
|
ammocount = 2 + random()*6;
|
|
if (ammotype == IID_AM_2MMEC)
|
|
ammocount = 1 + random()*2;
|
|
if (ammotype == IID_AM_762MM)
|
|
ammocount = 2 + random()*4;
|
|
if (ammotype == IID_AM_44MAGNUM)
|
|
ammocount = 2 + random()*4;
|
|
if (ammotype == IID_AM_45ACP)
|
|
ammocount = 5 + random()*10;
|
|
|
|
return ammocount;
|
|
};
|
|
|
|
void() ammo_touch =
|
|
{
|
|
local entity stemp;
|
|
local string ammoname, ammocountftos;
|
|
local float best, ammotype, ammocount;
|
|
|
|
if (!triggercantouch(self, other))
|
|
return;
|
|
|
|
if (other.classname != "player")
|
|
return;
|
|
if (other.health <= 0)
|
|
return;
|
|
|
|
// if the player was using his best weapon, change up to the new one if better
|
|
stemp = self;
|
|
self = other;
|
|
best = W_BestWeapon();
|
|
self = stemp;
|
|
|
|
sprint (other, 2, "a pre-war military ammo cache.\n");
|
|
sprint (other, 2, "found ");
|
|
|
|
ammotype = GetRandomAmmo();
|
|
|
|
ammocount = GetAmmoCount(ammotype);
|
|
ammocount = ceil(ammocount);
|
|
|
|
if (coop == 1 && total_players == 1)
|
|
ammocount = ammocount * 2;
|
|
|
|
ammoname = GetItemName(ammotype);
|
|
ammocountftos = ftos(ammocount);
|
|
|
|
if (random()*4 <= 2)
|
|
sound (other, CHAN_ITEM, "misc/item1.wav", 1, ATTN_NORM);
|
|
else
|
|
sound (other, CHAN_ITEM, "misc/item2.wav", 1, ATTN_NORM);
|
|
|
|
// shotgun
|
|
if (self.weapon == 1)
|
|
{
|
|
sprint (other, 2, ammoname);
|
|
sprint (other, 2, "(");
|
|
sprint (other, 2, ammocountftos);
|
|
sprint (other, 2, ") ");
|
|
|
|
TryGiveStackable(other, ammotype, ammocount);
|
|
|
|
if (random()*20 <= 10)
|
|
{
|
|
other.ammo_shells = other.ammo_shells + ceil(10+random()*10);
|
|
sprint (other, 2, "& money.\n");
|
|
}
|
|
else
|
|
sprint (other, 2, "\n");
|
|
|
|
}
|
|
|
|
// spikes
|
|
if (self.weapon == 2)
|
|
{
|
|
sprint (other, 2, ammoname);
|
|
sprint (other, 2, "(");
|
|
sprint (other, 2, ammocountftos);
|
|
sprint (other, 2, ") ");
|
|
|
|
TryGiveStackable(other, ammotype, ammocount);
|
|
|
|
if (random()*20 <= 10)
|
|
{
|
|
other.ammo_shells = other.ammo_shells + ceil(10+random()*10);
|
|
sprint (other, 2, " & money.\n");
|
|
}
|
|
else
|
|
sprint (other, 2, "\n");
|
|
}
|
|
|
|
// rockets
|
|
if (self.weapon == 3)
|
|
{
|
|
sprint (other, 2, ammoname);
|
|
sprint (other, 2, "(");
|
|
sprint (other, 2, ammocountftos);
|
|
sprint (other, 2, ") ");
|
|
|
|
TryGiveStackable(other, ammotype, ammocount);
|
|
|
|
if (random()*20 <= 10)
|
|
{
|
|
other.ammo_shells = other.ammo_shells + ceil(5+random()*25);
|
|
sprint (other, 2, " & money.\n");
|
|
}
|
|
else
|
|
sprint (other, 2, "\n");
|
|
|
|
}
|
|
|
|
// cells
|
|
if (self.weapon == 4)
|
|
{
|
|
sprint (other, 2, ammoname);
|
|
sprint (other, 2, "(");
|
|
sprint (other, 2, ammocountftos);
|
|
sprint (other, 2, ") ");
|
|
|
|
TryGiveStackable(other, ammotype, ammocount);
|
|
|
|
if (random()*20 <= 10)
|
|
{
|
|
other.ammo_shells = other.ammo_shells + ceil(5+random()*25);
|
|
sprint (other, 2, " & money.\n");
|
|
}
|
|
else
|
|
sprint (other, 2, "\n");
|
|
}
|
|
|
|
bound_other_ammo ();
|
|
|
|
|
|
|
|
// change to a better weapon if appropriate
|
|
/*
|
|
if ( other.weapon == best )
|
|
{
|
|
stemp = self;
|
|
self = other;
|
|
self.weapon = W_BestWeapon();
|
|
W_SetCurrentAmmo ();
|
|
self = stemp;
|
|
}*/
|
|
|
|
// if changed current ammo, update it
|
|
stemp = self;
|
|
self = other;
|
|
self = stemp;
|
|
|
|
// remove it in single player, or setup for respawning in deathmatch
|
|
self.model = string_null;
|
|
self.solid = SOLID_NOT;
|
|
if (coop != 1)
|
|
self.nextthink = time + 30;
|
|
|
|
// Xian -- If playing in DM 3.0 mode, halve the time ammo respawns
|
|
|
|
self.think = SUB_regen;
|
|
|
|
activator = other;
|
|
SUB_UseTargets(); // fire all targets / killtargets
|
|
};
|
|
|
|
|
|
|
|
|
|
float WEAPON_BIG2 = 1;
|
|
|
|
/*QUAKED item_shells (0 .5 .8) (0 0 0) (32 32 32) big
|
|
*/
|
|
|
|
void() item_shells =
|
|
{
|
|
if (deathmatch == 4)
|
|
return;
|
|
|
|
if (coop == 1 && random()*4 <= 2)
|
|
return;
|
|
|
|
self.islot1 = GetRandomAmmo();
|
|
self.islot1 = self.islot1 * 512;
|
|
self.flash = ceil(random()*30);
|
|
self.touch = AmmoTouch;
|
|
if (random()<0.5)
|
|
self.islot2 = ceil(random()*20);
|
|
|
|
|
|
|
|
if (self.spawnflags & WEAPON_BIG2)
|
|
{
|
|
precache_model ("maps/b_shell1.bsp");
|
|
setmodel (self, "maps/b_shell1.bsp");
|
|
self.aflag = 40;
|
|
}
|
|
else
|
|
{
|
|
precache_model ("maps/b_shell0.bsp");
|
|
setmodel (self, "maps/b_shell0.bsp");
|
|
self.aflag = 20;
|
|
}
|
|
self.weapon = 1;
|
|
self.netname = "shells";
|
|
setsize (self, '0 0 0', '32 32 56');
|
|
StartItem ();
|
|
};
|
|
|
|
/*QUAKED item_spikes (0 .5 .8) (0 0 0) (32 32 32) big
|
|
*/
|
|
|
|
void() item_spikes =
|
|
{
|
|
if (deathmatch == 4)
|
|
return;
|
|
|
|
if (coop == 1 && random()*4 <= 1)
|
|
return;
|
|
|
|
self.islot1 = GetRandomAmmo();
|
|
self.islot1 = self.islot1 * 512;
|
|
self.flash = ceil(random()*30);
|
|
self.touch = AmmoTouch;
|
|
if (random()<0.5)
|
|
self.islot2 = ceil(random()*20);
|
|
|
|
|
|
if (self.spawnflags & WEAPON_BIG2)
|
|
{
|
|
precache_model ("maps/b_nail1.bsp");
|
|
setmodel (self, "maps/b_nail1.bsp");
|
|
self.aflag = 50;
|
|
}
|
|
else
|
|
{
|
|
precache_model ("maps/b_nail0.bsp");
|
|
setmodel (self, "maps/b_nail0.bsp");
|
|
self.aflag = 25;
|
|
}
|
|
self.weapon = 2;
|
|
self.netname = "nails";
|
|
setsize (self, '0 0 0', '32 32 56');
|
|
StartItem ();
|
|
|
|
};
|
|
|
|
/*QUAKED item_rockets (0 .5 .8) (0 0 0) (32 32 32) big
|
|
*/
|
|
|
|
void() item_rockets =
|
|
{
|
|
if (deathmatch == 4)
|
|
return;
|
|
|
|
if (coop == 1 && random()*4 <= 3)
|
|
return;
|
|
|
|
self.islot1 = GetRandomAmmo();
|
|
self.islot1 = self.islot1 * 512;
|
|
self.flash = ceil(random()*30);
|
|
self.touch = AmmoTouch;
|
|
if (random()<0.5)
|
|
self.islot2 = ceil(random()*20);
|
|
|
|
|
|
if (self.spawnflags & WEAPON_BIG2)
|
|
{
|
|
precache_model ("maps/b_rock1.bsp");
|
|
setmodel (self, "maps/b_rock1.bsp");
|
|
self.aflag = 10;
|
|
}
|
|
else
|
|
{
|
|
precache_model ("maps/b_rock0.bsp");
|
|
setmodel (self, "maps/b_rock0.bsp");
|
|
self.aflag = 5;
|
|
}
|
|
self.weapon = 3;
|
|
self.netname = "rockets";
|
|
setsize (self, '0 0 0', '32 32 56');
|
|
StartItem ();
|
|
|
|
};
|
|
|
|
|
|
/*QUAKED item_cells (0 .5 .8) (0 0 0) (32 32 32) big
|
|
*/
|
|
|
|
void() item_cells =
|
|
{
|
|
if (deathmatch == 4)
|
|
return;
|
|
|
|
if (coop == 1 && random()*4 <= 2)
|
|
return;
|
|
|
|
self.islot1 = GetRandomAmmo();
|
|
self.islot1 = self.islot1 * 512;
|
|
self.flash = ceil(random()*30);
|
|
self.touch = AmmoTouch;
|
|
if (random()<0.5)
|
|
self.islot2 = ceil(random()*20);
|
|
|
|
|
|
if (self.spawnflags & WEAPON_BIG2)
|
|
{
|
|
precache_model ("maps/b_batt1.bsp");
|
|
setmodel (self, "maps/b_batt1.bsp");
|
|
self.aflag = 12;
|
|
}
|
|
else
|
|
{
|
|
precache_model ("maps/b_batt0.bsp");
|
|
setmodel (self, "maps/b_batt0.bsp");
|
|
self.aflag = 6;
|
|
}
|
|
self.weapon = 4;
|
|
self.netname = "cells";
|
|
setsize (self, '0 0 0', '32 32 56');
|
|
StartItem ();
|
|
|
|
};
|
|
|
|
|
|
/*QUAKED item_weapon (0 .5 .8) (0 0 0) (32 32 32) shotgun rocket spikes big
|
|
DO NOT USE THIS!!!! IT WILL BE REMOVED!
|
|
*/
|
|
|
|
float WEAPON_SHOTGUN = 1;
|
|
float WEAPON_ROCKET = 2;
|
|
float WEAPON_SPIKES = 4;
|
|
float WEAPON_BIG = 8;
|
|
void() item_weapon =
|
|
{
|
|
if (coop == 1 && random()*4 <= 3)
|
|
return;
|
|
|
|
self.islot1 = GetRandomAmmo();
|
|
self.islot1 = self.islot1 * 512;
|
|
self.flash = ceil(random()*30);
|
|
self.touch = AmmoTouch;
|
|
if (random()<0.5)
|
|
self.islot2 = ceil(random()*20);
|
|
|
|
precache_model ("progs/ammobox.mdl");
|
|
setmodel (self, "progs/ammobox.mdl");
|
|
self.skin = 2;
|
|
|
|
setsize (self, '0 0 0', '32 32 56');
|
|
StartItem ();
|
|
};
|
|
|
|
|
|
/*
|
|
===============================================================================
|
|
|
|
KEYS
|
|
|
|
===============================================================================
|
|
*/
|
|
|
|
void() key_touch =
|
|
{
|
|
if (!triggercantouch(self, other))
|
|
return;
|
|
|
|
if (other.classname != "player")
|
|
return;
|
|
if (other.health <= 0)
|
|
return;
|
|
if (other.items & self.items)
|
|
return;
|
|
|
|
sprint (other, PRINT_LOW, "You got the ");
|
|
sprint (other, PRINT_LOW, self.netname);
|
|
sprint (other,PRINT_LOW, "\n");
|
|
|
|
sound (other, CHAN_ITEM, self.noise, 1, ATTN_NORM);
|
|
stuffcmd (other, "bf\n");
|
|
other.items = other.items | self.items;
|
|
|
|
self.solid = SOLID_NOT;
|
|
self.model = string_null;
|
|
|
|
activator = other;
|
|
SUB_UseTargets(); // fire all targets / killtargets
|
|
};
|
|
|
|
|
|
void() key_setsounds =
|
|
{
|
|
if (world.worldtype == 0)
|
|
{
|
|
precache_sound ("misc/medkey.wav");
|
|
self.noise = "misc/medkey.wav";
|
|
}
|
|
if (world.worldtype == 1)
|
|
{
|
|
precache_sound ("misc/runekey.wav");
|
|
self.noise = "misc/runekey.wav";
|
|
}
|
|
if (world.worldtype == 2)
|
|
{
|
|
precache_sound2 ("misc/basekey.wav");
|
|
self.noise = "misc/basekey.wav";
|
|
}
|
|
};
|
|
|
|
/*QUAKED item_key1 (0 .5 .8) (-16 -16 -24) (16 16 32)
|
|
SILVER key
|
|
In order for keys to work
|
|
you MUST set your maps
|
|
worldtype to one of the
|
|
following:
|
|
0: medieval
|
|
1: metal
|
|
2: base
|
|
*/
|
|
|
|
void() item_key1 =
|
|
{
|
|
if (world.worldtype == 0)
|
|
{
|
|
precache_model ("progs/w_s_key.mdl");
|
|
setmodel (self, "progs/w_s_key.mdl");
|
|
self.netname = "silver key";
|
|
}
|
|
else if (world.worldtype == 1)
|
|
{
|
|
precache_model ("progs/m_s_key.mdl");
|
|
setmodel (self, "progs/m_s_key.mdl");
|
|
self.netname = "silver runekey";
|
|
}
|
|
else if (world.worldtype == 2)
|
|
{
|
|
precache_model2 ("progs/b_s_key.mdl");
|
|
setmodel (self, "progs/b_s_key.mdl");
|
|
self.netname = "silver keycard";
|
|
}
|
|
key_setsounds();
|
|
self.touch = key_touch;
|
|
self.items = IT_KEY1;
|
|
setsize (self, '-16 -16 -24', '16 16 32');
|
|
StartItem ();
|
|
};
|
|
|
|
/*QUAKED item_key2 (0 .5 .8) (-16 -16 -24) (16 16 32)
|
|
GOLD key
|
|
In order for keys to work
|
|
you MUST set your maps
|
|
worldtype to one of the
|
|
following:
|
|
0: medieval
|
|
1: metal
|
|
2: base
|
|
*/
|
|
|
|
void() item_key2 =
|
|
{
|
|
if (world.worldtype == 0)
|
|
{
|
|
precache_model ("progs/w_g_key.mdl");
|
|
setmodel (self, "progs/w_g_key.mdl");
|
|
self.netname = "gold key";
|
|
}
|
|
if (world.worldtype == 1)
|
|
{
|
|
precache_model ("progs/m_g_key.mdl");
|
|
setmodel (self, "progs/m_g_key.mdl");
|
|
self.netname = "gold runekey";
|
|
}
|
|
if (world.worldtype == 2)
|
|
{
|
|
precache_model2 ("progs/b_g_key.mdl");
|
|
setmodel (self, "progs/b_g_key.mdl");
|
|
self.netname = "gold keycard";
|
|
}
|
|
key_setsounds();
|
|
self.touch = key_touch;
|
|
self.items = IT_KEY2;
|
|
setsize (self, '-16 -16 -24', '16 16 32');
|
|
StartItem ();
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
===============================================================================
|
|
|
|
END OF LEVEL RUNES
|
|
|
|
===============================================================================
|
|
*/
|
|
|
|
void() sigil_touch =
|
|
{
|
|
if (!triggercantouch(self, other))
|
|
return;
|
|
|
|
if (other.classname != "player")
|
|
return;
|
|
if (other.health <= 0)
|
|
return;
|
|
|
|
centerprint (other, "You got the rune!");
|
|
|
|
sound (other, CHAN_ITEM, self.noise, 1, ATTN_NORM);
|
|
stuffcmd (other, "bf\n");
|
|
self.solid = SOLID_NOT;
|
|
self.model = string_null;
|
|
serverflags = serverflags | (self.spawnflags & 15);
|
|
self.classname = ""; // so rune doors won't find it
|
|
|
|
activator = other;
|
|
SUB_UseTargets(); // fire all targets / killtargets
|
|
};
|
|
|
|
|
|
/*QUAKED item_sigil (0 .5 .8) (-16 -16 -24) (16 16 32) E1 E2 E3 E4
|
|
End of level sigil, pick up to end episode and return to jrstart.
|
|
*/
|
|
|
|
void() item_sigil =
|
|
{
|
|
if (!self.spawnflags)
|
|
objerror ("no spawnflags");
|
|
|
|
precache_sound ("misc/runekey.wav");
|
|
self.noise = "misc/runekey.wav";
|
|
|
|
if (self.spawnflags & 1)
|
|
{
|
|
precache_model ("progs/end1.mdl");
|
|
setmodel (self, "progs/end1.mdl");
|
|
}
|
|
if (self.spawnflags & 2)
|
|
{
|
|
precache_model2 ("progs/end2.mdl");
|
|
setmodel (self, "progs/end2.mdl");
|
|
}
|
|
if (self.spawnflags & 4)
|
|
{
|
|
precache_model2 ("progs/end3.mdl");
|
|
setmodel (self, "progs/end3.mdl");
|
|
}
|
|
if (self.spawnflags & 8)
|
|
{
|
|
precache_model2 ("progs/end4.mdl");
|
|
setmodel (self, "progs/end4.mdl");
|
|
}
|
|
|
|
self.touch = sigil_touch;
|
|
setsize (self, '-16 -16 -24', '16 16 32');
|
|
StartItem ();
|
|
};
|
|
|
|
/*
|
|
===============================================================================
|
|
|
|
POWERUPS
|
|
|
|
===============================================================================
|
|
*/
|
|
|
|
void() powerup_touch;
|
|
|
|
|
|
void() powerup_touch =
|
|
{
|
|
if (!triggercantouch(self, other))
|
|
return;
|
|
|
|
if (other.classname != "player")
|
|
return;
|
|
if (other.health <= 0)
|
|
return;
|
|
|
|
sprint (other, PRINT_LOW, "You got the ");
|
|
sprint (other,PRINT_LOW, self.netname);
|
|
sprint (other,PRINT_LOW, "\n");
|
|
|
|
self.mdl = self.model;
|
|
|
|
if ((self.classname == "item_artifact_invulnerability") ||
|
|
(self.classname == "item_artifact_invisibility"))
|
|
self.nextthink = time + 60*5;
|
|
else
|
|
self.nextthink = time + 60;
|
|
|
|
self.think = SUB_regen;
|
|
|
|
sound (other, CHAN_VOICE, self.noise, 1, ATTN_NORM);
|
|
stuffcmd (other, "bf\n");
|
|
self.solid = SOLID_NOT;
|
|
other.items = other.items | self.items;
|
|
self.model = string_null;
|
|
|
|
// do the apropriate action
|
|
if (self.classname == "item_artifact_envirosuit")
|
|
{
|
|
other.rad_time = 1;
|
|
other.radsuit_finished = time + 30;
|
|
}
|
|
|
|
if (self.classname == "item_artifact_invulnerability")
|
|
{
|
|
other.invincible_time = 1;
|
|
other.invincible_finished = time + 30;
|
|
}
|
|
|
|
if (self.classname == "item_artifact_invisibility")
|
|
{
|
|
other.invisible_time = 1;
|
|
other.invisible_finished = time + 30;
|
|
}
|
|
|
|
if (self.classname == "item_artifact_super_damage")
|
|
{
|
|
if (deathmatch == 4)
|
|
{
|
|
other.armortype = 0;
|
|
other.armorvalue = 0 * 0.01;
|
|
other.ammo_cells = 0;
|
|
}
|
|
other.super_time = 1;
|
|
other.super_damage_finished = time + 30;
|
|
}
|
|
|
|
activator = other;
|
|
SUB_UseTargets(); // fire all targets / killtargets
|
|
};
|
|
|
|
|
|
|
|
/*QUAKED item_artifact_invulnerability (0 .5 .8) (-16 -16 -24) (16 16 32)
|
|
Player is invulnerable for 30 seconds
|
|
*/
|
|
void() item_artifact_invulnerability =
|
|
{
|
|
self.touch = powerup_touch;
|
|
|
|
precache_model ("progs/invulner.mdl");
|
|
precache_sound ("items/protect.wav");
|
|
precache_sound ("items/protect2.wav");
|
|
precache_sound ("items/protect3.wav");
|
|
self.noise = "items/protect.wav";
|
|
setmodel (self, "progs/invulner.mdl");
|
|
self.netname = "Pentagram of Protection";
|
|
self.effects = self.effects | EF_RED;
|
|
self.items = IT_INVULNERABILITY;
|
|
setsize (self, '-16 -16 -24', '16 16 32');
|
|
StartItem ();
|
|
};
|
|
|
|
/*QUAKED item_artifact_envirosuit (0 .5 .8) (-16 -16 -24) (16 16 32)
|
|
Player takes no damage from water or slime for 30 seconds
|
|
*/
|
|
void() item_artifact_envirosuit =
|
|
{
|
|
remove(self);
|
|
return;
|
|
};
|
|
|
|
|
|
/*QUAKED item_artifact_invisibility (0 .5 .8) (-16 -16 -24) (16 16 32)
|
|
Player is invisible for 30 seconds
|
|
*/
|
|
void() item_artifact_invisibility =
|
|
{
|
|
remove(self);
|
|
return;
|
|
};
|
|
|
|
|
|
/*QUAKED item_artifact_super_damage (0 .5 .8) (-16 -16 -24) (16 16 32)
|
|
The next attack from the player will do 4x damage
|
|
*/
|
|
void() item_artifact_super_damage =
|
|
{
|
|
|
|
remove(self);
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
===============================================================================
|
|
|
|
PLAYER BACKPACKS
|
|
|
|
===============================================================================
|
|
*/
|
|
|
|
void() BackpackTouch =
|
|
{
|
|
local float ammotype, ammocount;
|
|
local string ammoname;
|
|
|
|
if (other.classname != "player")
|
|
return;
|
|
|
|
sprint(other, 2, "loot: found");
|
|
|
|
if (self.weapon == 1)
|
|
{
|
|
TryGiveStackable(other, IID_AM_10MM, floor(9+random()*20));
|
|
sprint(other, 2, " 10mm ammo");
|
|
}
|
|
else if (self.weapon == 2)
|
|
{
|
|
TryGiveStackable(other, IID_AM_556MM, floor(5+random()*15));
|
|
sprint(other, 2, " 5mm ammo");
|
|
}
|
|
else if (self.weapon == 3)
|
|
{
|
|
TryGiveStackable(other, IID_AM_12GAUGESHELLS, floor(3+random()*6));
|
|
sprint(other, 2, " 12 gauge ammo");
|
|
}
|
|
else if (self.weapon == 4)
|
|
{
|
|
TryGiveStackable(other, IID_AM_10MM, floor(9+random()*20));
|
|
sprint(other, 2, " 10mm ammo");
|
|
}
|
|
else
|
|
{
|
|
ammotype = GetRandomAmmo();
|
|
ammocount = GetAmmoCount(ammotype);
|
|
ammoname = GetItemName(ammotype);
|
|
sprint(other, 2, " ");
|
|
sprint(other, 2, ammoname);
|
|
TryGiveStackable(other, ammotype, ammocount);
|
|
|
|
}
|
|
|
|
|
|
if (random()*4 <= 2)
|
|
{
|
|
other.ammo_shells = other.ammo_shells + ceil(5+random()*20);
|
|
sprint(other, 2, " & money.\n");
|
|
}
|
|
else
|
|
sprint(other, 2, ".\n");
|
|
|
|
remove(self);
|
|
return;
|
|
};
|
|
|
|
|
|
/*
|
|
===============
|
|
DropMoney
|
|
===============
|
|
*/
|
|
|
|
void() MoneyTouch =
|
|
{
|
|
if (other.classname != "player")
|
|
return;
|
|
|
|
if (other.health <= 0)
|
|
return;
|
|
|
|
sound (self, CHAN_VOICE, "items/item1.wav", 1, ATTN_IDLE);
|
|
sprint(other, 2, "a small amount of wasteland currency.\n");
|
|
other.ammo_shells += self.islot1;
|
|
remove(self);
|
|
};
|
|
|
|
void() DropMoney =
|
|
{
|
|
local entity item;
|
|
|
|
item = spawn();
|
|
item.origin = self.origin - '0 0 24';
|
|
|
|
item.weapon = self.weapon;
|
|
|
|
item.velocity_z = 300;
|
|
item.velocity_x = -100 + (random() * 200);
|
|
item.velocity_y = -100 + (random() * 200);
|
|
|
|
item.flags = FL_ITEM;
|
|
item.solid = SOLID_TRIGGER;
|
|
item.movetype = MOVETYPE_TOSS;
|
|
setmodel (item, "progs/money.mdl");
|
|
setsize (item, '-16 -16 0', '16 16 56');
|
|
item.islot1 = 2 + random()*5;
|
|
item.skin = ToIID(item.islot1) - 506;
|
|
|
|
item.touch = MoneyTouch;
|
|
|
|
item.nextthink = time + 120; // remove after 2 minutes
|
|
item.think = SUB_Remove;
|
|
};
|
|
|
|
|
|
/*
|
|
===============
|
|
DropBackpack
|
|
===============
|
|
*/
|
|
void() DropBackpack =
|
|
{
|
|
local entity item;
|
|
|
|
|
|
item = spawn();
|
|
item.origin = self.origin - '0 0 24';
|
|
|
|
item.weapon = self.weapon;
|
|
|
|
item.velocity_z = 300;
|
|
item.velocity_x = -100 + (random() * 200);
|
|
item.velocity_y = -100 + (random() * 200);
|
|
|
|
item.flags = FL_ITEM;
|
|
item.solid = SOLID_TRIGGER;
|
|
item.movetype = MOVETYPE_TOSS;
|
|
setmodel (item, "progs/ammobox.mdl");
|
|
setsize (item, '-16 -16 0', '16 16 56');
|
|
item.islot1 = GetRandomAmmo();
|
|
item.islot1 = item.islot1 * 512;
|
|
item.skin = ToIID(item.islot1) - 506;
|
|
|
|
item.flash = ceil(random()*30);
|
|
item.touch = AmmoTouch;
|
|
|
|
item.nextthink = time + 120; // remove after 2 minutes
|
|
item.think = SUB_Remove;
|
|
};
|
|
|
|
|