]> icculus.org git repositories - divverent/nexuiz.git/blob - TeamNexuiz/game/gamec/w_common.c
-Fixed Engineer bug where holding down +use key near a building would sprint spam...
[divverent/nexuiz.git] / TeamNexuiz / game / gamec / w_common.c
1 \r
2 // increments sprite frame, loops when end is hit.. simple\r
3 \r
4 float TE_SMOKE =77;\r
5 void (vector vec) WriteVec =\r
6 {\r
7                 WriteCoord (MSG_BROADCAST, vec_x);\r
8                 WriteCoord (MSG_BROADCAST, vec_y);\r
9                 WriteCoord (MSG_BROADCAST, vec_z);\r
10 }\r
11 void (vector org, vector dir, float counts) W_Smoke =\r
12 {\r
13                 WriteByte (MSG_BROADCAST, SVC_TEMPENTITY);\r
14                 WriteByte (MSG_BROADCAST, TE_SMOKE);\r
15                 WriteVec (org);\r
16                 WriteVec (dir);\r
17                 WriteByte (MSG_BROADCAST, counts);\r
18 }\r
19 \r
20 // increments sprite frame, loops when end is hit.. simple\r
21 void animate_sprite (float startframe, float frame_count)\r
22 {\r
23         if ((self.frame - startframe) >= (frame_count - 1 ))\r
24                 self.frame = startframe;\r
25         else\r
26                 self.frame = self.frame + 1;\r
27 }\r
28 \r
29 void W_UpdateAmmo (void)\r
30 {\r
31         /*\r
32         self.items = self.items - (self.items & (IT_NAILS | IT_SHELLS | IT_ROCKETS | IT_CELLS));\r
33 \r
34         if (self.weapon == IT_LASER)\r
35                 self.currentammo = 1;\r
36         else if (self.weapon == IT_SHOTGUN)\r
37         {\r
38                 self.currentammo = self.ammo_shells;\r
39                 self.items = self.items | IT_SHELLS;\r
40         }\r
41         else if (self.weapon == IT_UZI)\r
42         {\r
43                 self.currentammo = self.ammo_nails;\r
44                 self.items = self.items | IT_NAILS;\r
45         }\r
46         else if (self.weapon == IT_GRENADE_LAUNCHER || self.weapon == IT_HAGAR || self.weapon == IT_ROCKET_LAUNCHER)\r
47         {\r
48                 self.currentammo = self.ammo_rockets;\r
49                 self.items = self.items | IT_ROCKETS;\r
50         }\r
51         else if (self.weapon == IT_ELECTRO || self.weapon == IT_NEX || self.weapon == IT_CRYLINK)\r
52         {\r
53                 self.currentammo = self.ammo_cells;\r
54                 self.items = self.items | IT_CELLS;\r
55         }\r
56         */\r
57 }\r
58 \r
59 void W_UpdateWeapon (void)\r
60 {\r
61         /*\r
62         if (self.weapon == IT_LASER)\r
63                 self.weaponmodel = "models/weapons/w_laser.zym";\r
64         else if (self.weapon == IT_SHOTGUN)\r
65                 self.weaponmodel = "models/weapons/w_shotgun.zym";\r
66         else if (self.weapon == IT_UZI)\r
67                 self.weaponmodel = "models/weapons/w_uzi.zym";\r
68         else if (self.weapon == IT_GRENADE_LAUNCHER)\r
69                 self.weaponmodel = "models/weapons/w_gl.zym";\r
70         else if (self.weapon == IT_ELECTRO)\r
71                 self.weaponmodel = "models/weapons/w_electro.zym";\r
72         else if (self.weapon == IT_CRYLINK)\r
73                 self.weaponmodel = "models/weapons/w_crylink.zym";\r
74         else if (self.weapon == IT_NEX)\r
75                 self.weaponmodel = "models/weapons/w_nex.zym";\r
76         else if (self.weapon == IT_HAGAR)\r
77                 self.weaponmodel = "models/weapons/w_hagar.zym";\r
78         else if (self.weapon == IT_ROCKET_LAUNCHER)\r
79                 self.weaponmodel = "models/weapons/w_rl.zym";\r
80         else\r
81                 objerror ("Illegal weapon - please register your guns please!");\r
82         */\r
83 }\r
84 \r
85 float W_GetBestWeapon (entity e)\r
86 {\r
87         /*\r
88         if ((e.items & IT_ROCKET_LAUNCHER) && e.ammo_rockets)\r
89                 return IT_ROCKET_LAUNCHER;\r
90         else if ((e.items & IT_NEX) && e.ammo_cells)\r
91                 return IT_NEX;\r
92         else if ((e.items & IT_HAGAR) && e.ammo_rockets)\r
93                 return IT_HAGAR;\r
94         else if ((e.items & IT_GRENADE_LAUNCHER) && e.ammo_rockets)\r
95                 return IT_GRENADE_LAUNCHER;\r
96         else if ((e.items & IT_ELECTRO) && e.ammo_cells)\r
97                 return IT_ELECTRO;\r
98         else if ((e.items & IT_CRYLINK) && e.ammo_cells)\r
99                 return IT_CRYLINK;\r
100         else if ((e.items & IT_UZI) && e.ammo_nails)\r
101                 return IT_UZI;\r
102         else if ((e.items & IT_SHOTGUN) && e.ammo_shells)\r
103                 return IT_SHOTGUN;\r
104         else\r
105 \r
106                 */\r
107         return IT_LASER;\r
108 }\r
109 \r
110 void ResetExtraWeapon()\r
111 {\r
112         if(self.wpn5 == world)\r
113         {\r
114                 self.wpn5 = spawn();\r
115         }\r
116 \r
117         self.wpn5.weapon = 0;\r
118         self.wpn5.mass = 0;\r
119         self.items = self.items - (self.items & IT_WEP5);\r
120 }\r
121 \r
122 void W_GiveWeapon (entity e, float wep, string name, float wmass)\r
123 {\r
124         entity oldself;\r
125 \r
126         if (!wep)\r
127                 return;\r
128 \r
129         //e.items = e.items | wep;\r
130 \r
131         oldself = self;\r
132         self = e;\r
133 \r
134         self.wpn5.weapon = wep;\r
135         self.wpn5.mass = wmass;\r
136 \r
137         self.items = self.items | IT_WEP5;\r
138 \r
139         weapon_action(self.weapon, WR_UPDATECOUNTS);\r
140         if(self.weapon == WEP5)                                 // if using carrying # 5 already\r
141         {\r
142                 //bprint("reset wep5\n");\r
143                 self.wpn = self.wpn5.weapon;\r
144                 self.switchweapon = WEP5;\r
145                 weapon_action(self.weapon, WR_DROP);\r
146                 self.weapon = 0;\r
147                 //weapon_action(self.wpn, WR_SETUP);    // update the weapon we're holding\r
148                 //weapon_action(self.wpn, WR_RAISE);    // update the weapon we're holding\r
149         }\r
150 \r
151         if (other.classname == "player")\r
152         {\r
153                 sprint (other, "You got the ^2");\r
154                 sprint (other, name);\r
155                 sprint (other, "\n");\r
156         }\r
157 \r
158 \r
159 /*\r
160         W_UpdateWeapon ();\r
161         W_UpdateAmmo ();\r
162 */\r
163         self = oldself;\r
164 }\r
165 \r
166 /*\r
167 void W_SwitchWeapon (float wep)\r
168 {\r
169         float           nextwep;\r
170         var float       noammo = FALSE;\r
171 \r
172         if (wep == 1)\r
173                 nextwep = IT_LASER;\r
174         else if (wep == 2)\r
175         {\r
176                 nextwep = IT_SHOTGUN;\r
177                 if (!self.ammo_shells)\r
178                         noammo = TRUE;\r
179         }\r
180         else if (wep == 3)\r
181         {\r
182                 nextwep = IT_UZI;\r
183                 if (!self.ammo_nails)\r
184                         noammo = TRUE;\r
185         }\r
186         else if (wep == 4)\r
187         {\r
188                 nextwep = IT_CRYLINK;\r
189                 if (!self.ammo_cells)\r
190                         noammo = TRUE;\r
191         }\r
192         else if (wep == 5)\r
193         {\r
194                 nextwep = IT_ELECTRO;\r
195                 if (!self.ammo_cells)\r
196                         noammo = TRUE;\r
197         }\r
198         else if (wep == 6)\r
199         {\r
200                 nextwep = IT_GRENADE_LAUNCHER;\r
201                 if (!self.ammo_rockets)\r
202                         noammo = TRUE;\r
203         }\r
204         else if (wep == 7)\r
205         {\r
206                 nextwep = IT_HAGAR;\r
207                 if (!self.ammo_rockets)\r
208                         noammo = TRUE;\r
209         }\r
210         else if (wep == 8)\r
211         {\r
212                 nextwep = IT_NEX;\r
213                 if (!self.ammo_cells)\r
214                         noammo = TRUE;\r
215         }\r
216         else if (wep == 9)\r
217         {\r
218                 nextwep = IT_ROCKET_LAUNCHER;\r
219                 if (!self.ammo_rockets)\r
220                         noammo = TRUE;\r
221         }\r
222 \r
223 \r
224         if (!(self.items & nextwep))\r
225         {\r
226                 sprint (self, "You don't own that weapon\n");\r
227                 return;\r
228         }\r
229         else if (noammo)\r
230         {\r
231                 sprint (self, "You don't have any ammo for that weapon\n");\r
232                 return;\r
233         }\r
234 \r
235         self.weapon = nextwep;\r
236         W_UpdateWeapon ();\r
237         W_UpdateAmmo ();\r
238         self.attack_finished = time + 0.2;\r
239         if (self.viewzoom != 1)\r
240                 self.viewzoom = 1;\r
241 }\r
242 \r
243 void W_NextWeapon (void)\r
244 {\r
245         float   noammo;\r
246 \r
247         while (TRUE)\r
248         {\r
249                 noammo = FALSE;\r
250 \r
251                 if (self.weapon == IT_ROCKET_LAUNCHER)\r
252                         self.weapon = IT_LASER;\r
253                 else if (self.weapon == IT_LASER)\r
254                 {\r
255                         self.weapon = IT_SHOTGUN;\r
256                         if (!self.ammo_shells)\r
257                                 noammo = TRUE;\r
258                 }\r
259                 else if (self.weapon == IT_SHOTGUN)\r
260                 {\r
261                         self.weapon = IT_UZI;\r
262                         if (!self.ammo_nails)\r
263                                 noammo = TRUE;\r
264                 }\r
265                 else if (self.weapon == IT_UZI)\r
266                 {\r
267                         self.weapon = IT_CRYLINK;\r
268                         if (!self.ammo_cells)\r
269                         noammo = TRUE;\r
270                 }\r
271                 else if (self.weapon == IT_CRYLINK)\r
272                 {\r
273                         self.weapon = IT_ELECTRO;\r
274                         if (!self.ammo_cells)\r
275                                 noammo = TRUE;\r
276                 }\r
277                 else if (self.weapon == IT_ELECTRO)\r
278                 {\r
279                         self.weapon = IT_GRENADE_LAUNCHER;\r
280                         if (!self.ammo_cells)\r
281                                 noammo = TRUE;\r
282                 }\r
283                 else if (self.weapon == IT_GRENADE_LAUNCHER)\r
284                 {\r
285                         self.weapon = IT_HAGAR;\r
286                         if (!self.ammo_rockets)\r
287                                 noammo = TRUE;\r
288                 }\r
289                 else if (self.weapon == IT_HAGAR)\r
290                 {\r
291                         self.weapon = IT_NEX;\r
292                         if (!self.ammo_rockets)\r
293                         noammo = TRUE;\r
294                 }\r
295                 else if (self.weapon == IT_NEX)\r
296                 {\r
297                         self.weapon = IT_ROCKET_LAUNCHER;\r
298                         if (!self.ammo_cells)\r
299                         noammo = TRUE;\r
300                 }\r
301 \r
302                 if ((self.items & self.weapon) && !noammo)\r
303                 {\r
304                         W_UpdateWeapon ();\r
305                         W_UpdateAmmo ();\r
306                         return;\r
307                 }\r
308         }\r
309 }\r
310 \r
311 void W_PreviousWeapon (void)\r
312 {\r
313         float   noammo;\r
314 \r
315         while (TRUE)\r
316         {\r
317                 noammo = FALSE;\r
318 \r
319                 if (self.weapon == IT_SHOTGUN)\r
320                         self.weapon = IT_LASER;\r
321                 else if (self.weapon == IT_UZI)\r
322                 {\r
323                         self.weapon = IT_SHOTGUN;\r
324                         if (!self.ammo_shells)\r
325                                 noammo = TRUE;\r
326                 }\r
327                 else if (self.weapon == IT_CRYLINK)\r
328                 {\r
329                         self.weapon = IT_UZI;\r
330                         if (!self.ammo_nails)\r
331                                 noammo = TRUE;\r
332                 }\r
333                 else if (self.weapon == IT_ELECTRO)\r
334                 {\r
335                         self.weapon = IT_CRYLINK;\r
336                         if (!self.ammo_cells)\r
337                                 noammo = TRUE;\r
338                 }\r
339                 else if (self.weapon == IT_GRENADE_LAUNCHER)\r
340                 {\r
341                         self.weapon = IT_ELECTRO;\r
342                         if (!self.ammo_cells)\r
343                                 noammo = TRUE;\r
344                 }\r
345                 else if (self.weapon == IT_HAGAR)\r
346                 {\r
347                         self.weapon = IT_GRENADE_LAUNCHER;\r
348                         if (!self.ammo_rockets)\r
349                                 noammo = TRUE;\r
350                 }\r
351                 else if (self.weapon == IT_NEX)\r
352                 {\r
353                         self.weapon = IT_HAGAR;\r
354                         if (!self.ammo_rockets)\r
355                                 noammo = TRUE;\r
356                 }\r
357                 else if (self.weapon == IT_ROCKET_LAUNCHER)\r
358                 {\r
359                         self.weapon = IT_NEX;\r
360                         if (!self.ammo_cells)\r
361                                 noammo = TRUE;\r
362                 }\r
363                 else if (self.weapon == IT_LASER)\r
364                 {\r
365                         self.weapon = IT_ROCKET_LAUNCHER;\r
366                         if (!self.ammo_rockets)\r
367                                 noammo = TRUE;\r
368                 }\r
369 \r
370                 if ((self.items & self.weapon) && !noammo)\r
371                 {\r
372                         W_UpdateWeapon ();\r
373                         W_UpdateAmmo ();\r
374                         return;\r
375                 }\r
376         }\r
377 }\r
378 */\r
379 float W_CheckAmmo (void)\r
380 {\r
381         if ((cvar("g_instagib") == 1) | (cvar("g_rocketarena") == 1))\r
382                 return TRUE;\r
383 \r
384         W_UpdateAmmo ();\r
385         if (self.weapon == IT_LASER)\r
386                 return TRUE;\r
387         else if (self.currentammo)\r
388                 return TRUE;\r
389 \r
390         self.weapon = W_GetBestWeapon (self);\r
391         W_UpdateWeapon ();\r
392 \r
393         return FALSE;\r
394 }\r
395 \r
396 /*\r
397 void FireRailgunBullet (vector src, float bdamage, vector dir, float spread, float deathtype)\r
398 {\r
399         vector  v, lastpos;\r
400         entity  saveself, last;\r
401         vector  org;\r
402         org = self.origin + self.view_ofs;\r
403         if (bdamage < 1)\r
404                 return;\r
405 \r
406         last = self;\r
407         lastpos = src;\r
408 \r
409         while (bdamage > 0)\r
410         {\r
411                 traceline_hitcorpse (self, org, org + v_forward * 4096 + v_right * crandom () * spread + v_up * crandom () * spread, FALSE, self);\r
412                 last = trace_ent;\r
413                 lastpos = trace_endpos;\r
414                 if (trace_fraction != 1.0)\r
415                 {\r
416                         if (pointcontents(trace_endpos - dir*4) == CONTENT_SKY)\r
417                                 return;\r
418 \r
419                         if (trace_ent.takedamage || trace_ent.classname == "case")\r
420                         {\r
421                                 if (trace_ent.classname == "player" || trace_ent.classname == "corpse" || trace_ent.classname == "gib")\r
422                                         te_blood (trace_endpos, dir * bdamage * 16, bdamage);\r
423                                 Damage (trace_ent, self, self, bdamage, deathtype, trace_endpos, dir * bdamage);\r
424                         }\r
425                 }\r
426                 if (last.solid == SOLID_BSP)\r
427                         bdamage = 0;\r
428         }\r
429 }\r
430 */\r
431 \r
432 //vector railgun_hitlocation;\r
433 .vector head_shot_vector;\r
434 .float leg_damage;\r
435 void FireRailgunBullet (vector start, vector end, float bdamage, float deathtype)\r
436 {\r
437         local vector hitloc, force;\r
438         local entity ent;\r
439         //local entity explosion;\r
440 \r
441         force = normalize(end - start) * 800; //(bdamage * 10);\r
442 \r
443         // find how far the beam can go until it hits a wall\r
444         traceline (start, end, TRUE, self);\r
445         // go a little bit into the wall because we need to hit this wall later\r
446         end = trace_endpos + normalize(end - start);\r
447 \r
448         local float dam_mult;\r
449         local float zdif;\r
450         local float use_this;\r
451         local float tracline;\r
452         local float x;\r
453         local vector f;\r
454         local vector g;\r
455         local vector h;\r
456 \r
457         if (trace_ent && deathtype == WEP_RAILGUN)                      // Area damage?\r
458         {\r
459                 if ((trace_ent.classname == "player"))\r
460                 {\r
461                         f = (trace_endpos - start);\r
462                         g_x = trace_endpos_x;\r
463                         g_y = trace_endpos_y;\r
464                         g_z = 0;\r
465                         h_x = trace_ent.origin_x;\r
466                         h_y = trace_ent.origin_y;\r
467                         h_z = 0;\r
468                         x = vlen ((g - h));\r
469                         f = ((normalize (f) * x) + trace_endpos);\r
470                         zdif = (f_z - trace_ent.origin_z);\r
471                         deathmsg = 18;\r
472                         trace_ent.head_shot_vector = '0 0 0';\r
473                         if (zdif < 0)\r
474                         {\r
475                                 dam_mult = 0.5;\r
476                                 if ((trace_ent.team_no != self.team_no))\r
477                                 {\r
478                                         trace_ent.leg_damage = (trace_ent.leg_damage + 1);\r
479                                         TeamFortress_SetSpeed (trace_ent);\r
480                                         deathmsg = 28;\r
481                                         T_Damage (trace_ent, self, self, (self.heat * dam_mult));\r
482                                 }\r
483                                 if ((trace_ent.health > 0))\r
484                                 {\r
485                                         if ((trace_ent.team_no == self.team_no))\r
486                                         {\r
487                                                 sprint (self, "Stop shooting team mates!!!\n");\r
488                                         }\r
489                                         else\r
490                                         {\r
491                                                 sprint (trace_ent, "Leg injury!\n");\r
492                                                 sprint (self, "Leg shot - that'll slow him down!\n");\r
493                                         }\r
494                                 }\r
495                                 return;\r
496                         }\r
497                         else\r
498                         {\r
499                                 if (zdif > 20)\r
500                                 {\r
501                                         dam_mult = 3;\r
502                                         stuffcmd (trace_ent, "bf\n");\r
503                                         deathmsg = 29;\r
504                                         if ((trace_ent.health > 0))\r
505                                         {\r
506                                                 if ((trace_ent.team_no == self.team_no))\r
507                                                 {\r
508                                                         sprint (self, "Stop shooting team mates!!!\n");\r
509                                                 }\r
510                                                 else\r
511                                                 {\r
512                                                         trace_ent.head_shot_vector = (trace_ent.origin - self.origin);\r
513                                                         deathmsg = 29;\r
514                                                         T_Damage (trace_ent, self, self, (self.heat * dam_mult));\r
515                                                         sound (self, 0, "speech/excelent.wav", 1, 0);\r
516 //                                                      if ((trace_ent.health > 0))\r
517 //                                                      {\r
518 //                                                              sprint (trace_ent, 0, "Head injury!\n");\r
519 //                                                              sprint (self, 1, "Head shot - that's gotta hurt!\n");\r
520 //                                                      }\r
521                                                 }\r
522                                                 return;\r
523                                         }\r
524                                         else\r
525                                         {\r
526                                                 deathmsg = 18;\r
527                                         }\r
528                                 }\r
529                         }\r
530                 }\r
531         }\r
532 \r
533         // trace multiple times until we hit a wall, each obstacle will be made\r
534         // non-solid so we can hit the next, while doing this we spawn effects and\r
535         // note down which entities were hit so we can damage them later\r
536         while (1)\r
537         {\r
538                 traceline_hitcorpse (self, start, end, FALSE, self);\r
539 \r
540                 // if it is world we can't hurt it so stop now\r
541                 if (trace_ent == world || trace_fraction == 1)\r
542                         break;\r
543 \r
544                 // make the entity non-solid so we can hit the next one\r
545                 trace_ent.railgunhit = TRUE;\r
546                 trace_ent.railgunhitloc = trace_endpos;\r
547                 trace_ent.railgunhitsolidbackup = trace_ent.solid;\r
548 \r
549                 // stop if this is a wall\r
550                 if (trace_ent.solid == SOLID_BSP)\r
551                         break;\r
552 \r
553                 // make the entity non-solid\r
554                 trace_ent.solid = SOLID_NOT;\r
555         }\r
556 \r
557         // find all the entities the railgun hit and restore their solid state\r
558         ent = findfloat(world, railgunhit, TRUE);\r
559         while (ent)\r
560         {\r
561                 // restore their solid type\r
562                 ent.solid = ent.railgunhitsolidbackup;\r
563                 ent = findfloat(ent, railgunhit, TRUE);\r
564         }\r
565 \r
566         // spawn a temporary explosion entity for RadiusDamage calls\r
567         //explosion = spawn();\r
568 \r
569         // find all the entities the railgun hit and hurt them\r
570         ent = findfloat(world, railgunhit, TRUE);\r
571         while (ent)\r
572         {\r
573                 // get the details we need to call the damage function\r
574                 hitloc = ent.railgunhitloc;\r
575                 ent.railgunhitloc = '0 0 0';\r
576                 ent.railgunhitsolidbackup = SOLID_NOT;\r
577                 ent.railgunhit = FALSE;\r
578 \r
579                 // apply the damage\r
580                 if (ent.takedamage || ent.classname == "case")\r
581                         Damage (ent, self, self, bdamage, deathtype, hitloc, force);\r
582 \r
583                 // create a small explosion to throw gibs around (if applicable)\r
584                 //setorigin (explosion, hitloc);\r
585                 //RadiusDamage (explosion, self, 10, 0, 50, world, 300, deathtype);\r
586 \r
587                 // advance to the next entity\r
588                 ent = findfloat(ent, railgunhit, TRUE);\r
589         }\r
590 \r
591         // we're done with the explosion entity, remove it\r
592         //remove(explosion);\r
593 }\r
594 \r
595 void fireBullet2 (vector start, vector dir, float spread, float damage, float dtype, float tracer, float force)\r
596 {\r
597         vector  end;\r
598         float r;\r
599         local entity e;\r
600 \r
601         // use traceline_hitcorpse to make sure it can hit gibs and corpses too\r
602         dir = dir + randomvec() * spread;\r
603         end = start + dir * 4096;\r
604         traceline_hitcorpse (self, start, end, FALSE, self);\r
605 \r
606         if (tracer)\r
607         {\r
608                 e = spawn();\r
609                 e.owner = self;\r
610                 e.movetype = MOVETYPE_FLY;\r
611                 e.solid = SOLID_NOT;\r
612                 e.think = SUB_Remove;\r
613                 e.nextthink = time + vlen(trace_endpos - start) / 6000;\r
614                 e.velocity = dir * 6000;\r
615                 e.angles = vectoangles(e.velocity);\r
616                 setmodel (e, "models/tracer.mdl");\r
617                 setsize (e, '0 0 0', '0 0 0');\r
618                 setorigin (e, start);\r
619                 e.effects = e.effects | EF_ADDITIVE;\r
620         }\r
621 \r
622         // FIXME - causes excessive 'tinking'. Hopefully remove "tink1.wav" from the ricochets with csqc\r
623         if ((trace_fraction != 1.0) && (pointcontents (trace_endpos) != CONTENT_SKY))\r
624         {\r
625                 if (trace_ent.solid == SOLID_BSP)\r
626                 {\r
627                         pointcontents (self.origin);\r
628                         te_gunshot (trace_endpos);\r
629                         r = random ();\r
630                         if (r < 0.10)\r
631                                 PointSound (trace_endpos, "weapons/ric1.wav", 1, ATTN_NORM);\r
632                         else if (r < 0.20)\r
633                                 PointSound (trace_endpos, "weapons/ric2.wav", 1, ATTN_NORM);\r
634                         else if (r < 0.30)\r
635                                 PointSound (trace_endpos, "weapons/ric3.wav", 1, ATTN_NORM);\r
636                 }\r
637                 else if (trace_ent.classname == "player" || trace_ent.classname == "corpse" || trace_ent.classname == "gib")\r
638                         sound (self, CHAN_BODY, "misc/hit.wav", 1, ATTN_NORM);\r
639                 Damage (trace_ent, self, self, damage, dtype, trace_endpos, dir * force);\r
640         }\r
641 }\r
642 \r
643 void fireBullet (vector start, vector dir, float spread, float damage, float dtype, float tracer)\r
644 {\r
645         fireBullet2(start, dir, spread, damage, dtype, tracer, damage * 5); // default force value\r
646 }\r
647 \r
648 /*\r
649 void W_Attack (void)\r
650 {\r
651         if (self.deadflag != DEAD_NO)\r
652         {\r
653                 if (self.death_time < time)\r
654                         PutClientInServer();\r
655 \r
656                 return;\r
657         }\r
658 \r
659         if (!W_CheckAmmo ())\r
660                 return;\r
661 \r
662         makevectors (self.v_angle);\r
663         //if (self.weapon == IT_LASER)\r
664         //      W_Laser_Attack ();\r
665         //if (self.weapon == IT_SHOTGUN)\r
666                 //W_Shotgun_Attack ();\r
667         //else if (self.weapon == IT_UZI)\r
668                 //W_Uzi_Attack ();\r
669         if (self.weapon == IT_CRYLINK)\r
670                 W_Crylink_Attack ();\r
671         else if (self.weapon == IT_ELECTRO)\r
672                 {\r
673                 W_Electro_Attack (self.electrocount);\r
674                 self.electrocount = self.electrocount + 1;\r
675                 if (self.electrocount == 3)\r
676                         self.electrocount = 0;\r
677                 }\r
678         else if (self.weapon == IT_GRENADE_LAUNCHER)\r
679                 W_Grenade_Attack ();\r
680         else if (self.weapon == IT_HAGAR)\r
681                 W_Hagar_Attack ();\r
682         else if (self.weapon == IT_NEX)\r
683                 W_Nex_Attack ();\r
684         //else if (self.weapon == IT_ROCKET_LAUNCHER)\r
685         //      W_Rocket_Attack ();\r
686 \r
687         W_UpdateAmmo ();\r
688 }\r
689 \r
690 void W_SecondaryAttack (void)\r
691 {\r
692         if (self.deadflag != DEAD_NO)\r
693         {\r
694                 if (self.death_time < time)\r
695                         PutClientInServer();\r
696 \r
697                 return;\r
698         }\r
699 \r
700         if (!W_CheckAmmo ())\r
701                 return;\r
702 \r
703         makevectors (self.v_angle);\r
704         //if (self.weapon == IT_LASER)\r
705                 //W_Laser_Attack2 ();\r
706         //if (self.weapon == IT_SHOTGUN)\r
707                 //W_Shotgun_Attack2 ();\r
708         //else if (self.weapon == IT_UZI)\r
709                 //W_Uzi_Attack2 ();\r
710         else if (self.weapon == IT_CRYLINK)\r
711                 W_Crylink_Attack2 ();\r
712         else if (self.weapon == IT_ELECTRO) {\r
713                 W_Electro_Attack2 (self.electrocount);\r
714                 self.electrocount = self.electrocount + 1;\r
715                 if (self.electrocount == 3)\r
716                         self.electrocount = 0;\r
717                 }\r
718         else if (self.weapon == IT_GRENADE_LAUNCHER)\r
719                 W_Grenade_Attack2 ();\r
720         else if (self.weapon == IT_HAGAR)\r
721                 W_Hagar_Attack2 ();\r
722         else if (self.weapon == IT_NEX)\r
723                 W_Nex_Attack2 ();\r
724         //else if (self.weapon == IT_ROCKET_LAUNCHER)\r
725                 //W_Rocket_Attack2 ();\r
726 \r
727         W_UpdateAmmo ();\r
728 }\r
729 \r
730 void W_ThirdAttack (void)\r
731 {\r
732         if (self.deadflag != DEAD_NO)\r
733         {\r
734                 if (self.death_time < time)\r
735                         PutClientInServer();\r
736 \r
737                 return;\r
738         }\r
739 \r
740         if (!W_CheckAmmo ())\r
741                 return;\r
742 \r
743         makevectors (self.v_angle);\r
744         //if (self.weapon == IT_LASER)\r
745                 //W_Laser_Attack2 ();\r
746         //if (self.weapon == IT_SHOTGUN)\r
747                 //W_Shotgun_Attack2 ();\r
748         //else if (self.weapon == IT_UZI)\r
749                 //W_Uzi_Attack3 ();\r
750         else if (self.weapon == IT_CRYLINK)\r
751                 W_Crylink_Attack2 ();\r
752         else if (self.weapon == IT_ELECTRO) {\r
753                 W_Electro_Attack3 (self.electrocount);\r
754                 self.electrocount = self.electrocount + 1;\r
755                 if (self.electrocount == 3)\r
756                         self.electrocount = 0;\r
757                 }\r
758         else if (self.weapon == IT_GRENADE_LAUNCHER)\r
759                 W_Grenade_Attack3 ();\r
760         else if (self.weapon == IT_HAGAR)\r
761                 W_Hagar_Attack3 ();\r
762         else if (self.weapon == IT_NEX)\r
763                 W_Nex_Attack2 ();\r
764         //else if (self.weapon == IT_ROCKET_LAUNCHER)\r
765                 //W_Rocket_Attack3 ();\r
766 \r
767         W_UpdateAmmo ();\r
768 }\r
769 */\r
770 \r
771 float RateFlameDamage(float flametime, float flamedmg, float flamerate)\r
772 {\r
773         // return (how much longer flame will last) * (how much to damage per think / how often to think)\r
774         return (flametime - time) * (flamedmg / flamerate);\r
775 }\r
776 \r
777 /*void FlamePuffThink()\r
778 {\r
779         self.nextthink = time + self.cnt;\r
780         self.frame = self.frame + 1;\r
781         if(self.frame >= self.count)\r
782                 self.think = SUB_Remove;\r
783 }\r
784 */\r
785 void FlameBurnTarget();\r
786 \r
787 \r
788 entity CreateFlame(entity targ, entity attacker)\r
789 {\r
790         entity f;\r
791         f = targ.onfire = spawn();\r
792         f.classname = "burning";\r
793         f.owner = attacker;\r
794         f.enemy = targ;\r
795         f.dmg = 0;\r
796         f.ltime = 0;\r
797         f.wait = 0;\r
798         f.think = FlameBurnTarget;\r
799         f.onfire = spawn();\r
800 \r
801         f.effects = f.onfire.effects = EF_ADDITIVE;\r
802         f.scale = f.onfire.scale = 2;\r
803 \r
804         setmodel(f, "models/sprites/fire_top.spr32");\r
805         setattachment(f, f.enemy, "");\r
806         setmodel(f.onfire, "models/sprites/fire_base.spr32");\r
807 \r
808         setorigin(f.onfire, '0 0 -5' * f.scale);\r
809         setorigin(f, '0 0 2' * f.scale + '0 0 20');\r
810         setattachment(f.onfire, f, "");\r
811 \r
812         return f;\r
813 }\r
814 \r
815 entity IgniteTarget (entity targ, entity attacker, float flametime, float flamedmg, float flamerate, float rateflame)\r
816 {\r
817         entity f;\r
818 \r
819         //bprint("ignite targets?\n");\r
820 \r
821         if(targ.class == CLASS_PYRO)\r
822         {\r
823                 //flametime = flametime / 5; // pyros don't stay on fire for long\r
824                 return world; // don't set pyros on fire\r
825         }\r
826 \r
827         if(targ.class == CLASS_MEDIC)\r
828         {\r
829                 //flametime = flametime / 3; // medics don't stay on fire for long\r
830                 return world; // don't set medics on fire\r
831         }\r
832 \r
833         f = targ.onfire;\r
834         if(f == world)\r
835         {\r
836                 f = CreateFlame(targ, attacker);\r
837         }\r
838         else\r
839         {\r
840                 if(f.nextthink <= 0)\r
841                         f.nextthink = time + flamerate;\r
842         }\r
843 \r
844 \r
845 //      bprint(ftos(f.wait), ", ", ftos(f.dmg), ", ", ftos(f.ltime), "\n");\r
846         //bprint(ftos(flametime), ", ", ftos(flamedmg), ", ", ftos(flamerate), "\n");\r
847 //      bprint(ftos(RateFlameDamage(f.wait, f.dmg, f.ltime)), ", ", ftos(RateFlameDamage(flametime, flamedmg, flamerate)), "\n");\r
848 \r
849         if(rateflame && RateFlameDamage(f.wait, f.dmg, f.ltime) > RateFlameDamage(time + flametime, flamedmg, flamerate))\r
850         {\r
851                 f.owner = attacker; // give the new attacker ownership of the burn damage\r
852                 return f; // current damage rating is larger, don't replace it\r
853         }\r
854 \r
855         //bprint("burning success: ", ftos(flametime), "\n");\r
856 \r
857         f.dmg = flamedmg;\r
858         f.ltime = flamerate;\r
859         f.wait = time + flametime;\r
860 \r
861         if(!f.nextthink || f.nextthink > time + f.ltime)\r
862                 f.nextthink = time + f.ltime;\r
863 \r
864         return f;\r
865 }\r
866 \r
867 void ExtinguishFlame(entity targ)\r
868 {\r
869         entity f, b;\r
870         if(!targ.onfire)//targ.onfire.classname != "burning")\r
871                 return;\r
872         //bprint(strcat("ExtinguishFlame(", targ.classname, ") ", targ.onfire.classname, "\n"));\r
873 \r
874         f = targ.onfire;\r
875         b = targ.onfire.onfire;\r
876 \r
877         // fixme: hiss sound\r
878 \r
879         if(b)\r
880         {\r
881                 b.think = SUB_Null;\r
882                 b.nextthink = -1;\r
883                 setmodel(b, "models/sprites/null.spr");\r
884                 setattachment(b, world, "");\r
885                 remove(b); // remove base model\r
886                 //targ.onfire.onfire.think = SUB_Remove;\r
887                 //targ.onfire.onfire.nextthink = time + 0.1;\r
888                 f.onfire = world;\r
889         }\r
890         f.think = SUB_Null;\r
891         f.nextthink = -1;\r
892         setmodel(f, "models/sprites/null.spr");\r
893         setattachment(f, world, "");\r
894         //targ.onfire.think = SUB_Remove;\r
895         //targ.onfire.nextthink = time + 0.1;\r
896         remove(f);\r
897         targ.onfire = world;\r
898 }\r
899 \r
900 void FlameBurnTarget()\r
901 {\r
902         entity head;\r
903         float flametime, flametimemax, radius, flameratio, distratio, edgeratio, damage;\r
904         //bprint("flame burn target\n");\r
905 \r
906         if(self.enemy == world || self.enemy.classname == "gib")\r
907         {\r
908                 if(self.enemy != world)\r
909                 {\r
910                         //bprint("on a gib, remove flame\n");\r
911                         //self.enemy.onfire = world;\r
912                         ExtinguishFlame(self.enemy);\r
913                         return;\r
914                 }\r
915                 //bprint("enemy is world, remove flame\n");\r
916                 self.think = SUB_Null;\r
917                 self.nextthink = -1;\r
918                 setmodel(self, "models/sprites/null.spr");\r
919                 setattachment(self, world, "");\r
920                 if(self.onfire)\r
921                 {\r
922                         self.onfire.think = SUB_Null;\r
923                         self.onfire.nextthink = -1;\r
924                         setmodel(self.onfire, "models/sprites/null.spr");\r
925                         setattachment(self.onfire, world, "");\r
926                         remove(self.onfire);\r
927                         self.onfire = world;\r
928                 }\r
929                 remove(self);\r
930                 return;\r
931         }\r
932 \r
933         if(self.wait < time || self.enemy.waterlevel > 2 || (!self.enemy.takedamage && self.enemy.classname != "grenade"))\r
934         {\r
935                 /*if(self.enemy != world)\r
936                 {\r
937                         self.enemy.effects = self.enemy.effects - (self.enemy.effects & EF_FLAME);\r
938                         self.enemy.onfire = world;\r
939                 }\r
940                 remove(self);*/\r
941                 ExtinguishFlame(self.enemy);\r
942                 return;\r
943         }\r
944         self.nextthink = time + self.ltime;\r
945 \r
946         Damage (self.enemy, self, self.owner, self.dmg, DEATH_BURNING, self.enemy.origin - '0 0 5', '0 0 0');//'0 0 -1' * self.dmg);\r
947 \r
948         // flames spread to other targets\r
949 \r
950         radius = cvar("g_balance_heat_radius");\r
951         head = findradius(self.enemy.origin, radius);\r
952         while(head)\r
953         {\r
954                 if((head.takedamage || head.classname == "grenade") && head != self.enemy)\r
955                 {\r
956                         //bprint(strcat("maybe burn ", head.classname, "\n"));\r
957                         // don't harm or spread to allies or pyros\r
958                         //if(!(head.classname == "player" && (head.team == self.owner.team || head.class == CLASS_PYRO)) )\r
959                         if(head.classname == "player")\r
960                         {\r
961                                 if(head.class == CLASS_PYRO || head.class == CLASS_MEDIC) // pyros & medics are mostly immune to fire\r
962                                 {       head = head.chain;      continue;}\r
963                                 if(head.team == self.owner.team && teamplay) // don't hurt teammates\r
964                                 {       head = head.chain;      continue;}\r
965                         }\r
966 \r
967                         traceline(self.enemy.origin, head.origin, TRUE, self);\r
968                         if(trace_fraction >= 1)\r
969                         {\r
970                                 //bprint(strcat("burn ", head.classname, "\n"));\r
971                                 edgeratio = cvar("g_balance_heat_edgeratio");\r
972                                 distratio = ( 1 - (vlen(self.enemy.origin - head.origin) / radius) );\r
973                                 flameratio = edgeratio + distratio*(1 - edgeratio);\r
974 \r
975                                 // increase the player's heat; if it goes over the max it'll get capped automatically by the cooldown code\r
976                                 //head.flame_heat = cvar("g_balance_maxheat");\r
977                                 head.flame_heat = head.flame_heat + flameratio * cvar("g_balance_heatup_rate") * self.ltime;\r
978                                 //head.flame_heat_time = time;\r
979 \r
980                                 // deal direct damage from the heat radiated by the fire\r
981                                 damage = flameratio * self.dmg*cvar("g_balance_heat_damage");\r
982                                 Damage (head, self, self.owner, damage, DEATH_BURNING, self.enemy.origin, '0 0 0');\r
983 \r
984                                 if(head.flame_heat >= 1.0 && head.takedamage == DAMAGE_AIM && head.health > 0)\r
985                                 {\r
986                                         flametimemax = cvar("g_balance_heat_timemax");\r
987                                         flametime = (self.wait - time) * cvar("g_balance_heat_transfer");\r
988                                         if(flametime > flametimemax)\r
989                                                 flametime = flametimemax;\r
990 \r
991 \r
992                                         //bprint(ftos(time), " - flame transfer - ", ftos(flametime), "\n");\r
993 \r
994                                         \r
995                                         IgniteTarget(head, self.owner, flametime, self.dmg, self.ltime, TRUE);\r
996                                 }\r
997                         }\r
998                 }\r
999                 head = head.chain;\r
1000         }\r
1001 }\r
1002 \r
1003 \r
1004 void PoisonTarget(entity targ, float poisonDamage, float poisonTime, float ratepoison)\r
1005 {\r
1006         if(targ.class == CLASS_MEDIC)\r
1007         {\r
1008                 return; // don't poison medics\r
1009         }\r
1010 \r
1011         if(ratepoison && targ.poison_damage > poisonDamage)\r
1012                 return; // current damage rating is larger, don't replace it\r
1013 \r
1014         targ.poison_damage = poisonDamage;\r
1015         targ.poison_rate = poisonDamage / poisonTime;\r
1016 }\r
1017 \r
1018 \r
1019 \r
1020 \r
1021 \r
1022 \r
1023 \r
1024 \r
1025 \r
1026 \r
1027 \r
1028 float W_LimitNumEnts(string clname, float num_allowed, void() DeathFunc)\r
1029 {\r
1030         float num, oldest;\r
1031         local entity e, selected;\r
1032         e = world;\r
1033         oldest = time + 1;\r
1034         num = 0;\r
1035         do\r
1036         {\r
1037                 e = find(e, classname, clname);\r
1038                 if(e != world && e.owner == self)\r
1039                 {\r
1040                         num = num + 1;\r
1041                         if(e.create_time < oldest)\r
1042                         {\r
1043                                 selected = e;\r
1044                                 oldest = e.create_time;\r
1045                         }\r
1046                 }\r
1047         }while(e);\r
1048 \r
1049         if(num > num_allowed)\r
1050         {\r
1051                 if(selected != world)\r
1052                 {\r
1053                         e = self;\r
1054                         self = selected;\r
1055                         DeathFunc();\r
1056                         self = e;\r
1057                         return TRUE;\r
1058                 }\r
1059                 else\r
1060                         bprint("error (W_LimitNumEnts): cannot find oldest entity\n");\r
1061         }\r
1062         return FALSE;\r
1063 }\r
1064 \r