icculus quake2 doxygen  1.0 dev
g_weapon.c
Go to the documentation of this file.
1 /*
2 Copyright (C) 1997-2001 Id Software, Inc.
3 
4 This program is free software; you can redistribute it and/or
5 modify it under the terms of the GNU General Public License
6 as published by the Free Software Foundation; either version 2
7 of the License, or (at your option) any later version.
8 
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
12 
13 See the GNU General Public License for more details.
14 
15 You should have received a copy of the GNU General Public License
16 along with this program; if not, write to the Free Software
17 Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
18 
19 */
20 #include "g_local.h"
21 
22 
23 /*
24 =================
25 check_dodge
26 
27 This is a support routine used when a client is firing
28 a non-instant attack weapon. It checks to see if a
29 monster's dodge function should be called.
30 =================
31 */
32 static void check_dodge (edict_t *self, vec3_t start, vec3_t dir, int speed)
33 {
34  vec3_t end;
35  vec3_t v;
36  trace_t tr;
37  float eta;
38 
39  // easy mode only ducks one quarter the time
40  if (skill->value == 0)
41  {
42  if (random() > 0.25)
43  return;
44  }
45  VectorMA (start, 8192, dir, end);
46  tr = gi.trace (start, NULL, NULL, end, self, MASK_SHOT);
47  if ((tr.ent) && (tr.ent->svflags & SVF_MONSTER) && (tr.ent->health > 0) && (tr.ent->monsterinfo.dodge) && infront(tr.ent, self))
48  {
49  VectorSubtract (tr.endpos, start, v);
50  eta = (VectorLength(v) - tr.ent->maxs[0]) / speed;
51  tr.ent->monsterinfo.dodge (tr.ent, self, eta);
52  }
53 }
54 
55 
56 /*
57 =================
58 fire_hit
59 
60 Used for all impact (hit/punch/slash) attacks
61 =================
62 */
63 qboolean fire_hit (edict_t *self, vec3_t aim, int damage, int kick)
64 {
65  trace_t tr;
67  vec3_t v;
68  vec3_t point;
69  float range;
70  vec3_t dir;
71 
72  //see if enemy is in range
73  VectorSubtract (self->enemy->s.origin, self->s.origin, dir);
74  range = VectorLength(dir);
75  if (range > aim[0])
76  return false;
77 
78  if (aim[1] > self->mins[0] && aim[1] < self->maxs[0])
79  {
80  // the hit is straight on so back the range up to the edge of their bbox
81  range -= self->enemy->maxs[0];
82  }
83  else
84  {
85  // this is a side hit so adjust the "right" value out to the edge of their bbox
86  if (aim[1] < 0)
87  aim[1] = self->enemy->mins[0];
88  else
89  aim[1] = self->enemy->maxs[0];
90  }
91 
92  VectorMA (self->s.origin, range, dir, point);
93 
94  tr = gi.trace (self->s.origin, NULL, NULL, point, self, MASK_SHOT);
95  if (tr.fraction < 1)
96  {
97  if (!tr.ent->takedamage)
98  return false;
99  // if it will hit any client/monster then hit the one we wanted to hit
100  if ((tr.ent->svflags & SVF_MONSTER) || (tr.ent->client))
101  tr.ent = self->enemy;
102  }
103 
104  AngleVectors(self->s.angles, forward, right, up);
105  VectorMA (self->s.origin, range, forward, point);
106  VectorMA (point, aim[1], right, point);
107  VectorMA (point, aim[2], up, point);
108  VectorSubtract (point, self->enemy->s.origin, dir);
109 
110  // do the damage
111  T_Damage (tr.ent, self, self, dir, point, vec3_origin, damage, kick/2, DAMAGE_NO_KNOCKBACK, MOD_HIT);
112 
113  if (!(tr.ent->svflags & SVF_MONSTER) && (!tr.ent->client))
114  return false;
115 
116  // do our special form of knockback here
117  VectorMA (self->enemy->absmin, 0.5, self->enemy->size, v);
118  VectorSubtract (v, point, v);
119  VectorNormalize (v);
120  VectorMA (self->enemy->velocity, kick, v, self->enemy->velocity);
121  if (self->enemy->velocity[2] > 0)
122  self->enemy->groundentity = NULL;
123  return true;
124 }
125 
126 
127 /*
128 =================
129 fire_lead
130 
131 This is an internal support routine used for bullet/pellet based weapons.
132 =================
133 */
134 static void fire_lead (edict_t *self, vec3_t start, vec3_t aimdir, int damage, int kick, int te_impact, int hspread, int vspread, int mod)
135 {
136  trace_t tr;
137  vec3_t dir;
139  vec3_t end;
140  float r;
141  float u;
142  vec3_t water_start;
143  qboolean water = false;
144  int content_mask = MASK_SHOT | MASK_WATER;
145 
146  tr = gi.trace (self->s.origin, NULL, NULL, start, self, MASK_SHOT);
147  if (!(tr.fraction < 1.0))
148  {
149  vectoangles (aimdir, dir);
150  AngleVectors (dir, forward, right, up);
151 
152  r = crandom()*hspread;
153  u = crandom()*vspread;
154  VectorMA (start, 8192, forward, end);
155  VectorMA (end, r, right, end);
156  VectorMA (end, u, up, end);
157 
158  if (gi.pointcontents (start) & MASK_WATER)
159  {
160  water = true;
161  VectorCopy (start, water_start);
162  content_mask &= ~MASK_WATER;
163  }
164 
165  tr = gi.trace (start, NULL, NULL, end, self, content_mask);
166 
167  // see if we hit water
168  if (tr.contents & MASK_WATER)
169  {
170  int color;
171 
172  water = true;
173  VectorCopy (tr.endpos, water_start);
174 
175  if (!VectorCompare (start, tr.endpos))
176  {
177  if (tr.contents & CONTENTS_WATER)
178  {
179  if (strcmp(tr.surface->name, "*brwater") == 0)
180  color = SPLASH_BROWN_WATER;
181  else
182  color = SPLASH_BLUE_WATER;
183  }
184  else if (tr.contents & CONTENTS_SLIME)
185  color = SPLASH_SLIME;
186  else if (tr.contents & CONTENTS_LAVA)
187  color = SPLASH_LAVA;
188  else
189  color = SPLASH_UNKNOWN;
190 
191  if (color != SPLASH_UNKNOWN)
192  {
195  gi.WriteByte (8);
196  gi.WritePosition (tr.endpos);
197  gi.WriteDir (tr.plane.normal);
198  gi.WriteByte (color);
200  }
201 
202  // change bullet's course when it enters water
203  VectorSubtract (end, start, dir);
204  vectoangles (dir, dir);
205  AngleVectors (dir, forward, right, up);
206  r = crandom()*hspread*2;
207  u = crandom()*vspread*2;
208  VectorMA (water_start, 8192, forward, end);
209  VectorMA (end, r, right, end);
210  VectorMA (end, u, up, end);
211  }
212 
213  // re-trace ignoring water this time
214  tr = gi.trace (water_start, NULL, NULL, end, self, MASK_SHOT);
215  }
216  }
217 
218  // send gun puff / flash
219  if (!((tr.surface) && (tr.surface->flags & SURF_SKY)))
220  {
221  if (tr.fraction < 1.0)
222  {
223  if (tr.ent->takedamage)
224  {
225  T_Damage (tr.ent, self, self, aimdir, tr.endpos, tr.plane.normal, damage, kick, DAMAGE_BULLET, mod);
226  }
227  else
228  {
229  if (strncmp (tr.surface->name, "sky", 3) != 0)
230  {
232  gi.WriteByte (te_impact);
233  gi.WritePosition (tr.endpos);
234  gi.WriteDir (tr.plane.normal);
236 
237  if (self->client)
238  PlayerNoise(self, tr.endpos, PNOISE_IMPACT);
239  }
240  }
241  }
242  }
243 
244  // if went through water, determine where the end and make a bubble trail
245  if (water)
246  {
247  vec3_t pos;
248 
249  VectorSubtract (tr.endpos, water_start, dir);
250  VectorNormalize (dir);
251  VectorMA (tr.endpos, -2, dir, pos);
252  if (gi.pointcontents (pos) & MASK_WATER)
253  VectorCopy (pos, tr.endpos);
254  else
255  tr = gi.trace (pos, NULL, NULL, water_start, tr.ent, MASK_WATER);
256 
257  VectorAdd (water_start, tr.endpos, pos);
258  VectorScale (pos, 0.5, pos);
259 
262  gi.WritePosition (water_start);
263  gi.WritePosition (tr.endpos);
264  gi.multicast (pos, MULTICAST_PVS);
265  }
266 }
267 
268 
269 /*
270 =================
271 fire_bullet
272 
273 Fires a single round. Used for machinegun and chaingun. Would be fine for
274 pistols, rifles, etc....
275 =================
276 */
277 void fire_bullet (edict_t *self, vec3_t start, vec3_t aimdir, int damage, int kick, int hspread, int vspread, int mod)
278 {
279  fire_lead (self, start, aimdir, damage, kick, TE_GUNSHOT, hspread, vspread, mod);
280 }
281 
282 
283 /*
284 =================
285 fire_shotgun
286 
287 Shoots shotgun pellets. Used by shotgun and super shotgun.
288 =================
289 */
290 void fire_shotgun (edict_t *self, vec3_t start, vec3_t aimdir, int damage, int kick, int hspread, int vspread, int count, int mod)
291 {
292  int i;
293 
294  for (i = 0; i < count; i++)
295  fire_lead (self, start, aimdir, damage, kick, TE_SHOTGUN, hspread, vspread, mod);
296 }
297 
298 
299 /*
300 =================
301 fire_blaster
302 
303 Fires a single blaster bolt. Used by the blaster and hyper blaster.
304 =================
305 */
306 void blaster_touch (edict_t *self, edict_t *other, cplane_t *plane, csurface_t *surf)
307 {
308  int mod;
309 
310  if (other == self->owner)
311  return;
312 
313  if (surf && (surf->flags & SURF_SKY))
314  {
315  G_FreeEdict (self);
316  return;
317  }
318 
319  if (self->owner->client)
320  PlayerNoise(self->owner, self->s.origin, PNOISE_IMPACT);
321 
322  if (other->takedamage)
323  {
324  if (self->spawnflags & 1)
325  mod = MOD_HYPERBLASTER;
326  else
327  mod = MOD_BLASTER;
328  T_Damage (other, self, self->owner, self->velocity, self->s.origin, plane->normal, self->dmg, 1, DAMAGE_ENERGY, mod);
329  }
330  else
331  {
334  gi.WritePosition (self->s.origin);
335  if (!plane)
337  else
338  gi.WriteDir (plane->normal);
339  gi.multicast (self->s.origin, MULTICAST_PVS);
340  }
341 
342  G_FreeEdict (self);
343 }
344 
345 void fire_blaster (edict_t *self, vec3_t start, vec3_t dir, int damage, int speed, int effect, qboolean hyper)
346 {
347  edict_t *bolt;
348  trace_t tr;
349 
350  VectorNormalize (dir);
351 
352  bolt = G_Spawn();
353  bolt->svflags = SVF_DEADMONSTER;
354  // yes, I know it looks weird that projectiles are deadmonsters
355  // what this means is that when prediction is used against the object
356  // (blaster/hyperblaster shots), the player won't be solid clipped against
357  // the object. Right now trying to run into a firing hyperblaster
358  // is very jerky since you are predicted 'against' the shots.
359  VectorCopy (start, bolt->s.origin);
360  VectorCopy (start, bolt->s.old_origin);
361  vectoangles (dir, bolt->s.angles);
362  VectorScale (dir, speed, bolt->velocity);
364  bolt->clipmask = MASK_SHOT;
365  bolt->solid = SOLID_BBOX;
366  bolt->s.effects |= effect;
367  VectorClear (bolt->mins);
368  VectorClear (bolt->maxs);
369  bolt->s.modelindex = gi.modelindex ("models/objects/laser/tris.md2");
370  bolt->s.sound = gi.soundindex ("misc/lasfly.wav");
371  bolt->owner = self;
372  bolt->touch = blaster_touch;
373  bolt->nextthink = level.time + 2;
374  bolt->think = G_FreeEdict;
375  bolt->dmg = damage;
376  bolt->classname = "bolt";
377  if (hyper)
378  bolt->spawnflags = 1;
379  gi.linkentity (bolt);
380 
381  if (self->client)
382  check_dodge (self, bolt->s.origin, dir, speed);
383 
384  tr = gi.trace (self->s.origin, NULL, NULL, bolt->s.origin, bolt, MASK_SHOT);
385  if (tr.fraction < 1.0)
386  {
387  VectorMA (bolt->s.origin, -10, dir, bolt->s.origin);
388  bolt->touch (bolt, tr.ent, NULL, NULL);
389  }
390 }
391 
392 
393 /*
394 =================
395 fire_grenade
396 =================
397 */
398 static void Grenade_Explode (edict_t *ent)
399 {
400  vec3_t origin;
401  int mod;
402 
403  if (ent->owner->client)
404  PlayerNoise(ent->owner, ent->s.origin, PNOISE_IMPACT);
405 
406  //FIXME: if we are onground then raise our Z just a bit since we are a point?
407  if (ent->enemy)
408  {
409  float points;
410  vec3_t v;
411  vec3_t dir;
412 
413  VectorAdd (ent->enemy->mins, ent->enemy->maxs, v);
414  VectorMA (ent->enemy->s.origin, 0.5, v, v);
415  VectorSubtract (ent->s.origin, v, v);
416  points = ent->dmg - 0.5 * VectorLength (v);
417  VectorSubtract (ent->enemy->s.origin, ent->s.origin, dir);
418  if (ent->spawnflags & 1)
419  mod = MOD_HANDGRENADE;
420  else
421  mod = MOD_GRENADE;
422  T_Damage (ent->enemy, ent, ent->owner, dir, ent->s.origin, vec3_origin, (int)points, (int)points, DAMAGE_RADIUS, mod);
423  }
424 
425  if (ent->spawnflags & 2)
426  mod = MOD_HELD_GRENADE;
427  else if (ent->spawnflags & 1)
428  mod = MOD_HG_SPLASH;
429  else
430  mod = MOD_G_SPLASH;
431  T_RadiusDamage(ent, ent->owner, ent->dmg, ent->enemy, ent->dmg_radius, mod);
432 
433  VectorMA (ent->s.origin, -0.02, ent->velocity, origin);
435  if (ent->waterlevel)
436  {
437  if (ent->groundentity)
439  else
441  }
442  else
443  {
444  if (ent->groundentity)
446  else
448  }
449  gi.WritePosition (origin);
451 
452  G_FreeEdict (ent);
453 }
454 
455 static void Grenade_Touch (edict_t *ent, edict_t *other, cplane_t *plane, csurface_t *surf)
456 {
457  if (other == ent->owner)
458  return;
459 
460  if (surf && (surf->flags & SURF_SKY))
461  {
462  G_FreeEdict (ent);
463  return;
464  }
465 
466  if (!other->takedamage)
467  {
468  if (ent->spawnflags & 1)
469  {
470  if (random() > 0.5)
471  gi.sound (ent, CHAN_VOICE, gi.soundindex ("weapons/hgrenb1a.wav"), 1, ATTN_NORM, 0);
472  else
473  gi.sound (ent, CHAN_VOICE, gi.soundindex ("weapons/hgrenb2a.wav"), 1, ATTN_NORM, 0);
474  }
475  else
476  {
477  gi.sound (ent, CHAN_VOICE, gi.soundindex ("weapons/grenlb1b.wav"), 1, ATTN_NORM, 0);
478  }
479  return;
480  }
481 
482  ent->enemy = other;
483  Grenade_Explode (ent);
484 }
485 
486 void fire_grenade (edict_t *self, vec3_t start, vec3_t aimdir, int damage, int speed, float timer, float damage_radius)
487 {
488  edict_t *grenade;
489  vec3_t dir;
491 
492  vectoangles (aimdir, dir);
493  AngleVectors (dir, forward, right, up);
494 
495  grenade = G_Spawn();
496  VectorCopy (start, grenade->s.origin);
497  VectorScale (aimdir, speed, grenade->velocity);
498  VectorMA (grenade->velocity, 200 + crandom() * 10.0, up, grenade->velocity);
499  VectorMA (grenade->velocity, crandom() * 10.0, right, grenade->velocity);
500  VectorSet (grenade->avelocity, 300, 300, 300);
501  grenade->movetype = MOVETYPE_BOUNCE;
502  grenade->clipmask = MASK_SHOT;
503  grenade->solid = SOLID_BBOX;
504  grenade->s.effects |= EF_GRENADE;
505  VectorClear (grenade->mins);
506  VectorClear (grenade->maxs);
507  grenade->s.modelindex = gi.modelindex ("models/objects/grenade/tris.md2");
508  grenade->owner = self;
509  grenade->touch = Grenade_Touch;
510  grenade->nextthink = level.time + timer;
511  grenade->think = Grenade_Explode;
512  grenade->dmg = damage;
513  grenade->dmg_radius = damage_radius;
514  grenade->classname = "grenade";
515 
516  gi.linkentity (grenade);
517 }
518 
519 void fire_grenade2 (edict_t *self, vec3_t start, vec3_t aimdir, int damage, int speed, float timer, float damage_radius, qboolean held)
520 {
521  edict_t *grenade;
522  vec3_t dir;
524 
525  vectoangles (aimdir, dir);
526  AngleVectors (dir, forward, right, up);
527 
528  grenade = G_Spawn();
529  VectorCopy (start, grenade->s.origin);
530  VectorScale (aimdir, speed, grenade->velocity);
531  VectorMA (grenade->velocity, 200 + crandom() * 10.0, up, grenade->velocity);
532  VectorMA (grenade->velocity, crandom() * 10.0, right, grenade->velocity);
533  VectorSet (grenade->avelocity, 300, 300, 300);
534  grenade->movetype = MOVETYPE_BOUNCE;
535  grenade->clipmask = MASK_SHOT;
536  grenade->solid = SOLID_BBOX;
537  grenade->s.effects |= EF_GRENADE;
538  VectorClear (grenade->mins);
539  VectorClear (grenade->maxs);
540  grenade->s.modelindex = gi.modelindex ("models/objects/grenade2/tris.md2");
541  grenade->owner = self;
542  grenade->touch = Grenade_Touch;
543  grenade->nextthink = level.time + timer;
544  grenade->think = Grenade_Explode;
545  grenade->dmg = damage;
546  grenade->dmg_radius = damage_radius;
547  grenade->classname = "hgrenade";
548  if (held)
549  grenade->spawnflags = 3;
550  else
551  grenade->spawnflags = 1;
552  grenade->s.sound = gi.soundindex("weapons/hgrenc1b.wav");
553 
554  if (timer <= 0.0)
555  Grenade_Explode (grenade);
556  else
557  {
558  gi.sound (self, CHAN_WEAPON, gi.soundindex ("weapons/hgrent1a.wav"), 1, ATTN_NORM, 0);
559  gi.linkentity (grenade);
560  }
561 }
562 
563 
564 /*
565 =================
566 fire_rocket
567 =================
568 */
569 void rocket_touch (edict_t *ent, edict_t *other, cplane_t *plane, csurface_t *surf)
570 {
571  vec3_t origin;
572  int n;
573 
574  if (other == ent->owner)
575  return;
576 
577  if (surf && (surf->flags & SURF_SKY))
578  {
579  G_FreeEdict (ent);
580  return;
581  }
582 
583  if (ent->owner->client)
584  PlayerNoise(ent->owner, ent->s.origin, PNOISE_IMPACT);
585 
586  // calculate position for the explosion entity
587  VectorMA (ent->s.origin, -0.02, ent->velocity, origin);
588 
589  if (other->takedamage)
590  {
591  T_Damage (other, ent, ent->owner, ent->velocity, ent->s.origin, plane->normal, ent->dmg, 0, 0, MOD_ROCKET);
592  }
593  else
594  {
595  // don't throw any debris in net games
596  if (!deathmatch->value && !coop->value)
597  {
598  if ((surf) && !(surf->flags & (SURF_WARP|SURF_TRANS33|SURF_TRANS66|SURF_FLOWING)))
599  {
600  n = rand() % 5;
601  while(n--)
602  ThrowDebris (ent, "models/objects/debris2/tris.md2", 2, ent->s.origin);
603  }
604  }
605  }
606 
607  T_RadiusDamage(ent, ent->owner, ent->radius_dmg, other, ent->dmg_radius, MOD_R_SPLASH);
608 
610  if (ent->waterlevel)
612  else
614  gi.WritePosition (origin);
616 
617  G_FreeEdict (ent);
618 }
619 
620 void fire_rocket (edict_t *self, vec3_t start, vec3_t dir, int damage, int speed, float damage_radius, int radius_damage)
621 {
622  edict_t *rocket;
623 
624  rocket = G_Spawn();
625  VectorCopy (start, rocket->s.origin);
626  VectorCopy (dir, rocket->movedir);
627  vectoangles (dir, rocket->s.angles);
628  VectorScale (dir, speed, rocket->velocity);
629  rocket->movetype = MOVETYPE_FLYMISSILE;
630  rocket->clipmask = MASK_SHOT;
631  rocket->solid = SOLID_BBOX;
632  rocket->s.effects |= EF_ROCKET;
633  VectorClear (rocket->mins);
634  VectorClear (rocket->maxs);
635  rocket->s.modelindex = gi.modelindex ("models/objects/rocket/tris.md2");
636  rocket->owner = self;
637  rocket->touch = rocket_touch;
638  rocket->nextthink = level.time + 8000/speed;
639  rocket->think = G_FreeEdict;
640  rocket->dmg = damage;
641  rocket->radius_dmg = radius_damage;
642  rocket->dmg_radius = damage_radius;
643  rocket->s.sound = gi.soundindex ("weapons/rockfly.wav");
644  rocket->classname = "rocket";
645 
646  if (self->client)
647  check_dodge (self, rocket->s.origin, dir, speed);
648 
649  gi.linkentity (rocket);
650 }
651 
652 
653 /*
654 =================
655 fire_rail
656 =================
657 */
658 void fire_rail (edict_t *self, vec3_t start, vec3_t aimdir, int damage, int kick)
659 {
660  vec3_t from;
661  vec3_t end;
662  trace_t tr;
663  edict_t *ignore;
664  int mask;
665  qboolean water;
666 
667  VectorMA (start, 8192, aimdir, end);
668  VectorCopy (start, from);
669  ignore = self;
670  water = false;
672  while (ignore) {
673  tr = gi.trace (from, NULL, NULL, end, ignore, mask);
674 
677  water = true;
678  }
679  else {
680  //ZOID--added so rail goes through SOLID_BBOX entities (gibs, etc)
681  if ((tr.ent->svflags & SVF_MONSTER) || (tr.ent->client) ||
682  (tr.ent->solid == SOLID_BBOX))
683  ignore = tr.ent;
684  else
685  ignore = NULL;
686 
687  if ((tr.ent != self) && (tr.ent->takedamage))
688  T_Damage (tr.ent, self, self, aimdir, tr.endpos, tr.plane.normal, damage, kick, 0, MOD_RAILGUN);
689  else
690  ignore = NULL;
691  }
692 
693  VectorCopy (tr.endpos, from);
694  }
695 
696  // send gun puff / flash
699  gi.WritePosition (start);
700  gi.WritePosition (tr.endpos);
701  gi.multicast (self->s.origin, MULTICAST_PHS);
702 // gi.multicast (start, MULTICAST_PHS);
703  if (water)
704  {
707  gi.WritePosition (start);
708  gi.WritePosition (tr.endpos);
710  }
711 
712  if (self->client)
713  PlayerNoise(self, tr.endpos, PNOISE_IMPACT);
714 }
715 
716 
717 /*
718 =================
719 fire_bfg
720 =================
721 */
722 void bfg_explode (edict_t *self)
723 {
724  edict_t *ent;
725  float points;
726  vec3_t v;
727  float dist;
728 
729  if (self->s.frame == 0)
730  {
731  // the BFG effect
732  ent = NULL;
733  while ((ent = findradius(ent, self->s.origin, self->dmg_radius)) != NULL)
734  {
735  if (!ent->takedamage)
736  continue;
737  if (ent == self->owner)
738  continue;
739  if (!CanDamage (ent, self))
740  continue;
741  if (!CanDamage (ent, self->owner))
742  continue;
743 
744  VectorAdd (ent->mins, ent->maxs, v);
745  VectorMA (ent->s.origin, 0.5, v, v);
746  VectorSubtract (self->s.origin, v, v);
747  dist = VectorLength(v);
748  points = self->radius_dmg * (1.0 - sqrt(dist/self->dmg_radius));
749  if (ent == self->owner)
750  points = points * 0.5;
751 
754  gi.WritePosition (ent->s.origin);
756  T_Damage (ent, self, self->owner, self->velocity, ent->s.origin, vec3_origin, (int)points, 0, DAMAGE_ENERGY, MOD_BFG_EFFECT);
757  }
758  }
759 
760  self->nextthink = level.time + FRAMETIME;
761  self->s.frame++;
762  if (self->s.frame == 5)
763  self->think = G_FreeEdict;
764 }
765 
766 void bfg_touch (edict_t *self, edict_t *other, cplane_t *plane, csurface_t *surf)
767 {
768  if (other == self->owner)
769  return;
770 
771  if (surf && (surf->flags & SURF_SKY))
772  {
773  G_FreeEdict (self);
774  return;
775  }
776 
777  if (self->owner->client)
778  PlayerNoise(self->owner, self->s.origin, PNOISE_IMPACT);
779 
780  // core explosion - prevents firing it into the wall/floor
781  if (other->takedamage)
782  T_Damage (other, self, self->owner, self->velocity, self->s.origin, plane->normal, 200, 0, 0, MOD_BFG_BLAST);
783  T_RadiusDamage(self, self->owner, 200, other, 100, MOD_BFG_BLAST);
784 
785  gi.sound (self, CHAN_VOICE, gi.soundindex ("weapons/bfg__x1b.wav"), 1, ATTN_NORM, 0);
786  self->solid = SOLID_NOT;
787  self->touch = NULL;
788  VectorMA (self->s.origin, -1 * FRAMETIME, self->velocity, self->s.origin);
789  VectorClear (self->velocity);
790  self->s.modelindex = gi.modelindex ("sprites/s_bfg3.sp2");
791  self->s.frame = 0;
792  self->s.sound = 0;
793  self->s.effects &= ~EF_ANIM_ALLFAST;
794  self->think = bfg_explode;
795  self->nextthink = level.time + FRAMETIME;
796  self->enemy = other;
797 
800  gi.WritePosition (self->s.origin);
801  gi.multicast (self->s.origin, MULTICAST_PVS);
802 }
803 
804 
805 void bfg_think (edict_t *self)
806 {
807  edict_t *ent;
808  edict_t *ignore;
809  vec3_t point;
810  vec3_t dir;
811  vec3_t start;
812  vec3_t end;
813  int dmg;
814  trace_t tr;
815 
816  if (deathmatch->value)
817  dmg = 5;
818  else
819  dmg = 10;
820 
821  ent = NULL;
822  while ((ent = findradius(ent, self->s.origin, 256)) != NULL)
823  {
824  if (ent == self)
825  continue;
826 
827  if (ent == self->owner)
828  continue;
829 
830  if (!ent->takedamage)
831  continue;
832 
833  if (!(ent->svflags & SVF_MONSTER) && (!ent->client) && (strcmp(ent->classname, "misc_explobox") != 0))
834  continue;
835 
836  VectorMA (ent->absmin, 0.5, ent->size, point);
837 
838  VectorSubtract (point, self->s.origin, dir);
839  VectorNormalize (dir);
840 
841  ignore = self;
842  VectorCopy (self->s.origin, start);
843  VectorMA (start, 2048, dir, end);
844  while(1)
845  {
847 
848  if (!tr.ent)
849  break;
850 
851  // hurt it if we can
852  if ((tr.ent->takedamage) && !(tr.ent->flags & FL_IMMUNE_LASER) && (tr.ent != self->owner))
853  T_Damage (tr.ent, self, self->owner, dir, tr.endpos, vec3_origin, dmg, 1, DAMAGE_ENERGY, MOD_BFG_LASER);
854 
855  // if we hit something that's not a monster or player we're done
856  if (!(tr.ent->svflags & SVF_MONSTER) && (!tr.ent->client))
857  {
860  gi.WriteByte (4);
861  gi.WritePosition (tr.endpos);
862  gi.WriteDir (tr.plane.normal);
863  gi.WriteByte (self->s.skinnum);
865  break;
866  }
867 
868  ignore = tr.ent;
869  VectorCopy (tr.endpos, start);
870  }
871 
874  gi.WritePosition (self->s.origin);
875  gi.WritePosition (tr.endpos);
876  gi.multicast (self->s.origin, MULTICAST_PHS);
877  }
878 
879  self->nextthink = level.time + FRAMETIME;
880 }
881 
882 
883 void fire_bfg (edict_t *self, vec3_t start, vec3_t dir, int damage, int speed, float damage_radius)
884 {
885  edict_t *bfg;
886 
887  bfg = G_Spawn();
888  VectorCopy (start, bfg->s.origin);
889  VectorCopy (dir, bfg->movedir);
890  vectoangles (dir, bfg->s.angles);
891  VectorScale (dir, speed, bfg->velocity);
893  bfg->clipmask = MASK_SHOT;
894  bfg->solid = SOLID_BBOX;
895  bfg->s.effects |= EF_BFG | EF_ANIM_ALLFAST;
896  VectorClear (bfg->mins);
897  VectorClear (bfg->maxs);
898  bfg->s.modelindex = gi.modelindex ("sprites/s_bfg1.sp2");
899  bfg->owner = self;
900  bfg->touch = bfg_touch;
901  bfg->nextthink = level.time + 8000/speed;
902  bfg->think = G_FreeEdict;
903  bfg->radius_dmg = damage;
904  bfg->dmg_radius = damage_radius;
905  bfg->classname = "bfg blast";
906  bfg->s.sound = gi.soundindex ("weapons/bfg__l1a.wav");
907 
908  bfg->think = bfg_think;
909  bfg->nextthink = level.time + FRAMETIME;
910  bfg->teammaster = bfg;
911  bfg->teamchain = NULL;
912 
913  if (self->client)
914  check_dodge (self, bfg->s.origin, dir, speed);
915 
916  gi.linkentity (bfg);
917 }
entity_state_s::old_origin
vec3_t old_origin
Definition: q_shared.h:1175
gi
game_import_t gi
Definition: g_main.c:25
trace_t::contents
int contents
Definition: q_shared.h:457
fire_bfg
void fire_bfg(edict_t *self, vec3_t start, vec3_t dir, int damage, int speed, float damage_radius)
Definition: g_weapon.c:883
csurface_s::name
char name[16]
Definition: q_shared.h:437
cplane_s::normal
vec3_t normal
Definition: q_shared.h:411
edict_s::s
entity_state_t s
Definition: g_local.h:964
edict_s::groundentity
edict_t * groundentity
Definition: g_local.h:1073
deathmatch
cvar_t * deathmatch
Definition: g_main.c:35
CONTENTS_MONSTER
#define CONTENTS_MONSTER
Definition: qfiles.h:359
game_import_t::trace
trace_t(* trace)(vec3_t start, vec3_t mins, vec3_t maxs, vec3_t end, edict_t *passent, int contentmask)
Definition: game.h:128
MOD_BFG_BLAST
#define MOD_BFG_BLAST
Definition: g_local.h:476
G_Spawn
edict_t * G_Spawn(void)
Definition: g_utils.c:420
trace_t::fraction
float fraction
Definition: q_shared.h:453
TE_GRENADE_EXPLOSION_WATER
@ TE_GRENADE_EXPLOSION_WATER
Definition: q_shared.h:953
TE_SHOTGUN
@ TE_SHOTGUN
Definition: q_shared.h:939
fire_shotgun
void fire_shotgun(edict_t *self, vec3_t start, vec3_t aimdir, int damage, int kick, int hspread, int vspread, int count, int mod)
Definition: g_weapon.c:290
VectorSubtract
#define VectorSubtract(a, b, c)
Definition: q_shared.h:156
game_import_t::WritePosition
void(* WritePosition)(vec3_t pos)
Definition: game.h:152
trace_t::plane
cplane_t plane
Definition: q_shared.h:455
CONTENTS_WATER
#define CONTENTS_WATER
Definition: qfiles.h:338
game_import_t::multicast
void(* multicast)(vec3_t origin, multicast_t to)
Definition: game.h:144
fire_grenade2
void fire_grenade2(edict_t *self, vec3_t start, vec3_t aimdir, int damage, int speed, float timer, float damage_radius, qboolean held)
Definition: g_weapon.c:519
SOLID_BBOX
@ SOLID_BBOX
Definition: game.h:37
MOD_GRENADE
#define MOD_GRENADE
Definition: g_local.h:469
FRAMETIME
#define FRAMETIME
Definition: g_local.h:73
TE_ROCKET_EXPLOSION_WATER
@ TE_ROCKET_EXPLOSION_WATER
Definition: q_shared.h:952
fire_hit
qboolean fire_hit(edict_t *self, vec3_t aim, int damage, int kick)
Definition: g_weapon.c:63
fire_rail
void fire_rail(edict_t *self, vec3_t start, vec3_t aimdir, int damage, int kick)
Definition: g_weapon.c:658
TE_SPLASH
@ TE_SPLASH
Definition: q_shared.h:945
v
GLdouble v
Definition: qgl_win.c:143
TE_BUBBLETRAIL
@ TE_BUBBLETRAIL
Definition: q_shared.h:946
entity_state_s::origin
vec3_t origin
Definition: q_shared.h:1173
TE_RAILTRAIL
@ TE_RAILTRAIL
Definition: q_shared.h:938
DAMAGE_ENERGY
#define DAMAGE_ENERGY
Definition: g_local.h:659
MASK_WATER
#define MASK_WATER
Definition: q_shared.h:395
CONTENTS_DEADMONSTER
#define CONTENTS_DEADMONSTER
Definition: qfiles.h:360
VectorScale
void VectorScale(vec3_t in, vec_t scale, vec3_t out)
Definition: q_shared.c:782
qboolean
qboolean
Definition: q_shared.h:56
DAMAGE_RADIUS
#define DAMAGE_RADIUS
Definition: g_local.h:657
trace_t
Definition: q_shared.h:449
VectorClear
#define VectorClear(a)
Definition: q_shared.h:159
i
int i
Definition: q_shared.c:305
check_dodge
static void check_dodge(edict_t *self, vec3_t start, vec3_t dir, int speed)
Definition: g_weapon.c:32
MOVETYPE_FLYMISSILE
@ MOVETYPE_FLYMISSILE
Definition: g_local.h:197
MOD_RAILGUN
#define MOD_RAILGUN
Definition: g_local.h:474
EF_BFG
#define EF_BFG
Definition: q_shared.h:564
edict_s::client
struct gclient_s * client
Definition: g_local.h:965
game_import_t::sound
void(* sound)(edict_t *ent, int channel, int soundindex, float volume, float attenuation, float timeofs)
Definition: game.h:109
MOD_HG_SPLASH
#define MOD_HG_SPLASH
Definition: g_local.h:479
edict_s::movedir
vec3_t movedir
Definition: g_local.h:1021
MASK_SHOT
#define MASK_SHOT
Definition: q_shared.h:397
bfg_touch
void bfg_touch(edict_t *self, edict_t *other, cplane_t *plane, csurface_t *surf)
Definition: g_weapon.c:766
CONTENTS_SOLID
#define CONTENTS_SOLID
Definition: qfiles.h:333
edict_s::mins
vec3_t mins
Definition: g_local.h:984
range
GLsizei range
Definition: qgl_win.c:121
SURF_WARP
#define SURF_WARP
Definition: qfiles.h:372
rocket_touch
void rocket_touch(edict_t *ent, edict_t *other, cplane_t *plane, csurface_t *surf)
Definition: g_weapon.c:569
ATTN_NORM
#define ATTN_NORM
Definition: q_shared.h:1019
ThrowDebris
void ThrowDebris(edict_t *self, char *modelname, float speed, vec3_t origin)
Definition: g_misc.c:286
T_Damage
void T_Damage(edict_t *targ, edict_t *inflictor, edict_t *attacker, vec3_t dir, vec3_t point, vec3_t normal, int damage, int knockback, int dflags, int mod)
Definition: g_combat.c:378
monsterinfo_t::dodge
void(* dodge)(edict_t *self, edict_t *other, float eta)
Definition: g_local.h:428
edict_s::movetype
int movetype
Definition: g_local.h:995
fire_rocket
void fire_rocket(edict_t *self, vec3_t start, vec3_t dir, int damage, int speed, float damage_radius, int radius_damage)
Definition: g_weapon.c:620
infront
qboolean infront(edict_t *self, edict_t *other)
Definition: g_ai.c:312
fire_blaster
void fire_blaster(edict_t *self, vec3_t start, vec3_t dir, int damage, int speed, int effect, qboolean hyper)
Definition: g_weapon.c:345
SURF_TRANS66
#define SURF_TRANS66
Definition: qfiles.h:374
entity_state_s::effects
unsigned int effects
Definition: q_shared.h:1180
DAMAGE_NO_KNOCKBACK
#define DAMAGE_NO_KNOCKBACK
Definition: g_local.h:660
TE_GRENADE_EXPLOSION
@ TE_GRENADE_EXPLOSION
Definition: q_shared.h:943
csurface_s::flags
int flags
Definition: q_shared.h:438
EF_GRENADE
#define EF_GRENADE
Definition: q_shared.h:562
entity_state_s::sound
int sound
Definition: q_shared.h:1185
bfg_explode
void bfg_explode(edict_t *self)
Definition: g_weapon.c:722
SVF_DEADMONSTER
#define SVF_DEADMONSTER
Definition: game.h:28
AngleVectors
void AngleVectors(vec3_t angles, vec3_t forward, vec3_t right, vec3_t up)
Definition: q_shared.c:93
Grenade_Touch
static void Grenade_Touch(edict_t *ent, edict_t *other, cplane_t *plane, csurface_t *surf)
Definition: g_weapon.c:455
MULTICAST_PHS
@ MULTICAST_PHS
Definition: q_shared.h:110
edict_s::svflags
int svflags
Definition: g_local.h:983
CHAN_VOICE
#define CHAN_VOICE
Definition: q_shared.h:1009
DAMAGE_BULLET
#define DAMAGE_BULLET
Definition: g_local.h:661
TE_BFG_BIGEXPLOSION
@ TE_BFG_BIGEXPLOSION
Definition: q_shared.h:956
vectoangles
void vectoangles(vec3_t vec, vec3_t angles)
Definition: g_utils.c:356
edict_s::classname
char * classname
Definition: g_local.h:1005
edict_s
Definition: g_local.h:962
MOD_BFG_LASER
#define MOD_BFG_LASER
Definition: g_local.h:475
findradius
edict_t * findradius(edict_t *from, vec3_t org, float rad)
Definition: g_utils.c:78
EF_ROCKET
#define EF_ROCKET
Definition: q_shared.h:561
G_FreeEdict
void G_FreeEdict(edict_t *e)
Definition: g_utils.c:452
r
GLdouble GLdouble r
Definition: qgl_win.c:336
TE_BFG_EXPLOSION
@ TE_BFG_EXPLOSION
Definition: q_shared.h:955
csurface_s
Definition: q_shared.h:435
MOD_HIT
#define MOD_HIT
Definition: g_local.h:495
MOD_HANDGRENADE
#define MOD_HANDGRENADE
Definition: g_local.h:478
game_import_t::soundindex
int(* soundindex)(char *name)
Definition: game.h:122
forward
static vec3_t forward
Definition: p_view.c:29
edict_s::clipmask
int clipmask
Definition: g_local.h:987
edict_s::spawnflags
int spawnflags
Definition: g_local.h:1006
VectorLength
vec_t VectorLength(vec3_t v)
Definition: q_shared.c:762
crandom
#define crandom()
Definition: g_local.h:510
game_import_t::pointcontents
int(* pointcontents)(vec3_t point)
Definition: game.h:129
edict_s::owner
edict_t * owner
Definition: g_local.h:988
game_import_t::modelindex
int(* modelindex)(char *name)
Definition: game.h:121
TE_ROCKET_EXPLOSION
@ TE_ROCKET_EXPLOSION
Definition: q_shared.h:942
SURF_FLOWING
#define SURF_FLOWING
Definition: qfiles.h:375
SURF_TRANS33
#define SURF_TRANS33
Definition: qfiles.h:373
edict_s::dmg_radius
float dmg_radius
Definition: g_local.h:1065
edict_s::nextthink
float nextthink
Definition: g_local.h:1036
cvar_s::value
float value
Definition: q_shared.h:324
edict_s::think
void(* think)(edict_t *self)
Definition: g_local.h:1038
random
#define random()
Definition: g_local.h:509
MOD_G_SPLASH
#define MOD_G_SPLASH
Definition: g_local.h:470
VectorNormalize
vec_t VectorNormalize(vec3_t v)
Definition: q_shared.c:681
EF_ANIM_ALLFAST
#define EF_ANIM_ALLFAST
Definition: q_shared.h:570
NULL
#define NULL
Definition: q_shared.h:60
edict_s::solid
solid_t solid
Definition: g_local.h:986
MOD_R_SPLASH
#define MOD_R_SPLASH
Definition: g_local.h:472
PNOISE_IMPACT
#define PNOISE_IMPACT
Definition: g_local.h:182
CanDamage
qboolean CanDamage(edict_t *targ, edict_t *inflictor)
Definition: g_combat.c:32
skill
cvar_t * skill
Definition: g_main.c:38
SOLID_NOT
@ SOLID_NOT
Definition: game.h:35
edict_s::dmg
int dmg
Definition: g_local.h:1063
edict_s::velocity
vec3_t velocity
Definition: g_local.h:1024
SPLASH_BROWN_WATER
#define SPLASH_BROWN_WATER
Definition: q_shared.h:998
VectorAdd
#define VectorAdd(a, b, c)
Definition: q_shared.h:157
CONTENTS_SLIME
#define CONTENTS_SLIME
Definition: qfiles.h:337
svc_temp_entity
@ svc_temp_entity
Definition: qcommon.h:210
edict_s::radius_dmg
int radius_dmg
Definition: g_local.h:1064
VectorCopy
#define VectorCopy(a, b)
Definition: q_shared.h:158
coop
cvar_t * coop
Definition: g_main.c:36
TE_LASER_SPARKS
@ TE_LASER_SPARKS
Definition: q_shared.h:950
edict_s::monsterinfo
monsterinfo_t monsterinfo
Definition: g_local.h:1108
trace_t::endpos
vec3_t endpos
Definition: q_shared.h:454
vec3_origin
vec3_t vec3_origin
Definition: q_shared.c:24
MOD_BLASTER
#define MOD_BLASTER
Definition: g_local.h:464
FL_IMMUNE_LASER
#define FL_IMMUNE_LASER
Definition: g_local.h:59
up
static vec3_t up
Definition: p_view.c:29
game_import_t::WriteDir
void(* WriteDir)(vec3_t pos)
Definition: game.h:153
CHAN_WEAPON
#define CHAN_WEAPON
Definition: q_shared.h:1008
game_import_t::WriteByte
void(* WriteByte)(int c)
Definition: game.h:147
MOD_HELD_GRENADE
#define MOD_HELD_GRENADE
Definition: g_local.h:487
edict_s::takedamage
int takedamage
Definition: g_local.h:1062
level
GLint level
Definition: qgl_win.c:116
TE_BFG_LASER
@ TE_BFG_LASER
Definition: q_shared.h:958
fire_grenade
void fire_grenade(edict_t *self, vec3_t start, vec3_t aimdir, int damage, int speed, float timer, float damage_radius)
Definition: g_weapon.c:486
edict_s::flags
int flags
Definition: g_local.h:996
sqrt
double sqrt(double x)
VectorMA
void VectorMA(vec3_t veca, float scale, vec3_t vecb, vec3_t vecc)
Definition: q_shared.c:719
MOD_BFG_EFFECT
#define MOD_BFG_EFFECT
Definition: g_local.h:477
SPLASH_SLIME
#define SPLASH_SLIME
Definition: q_shared.h:999
TE_BLASTER
@ TE_BLASTER
Definition: q_shared.h:937
CONTENTS_LAVA
#define CONTENTS_LAVA
Definition: qfiles.h:336
fire_bullet
void fire_bullet(edict_t *self, vec3_t start, vec3_t aimdir, int damage, int kick, int hspread, int vspread, int mod)
Definition: g_weapon.c:277
edict_s::maxs
vec3_t maxs
Definition: g_local.h:984
MOD_HYPERBLASTER
#define MOD_HYPERBLASTER
Definition: g_local.h:473
SPLASH_BLUE_WATER
#define SPLASH_BLUE_WATER
Definition: q_shared.h:997
edict_s::avelocity
vec3_t avelocity
Definition: g_local.h:1025
fire_lead
static void fire_lead(edict_t *self, vec3_t start, vec3_t aimdir, int damage, int kick, int te_impact, int hspread, int vspread, int mod)
Definition: g_weapon.c:134
PlayerNoise
void PlayerNoise(edict_t *who, vec3_t where, int type)
Definition: p_weapon.c:58
edict_s::teammaster
edict_t * teammaster
Definition: g_local.h:1076
entity_state_s::modelindex
int modelindex
Definition: q_shared.h:1176
TE_GUNSHOT
@ TE_GUNSHOT
Definition: q_shared.h:935
trace_t::ent
struct edict_s * ent
Definition: q_shared.h:458
SVF_MONSTER
#define SVF_MONSTER
Definition: game.h:29
edict_s::touch
void(* touch)(edict_t *self, edict_t *other, cplane_t *plane, csurface_t *surf)
Definition: g_local.h:1040
MOVETYPE_BOUNCE
@ MOVETYPE_BOUNCE
Definition: g_local.h:198
trace_t::surface
csurface_t * surface
Definition: q_shared.h:456
VectorSet
#define VectorSet(v, x, y, z)
Definition: q_shared.h:161
MOD_ROCKET
#define MOD_ROCKET
Definition: g_local.h:471
right
GLdouble right
Definition: qgl_win.c:159
SPLASH_UNKNOWN
#define SPLASH_UNKNOWN
Definition: q_shared.h:995
edict_s::waterlevel
int waterlevel
Definition: g_local.h:1094
Grenade_Explode
static void Grenade_Explode(edict_t *ent)
Definition: g_weapon.c:398
VectorCompare
int VectorCompare(vec3_t v1, vec3_t v2)
Definition: q_shared.c:672
bfg_think
void bfg_think(edict_t *self)
Definition: g_weapon.c:805
cplane_s
Definition: q_shared.h:409
points
GLdouble GLdouble GLint GLint const GLdouble * points
Definition: qgl_win.c:225
mask
GLint GLuint mask
Definition: qgl_win.c:317
edict_s::enemy
edict_t * enemy
Definition: g_local.h:1070
edict_s::teamchain
edict_t * teamchain
Definition: g_local.h:1075
game_import_t::linkentity
void(* linkentity)(edict_t *ent)
Definition: game.h:138
edict_s::size
vec3_t size
Definition: g_local.h:985
entity_state_s::angles
vec3_t angles
Definition: q_shared.h:1174
MULTICAST_PVS
@ MULTICAST_PVS
Definition: q_shared.h:111
vec3_t
vec_t vec3_t[3]
Definition: q_shared.h:127
SPLASH_LAVA
#define SPLASH_LAVA
Definition: q_shared.h:1000
edict_s::absmin
vec3_t absmin
Definition: g_local.h:985
blaster_touch
void blaster_touch(edict_t *self, edict_t *other, cplane_t *plane, csurface_t *surf)
Definition: g_weapon.c:306
count
GLint GLsizei count
Definition: qgl_win.c:128
SURF_SKY
#define SURF_SKY
Definition: qfiles.h:371
g_local.h
T_RadiusDamage
void T_RadiusDamage(edict_t *inflictor, edict_t *attacker, float damage, edict_t *ignore, float radius, int mod)
Definition: g_combat.c:548
edict_s::health
int health
Definition: g_local.h:1051