1 //void Item_ClearRespawnEffect (void) {
2 // self.effects = self.effects - (self.effects & EF_STARDUST);
5 void Item_Respawn (void)
7 self.model = self.mdl; // restore original model
8 self.solid = SOLID_TRIGGER; // allow it to be touched again
9 sound (self, CHAN_VOICE, "misc/itemrespawn.ogg", 1, ATTN_NORM); // play respawn sound
10 setorigin (self, self.origin);
12 // LordHavoc: replaced respawn stardust effect with a custom te_wizspike
13 te_wizspike(self.origin + self.mins_z * '0 0 1' + '0 0 48');
14 //// Savage: Add simple Respawn effect and make sure it gets removed
15 //self.effects = self.effects | EF_STARDUST;
16 //self.think = Item_ClearRespawnEffect;
17 //self.nextthink = time + 0.1;
20 void Item_Touch (void)
23 local float _switchweapon;
25 // remove the item if it's currnetly in a NODROP brush or hits a NOIMPACT surface (such as sky)
26 if (((trace_dpstartcontents | trace_dphitcontents) & DPCONTENTS_NODROP) || (trace_dphitq3surfaceflags & Q3SURFACEFLAG_NOIMPACT))
31 if (other.classname != "player")
35 if (self.solid != SOLID_TRIGGER)
37 if (self.health && other.health >= other.max_health)
39 if (self.owner == other)
41 // Savage: Remove the respawn effect if still present
42 self.effects = self.effects - (self.effects & EF_STARDUST);
44 sound (self, CHAN_BODY, self.noise, 1, ATTN_NORM);
46 // in case the player has autoswitch enabled do the following:
47 // if the player is using their best weapon before items are given, they
48 // probably want to switch to an even better weapon after items are given
49 _switchweapon = (other.autoswitch && (other.switchweapon == w_getbestweapon(other)) || cvar("g_minstagib"));
51 if (cvar("g_minstagib"))
55 // play some cool sounds ;)
56 centerprint(other, "\n");
58 stuffcmd(other, "play2 announcer/robotic/lastsecond.ogg\n");
59 else if(other.health < 50)
60 stuffcmd(other, "play2 announcer/robotic/narrowly.ogg\n");
61 // sound not available
62 // else if(self.items == IT_CELLS)
63 // stuffcmd(other, "play2 announce/robotic/ammo.ogg\n");
65 if (self.items & IT_NEX)
66 W_GiveWeapon (other, IT_NEX, "Nex");
68 other.ammo_cells = min (other.ammo_cells + cvar("g_minstagib_ammo_drop"), 999);
75 // sound not available
76 // stuffcmd(other, "play2 announce/robotic/extra.ogg\nplay2 announce/robotic/_lives.ogg\n");
77 other.armorvalue = other.armorvalue + cvar("g_minstagib_extralives");
78 sprint(other, "^3You picked up some extra lives\n");
82 if (self.strength_finished)
84 // sound not available
85 // stuffcmd(other, "play2 announce/robotic/invisible.ogg\n");
86 other.strength_finished = max(other.strength_finished, time) + cvar("g_balance_powerup_strength_time");
90 if (self.invincible_finished)
92 // sound not available
93 // stuffcmd(other, "play2 announce/robotic/speed.ogg\n");
94 other.invincible_finished = max(other.invincible_finished, time) + cvar("g_balance_powerup_strength_time");
99 if (cvar("deathmatch") == 2 || cvar("g_weapon_stay"))
101 if (self.flags & FL_WEAPON && other.items & self.items && self.classname != "droppedweapon")
103 if (other.items & self.items && self.flags & FL_TOSSED) // don't let players stack ammo by tossing weapons
107 if (self.ammo_shells)
108 other.ammo_shells = min (other.ammo_shells + self.ammo_shells, 999);
110 other.ammo_nails = min (other.ammo_nails + self.ammo_nails, 999);
111 if (self.ammo_rockets)
112 other.ammo_rockets = min (other.ammo_rockets + self.ammo_rockets, 999);
114 other.ammo_cells = min (other.ammo_cells + self.ammo_cells, 999);
116 if (self.items & IT_UZI) W_GiveWeapon (other, IT_UZI, self.netname);
117 if (self.items & IT_SHOTGUN) W_GiveWeapon (other, IT_SHOTGUN, self.netname);
118 if (self.items & IT_GRENADE_LAUNCHER) W_GiveWeapon (other, IT_GRENADE_LAUNCHER, self.netname);
119 if (self.items & IT_ELECTRO) W_GiveWeapon (other, IT_ELECTRO, self.netname);
120 if (self.items & IT_NEX) W_GiveWeapon (other, IT_NEX, self.netname);
121 if (self.items & IT_HAGAR) W_GiveWeapon (other, IT_HAGAR, self.netname);
122 if (self.items & IT_ROCKET_LAUNCHER) W_GiveWeapon (other, IT_ROCKET_LAUNCHER, self.netname);
123 if (self.items & IT_CRYLINK) W_GiveWeapon (other, IT_CRYLINK, self.netname);
125 if (self.strength_finished)
126 other.strength_finished = max(other.strength_finished, time) + cvar("g_balance_powerup_strength_time");
127 if (self.invincible_finished)
128 other.invincible_finished = max(other.invincible_finished, time) + cvar("g_balance_powerup_invincible_time");
129 //if (self.speed_finished)
130 // other.speed_finished = max(other.speed_finished, time) + cvar("g_balance_powerup_speed_time");
131 //if (self.slowmo_finished)
132 // other.slowmo_finished = max(other.slowmo_finished, time) + (cvar("g_balance_powerup_slowmo_time") * cvar("g_balance_powerup_slowmo_speed"));
136 other.health = other.health + self.max_health;
137 other.pauserothealth_finished = max(other.pauserothealth_finished, time + cvar("g_balance_pause_health_rot"));
139 if (self.health && other.health < other.max_health)
140 other.health = min(other.health + self.health, other.max_health);
143 other.armorvalue = other.armorvalue + self.armorvalue;
144 other.pauserotarmor_finished = max(other.pauserotarmor_finished, time + cvar("g_balance_pause_armor_rot"));
148 sound (other, CHAN_AUTO, self.item_pickupsound, 1, ATTN_NORM);
154 self.switchweapon = w_getbestweapon(self);
155 if (self.switchweapon != self.weapon)
156 self.cnt = self.weapon;
160 if (self.classname == "droppedweapon")
162 else if(self.flags & FL_WEAPON && (cvar("deathmatch") == 2 || cvar("g_weapon_stay")))
166 self.solid = SOLID_NOT;
167 self.model = string_null;
168 self.nextthink = time + self.respawntime;
169 self.think = Item_Respawn;
170 setorigin (self, self.origin);
174 // Savage: used for item garbage-collection
175 // TODO: perhaps nice special effect?
176 void RemoveItem(void) = {
180 // pickup evaluation functions
181 // these functions decide how desirable an item is to the bots
183 float(entity player, entity item) generic_pickupevalfunc = {return item.bot_pickupbasevalue;};
185 float(entity player, entity item) weapon_pickupevalfunc =
187 // if we already have the weapon, rate it 1/5th normal value
188 if ((player.items & item.items) == item.items)
189 return item.bot_pickupbasevalue * 0.2;
190 return item.bot_pickupbasevalue;
193 float(entity player, entity item) commodity_pickupevalfunc =
197 // TODO: figure out if the player even has the weapon this ammo is for?
198 // may not affect strategy much though...
199 // find out how much ammo the player has, in terms of this ammo pickup
200 // (how many of these ammo pickups it would take to total the player's
202 if (item.ammo_shells)
203 c = c + player.ammo_shells / item.ammo_shells;
205 c = c + player.ammo_nails / item.ammo_nails;
206 if (item.ammo_rockets)
207 c = c + player.ammo_rockets / item.ammo_rockets;
209 c = c + player.ammo_cells / item.ammo_cells;
211 c = c + player.armorvalue / item.armorvalue;
213 c = c + player.health / item.health / 10;
214 // the more ammo the player has, the less desirable this pickup becomes
217 if (cvar("deathmatch") == 2) // weapon stay is on, so weapons the player already has are of no interest
218 if (item.flags & FL_WEAPON)
219 if (self.items & item.items)
220 if (item.classname != "droppedweapon")
223 return item.bot_pickupbasevalue * c;
228 void StartItem (string itemmodel, string pickupsound, float defaultrespawntime, string itemname, float itemid, float itemflags, float(entity player, entity item) pickupevalfunc, float pickupbasevalue)
233 startitem_failed = FALSE;
235 if (self.classname != "droppedweapon" && !self.noalign)
241 if (itemid == IT_SHELLS)
243 else if (itemid == IT_ROCKETS)
245 else if (itemid == IT_NAILS)
247 else if (itemid == IT_25HP)
249 else if (itemid == IT_ARMOR)
251 else if (itemid == IT_SHOTGUN)
254 org = find_floor(org) + z_offset;
255 setorigin(self, org);
258 if(cvar("spawn_debug") >= 2)
260 if(self.classname != "droppedweapon")
263 for(otheritem = findradius(org, 3); otheritem; otheritem = otheritem.chain)
265 if(otheritem.is_item)
267 dprint("XXX Found duplicated item: ", itemname, vtos(self.origin));
268 dprint(" vs ", otheritem.netname, vtos(otheritem.origin), "\n");
269 error("Mapper sucks.");
276 if (self.classname != "droppedweapon")
277 waypoint_spawnforitem(self);
279 if (!(cvar("g_pickup_items") && !cvar("g_nixnex")) && !cvar("g_minstagib") &&
280 itemid != IT_STRENGTH && itemid != IT_INVINCIBLE && itemid != IT_HEALTH)
282 startitem_failed = TRUE;
287 if (cvar("g_minstagib"))
289 // don't remove dropped items and powerups
290 if (self.classname != "droppedweapon" &&
291 self.classname != "minstagib")
293 startitem_failed = TRUE;
299 if(cvar("g_lms") && (self.classname != "droppedweapon"))
301 startitem_failed = TRUE;
306 if(cvar("g_instagib") || cvar("g_rocketarena"))
308 startitem_failed = TRUE;
313 if (self.classname == "droppedweapon")
315 // don't drop if in a NODROP zone (such as lava)
316 traceline(self.origin, self.origin, MOVE_NORMAL, self);
317 if (trace_dpstartcontents & DPCONTENTS_NODROP)
319 startitem_failed = TRUE;
325 if(itemid & (IT_STRENGTH | IT_INVINCIBLE | IT_HEALTH | IT_ARMOR | IT_KEY1 | IT_KEY2 |
326 IT_ROCKET_LAUNCHER | IT_HAGAR | IT_NEX | IT_CRYLINK | IT_ELECTRO |
327 IT_GRENADE_LAUNCHER | IT_UZI | IT_SHOTGUN | IT_LASER) && self.classname != "droppedweapon")
329 self.target = "###item###"; // for finding the nearest item using find()
331 self.bot_pickup = TRUE;
332 self.bot_pickupevalfunc = pickupevalfunc;
333 self.bot_pickupbasevalue = pickupbasevalue;
334 self.mdl = itemmodel;
335 //self.noise = pickupsound;
336 self.item_pickupsound = pickupsound;
337 // let mappers override respawntime
338 if (!self.respawntime)
339 self.respawntime = defaultrespawntime;
340 self.netname = itemname;
342 self.flags = FL_ITEM | itemflags;
343 setmodel (self, self.mdl);
344 if (itemflags & FL_WEAPON)
346 setorigin (self, self.origin + '0 0 23');
347 setsize (self, '-12 -12 -12', '12 12 12');
349 // neutral team color for pickup weapons
350 self.colormap = 160 * 1024 + 160;
354 setorigin (self, self.origin + '0 0 25');
355 // setsize (self, '-8 -8 -5', '8 8 8');
357 self.movetype = MOVETYPE_TOSS;
358 self.solid = SOLID_TRIGGER;
359 self.touch = Item_Touch;
361 // Savage: remove thrown items after a certain period of time ("garbage collection")
362 if (self.classname == "droppedweapon")
364 self.think = RemoveItem;
365 self.nextthink = time + 60;
369 self.movetype = MOVETYPE_NONE;
370 setorigin(self, org);
373 if (cvar("g_fullbrightitems"))
374 self.effects = self.effects | EF_FULLBRIGHT;
377 /* replace items in minstagib
378 * IT_STRENGTH = invisibility
379 * IT_NAILS = extra lives
380 * IT_INVINCIBLE = speed
382 void minstagib_items (float itemid)
384 // we don't want to replace dropped weapons ;)
385 if (self.classname == "droppedweapon")
387 self.ammo_cells = 25;
388 StartItem ("models/weapons/g_nex.md3",
389 "weapons/weaponpickup.ogg", 15,
390 "Nex Gun", IT_NEX, FL_WEAPON, generic_pickupevalfunc, 1000);
395 self.classname = "minstagib";
397 // replace rocket launchers and nex guns with ammo cells
398 if (itemid == IT_CELLS)
401 StartItem ("models/items/a_cells.md3",
402 "misc/itempickup.ogg", 45,
403 "Nex Ammo", IT_CELLS, 0, generic_pickupevalfunc, 100);
410 itemid = IT_STRENGTH;
414 itemid = IT_INVINCIBLE;
416 // replace with invis
417 if (itemid == IT_STRENGTH)
419 self.effects = EF_ADDITIVE;
420 self.strength_finished = 30;
421 StartItem ("models/items/g_strength.md3",
422 "misc/powerup.ogg", 120,
423 "Invisibility", IT_STRENGTH, FL_POWERUP, generic_pickupevalfunc, 1000);
425 // replace with extra lives
426 if (itemid == IT_NAILS)
429 StartItem ("models/items/g_h100.md3",
430 "misc/megahealth.ogg", 120,
431 "Extralife", IT_NAILS, FL_POWERUP, generic_pickupevalfunc, 1000);
434 // replace with speed
435 if (itemid == IT_INVINCIBLE)
437 self.effects = EF_ADDITIVE;
438 self.invincible_finished = 30;
439 StartItem ("models/items/g_invincible.md3",
440 "misc/powerup_shield.ogg", 120,
441 "Speed", IT_INVINCIBLE, FL_POWERUP, generic_pickupevalfunc, 1000);
446 void weapon_uzi (void) {
448 self.ammo_nails = 120;
449 StartItem ("models/weapons/g_uzi.md3", "weapons/weaponpickup.ogg", 15, W_Name(WEP_UZI), IT_UZI, FL_WEAPON, weapon_pickupevalfunc, 1000);
452 void weapon_shotgun (void) {
453 if(!self.ammo_shells)
454 self.ammo_shells = 15;
455 StartItem ("models/weapons/g_shotgun.md3", "weapons/weaponpickup.ogg", 15, W_Name(WEP_SHOTGUN), IT_SHOTGUN, FL_WEAPON, weapon_pickupevalfunc, 1000);
458 void weapon_grenadelauncher (void) {
459 if(!self.ammo_rockets)
460 self.ammo_rockets = 15;
461 StartItem ("models/weapons/g_gl.md3", "weapons/weaponpickup.ogg", 15, W_Name(WEP_GRENADE_LAUNCHER), IT_GRENADE_LAUNCHER, FL_WEAPON, weapon_pickupevalfunc, 1000);
464 void weapon_electro (void) {
466 self.ammo_cells = 25;
467 StartItem ("models/weapons/g_electro.md3", "weapons/weaponpickup.ogg", 15, W_Name(WEP_ELECTRO), IT_ELECTRO, FL_WEAPON, weapon_pickupevalfunc, 1000);
470 void weapon_crylink (void) {
472 self.ammo_cells = 25;
473 StartItem ("models/weapons/g_crylink.md3", "weapons/weaponpickup.ogg", 15, W_Name(WEP_CRYLINK), IT_CRYLINK, FL_WEAPON, weapon_pickupevalfunc, 1000);
476 void weapon_nex (void) {
477 if (cvar("g_minstagib")) {
478 minstagib_items(IT_CELLS);
482 self.ammo_cells = 25;
483 nextime = cvar("g_balance_nex_respawntime_modifier");
485 nextime = 15 * nextime;
488 StartItem ("models/weapons/g_nex.md3", "weapons/weaponpickup.ogg", nextime, W_Name(WEP_NEX), IT_NEX, FL_WEAPON, weapon_pickupevalfunc, 1000);
492 void weapon_hagar (void) {
493 if(!self.ammo_rockets)
494 self.ammo_rockets = 15;
495 StartItem ("models/weapons/g_hagar.md3", "weapons/weaponpickup.ogg", 15, W_Name(WEP_HAGAR), IT_HAGAR, FL_WEAPON, weapon_pickupevalfunc, 1000);
498 void weapon_rocketlauncher (void) {
499 if (cvar("g_minstagib")) {
500 minstagib_items(IT_CELLS);
502 if(!self.ammo_rockets)
503 self.ammo_rockets = 15;
504 StartItem ("models/weapons/g_rl.md3", "weapons/weaponpickup.ogg", 15, W_Name(WEP_ROCKET_LAUNCHER), IT_ROCKET_LAUNCHER, FL_WEAPON, weapon_pickupevalfunc, 1000);
508 void item_rockets (void) {
509 if(!self.ammo_rockets)
510 self.ammo_rockets = 15;
511 StartItem ("models/items/a_rockets.md3", "misc/itempickup.ogg", 15, "rockets", IT_ROCKETS, 0, commodity_pickupevalfunc, 100);
514 void item_bullets (void) {
516 self.ammo_nails = 120;
517 StartItem ("models/items/a_bullets.mdl", "misc/itempickup.ogg", 15, "bullets", IT_NAILS, 0, commodity_pickupevalfunc, 100);
520 void item_cells (void) {
522 self.ammo_cells = 25;
523 StartItem ("models/items/a_cells.md3", "misc/itempickup.ogg", 15, "cells", IT_CELLS, 0, commodity_pickupevalfunc, 100);
526 void item_shells (void) {
527 if(!self.ammo_shells)
528 self.ammo_shells = 15;
529 StartItem ("models/items/a_shells.md3", "misc/itempickup.ogg", 15, "shells", IT_SHELLS, 0, commodity_pickupevalfunc, 100);
532 void item_armor1 (void) {
535 StartItem ("models/items/g_a1.md3", "misc/armor1.wav", 15, "Armor Shard", IT_ARMOR_SHARD, 0, commodity_pickupevalfunc, 100);
538 void item_armor25 (void) {
540 self.armorvalue = 100;
541 StartItem ("models/items/g_a25.md3", "misc/armor25.wav", 30, "Armor", IT_ARMOR, 0, commodity_pickupevalfunc, 2000);
544 void item_health1 (void) {
547 StartItem ("models/items/g_h1.md3", "misc/minihealth.ogg", 15, "5 Health", IT_5HP, 0, commodity_pickupevalfunc, 100);
550 void item_health25 (void) {
552 self.max_health = 25;
553 StartItem ("models/items/g_h25.md3", "misc/mediumhealth.ogg", 15, "25 Health", IT_25HP, 0, commodity_pickupevalfunc, 500);
556 void item_health100 (void) {
557 if(!cvar("g_powerup_superhealth"))
560 if(cvar("g_arena") && !cvar("g_arena_powerups"))
563 if(cvar("g_minstagib")) {
564 minstagib_items(IT_NAILS);
567 self.max_health = 100;
568 StartItem ("models/items/g_h100.md3", "misc/megahealth.ogg", 30, "100 Health", IT_HEALTH, 0, commodity_pickupevalfunc, 2000);
572 void item_strength (void) {
573 if(!cvar("g_powerup_strength"))
576 if(cvar("g_arena") && !cvar("g_arena_powerups"))
579 if(cvar("g_minstagib")) {
580 minstagib_items(IT_STRENGTH);
582 self.strength_finished = 30;
583 self.effects = EF_ADDITIVE;StartItem ("models/items/g_strength.md3", "misc/powerup.ogg", 120, "Strength Powerup", IT_STRENGTH, FL_POWERUP, generic_pickupevalfunc, 10000);
587 void item_invincible (void) {
588 if(!cvar("g_powerup_shield"))
591 if(cvar("g_arena") && !cvar("g_arena_powerups"))
594 if(cvar("g_minstagib")) {
595 minstagib_items(IT_INVINCIBLE);
597 self.invincible_finished = 30;
598 self.effects = EF_ADDITIVE;
599 StartItem ("models/items/g_invincible.md3", "misc/powerup_shield.ogg", 120, "Invulnerability", IT_INVINCIBLE, FL_POWERUP, generic_pickupevalfunc, 10000);
602 //void item_speed (void) {self.speed_finished = 30;StartItem ("models/items/g_speed.md3", "misc/powerup.wav", 120, "Speed Powerup", IT_SPEED, FL_POWERUP, generic_pickupevalfunc, 10000);}
603 //void item_slowmo (void) {self.slowmo_finished = 30;StartItem ("models/items/g_slowmo.md3", "misc/powerup.wav", 120, "Slow Motion", IT_SLOWMO, FL_POWERUP, generic_pickupevalfunc, 10000);}
606 void item_quad (void) {self.classname = "item_strength";item_strength();}
608 void misc_models (void)
610 precache_model (self.model);
611 setmodel (self, self.model);
612 setsize (self, self.mins, self.maxs);
619 floatfield Item_CounterField(float it)
623 case IT_SHELLS: return ammo_shells;
624 case IT_NAILS: return ammo_nails;
625 case IT_ROCKETS: return ammo_rockets;
626 case IT_CELLS: return ammo_cells;
627 case IT_5HP: return health;
628 case IT_25HP: return health;
629 case IT_HEALTH: return health;
630 case IT_ARMOR_SHARD: return armorvalue;
631 case IT_ARMOR: return armorvalue;
632 // add more things here (health, armor)
633 default: error("requested item has no counter field");
637 float Item_WeaponCode(float it)
641 case IT_LASER: return WEP_LASER;
642 case IT_SHOTGUN: return WEP_SHOTGUN;
643 case IT_UZI: return WEP_UZI;
644 case IT_GRENADE_LAUNCHER: return WEP_GRENADE_LAUNCHER;
645 case IT_ELECTRO: return WEP_ELECTRO;
646 case IT_CRYLINK: return WEP_CRYLINK;
647 case IT_NEX: return WEP_NEX;
648 case IT_HAGAR: return WEP_HAGAR;
649 case IT_ROCKET_LAUNCHER: return WEP_ROCKET_LAUNCHER;
654 spawnfunc Item_SpawnFunc(float it)
658 case IT_SHOTGUN: return weapon_shotgun;
659 case IT_UZI: return weapon_uzi;
660 case IT_GRENADE_LAUNCHER: return weapon_grenadelauncher;
661 case IT_ELECTRO: return weapon_electro;
662 case IT_CRYLINK: return weapon_crylink;
663 case IT_NEX: return weapon_nex;
664 case IT_HAGAR: return weapon_hagar;
665 case IT_ROCKET_LAUNCHER: return weapon_rocketlauncher;
666 // add all other item spawn functions here
667 default: error("requested item can't be spawned");