]> icculus.org git repositories - divverent/nexuiz.git/blob - data/qcsrc/server/w_seeker.qc
git-svn-id: svn://svn.icculus.org/nexuiz/trunk@4832 f962a42d-fe04-0410-a3ab-8c8b0445ebaa
[divverent/nexuiz.git] / data / qcsrc / server / w_seeker.qc
1 //.float speed; = switchweapon\r
2 //.float proxytime; = autoswitch\r
3 //.float tl; = wait\r
4 \r
5 void Seeker_Missile_Explode ()\r
6 {\r
7     vector      org2;\r
8     float b;\r
9     org2 = findbetterlocation (self.origin, 12);\r
10     te_explosion (org2);\r
11 \r
12     b = crandom();\r
13     if (b<-0.7)\r
14         sound (self, CHAN_PROJECTILE, "weapons/hagexp1.ogg", 1, ATTN_NORM);\r
15     else if (b<0.4)\r
16         sound (self, CHAN_PROJECTILE, "weapons/hagexp2.ogg", 1, ATTN_NORM);\r
17     else if (b<1)\r
18         sound (self, CHAN_PROJECTILE, "weapons/hagexp3.ogg", 1, ATTN_NORM);\r
19 \r
20     self.event_damage = SUB_Null;\r
21     RadiusDamage (self, self.owner, cvar("g_balance_seeker_missile_damage"), cvar("g_balance_seeker_missile_edgedamage"), cvar("g_balance_seeker_missile_radius"), world, cvar("g_balance_seeker_missile_force"), self.projectiledeathtype, other);\r
22 \r
23     remove (self);\r
24 }\r
25 \r
26 void Seeker_Missile_Touch()\r
27 {\r
28     if (other == self.owner)\r
29         return;\r
30 \r
31     Seeker_Missile_Explode();\r
32 }\r
33 \r
34 void Seeker_Missile_Think()\r
35 {\r
36     entity e;\r
37     vector desireddir, olddir, newdir;\r
38     float turnrate;\r
39     float dist;\r
40 \r
41     if (time > self.cnt)\r
42         Seeker_Missile_Explode();\r
43 \r
44     if (!self.switchweapon)\r
45         self.switchweapon = cvar("g_balance_seeker_missile_speed");\r
46 \r
47     if ((self.switchweapon < cvar("g_balance_seeker_missile_speed_max")) && cvar("g_balance_seeker_missile_speed_accel"))\r
48         self.switchweapon = self.switchweapon * cvar("g_balance_seeker_missile_accel");\r
49 \r
50     if (self.switchweapon > cvar("g_balance_seeker_missile_speed_max"))\r
51         self.switchweapon = self.switchweapon * cvar("g_balance_seeker_missile_decel");\r
52 \r
53     if (self.enemy != world)\r
54     {\r
55         e               = self.enemy;\r
56         turnrate        = cvar("g_balance_seeker_missile_turnrate");                // how fast to turn\r
57         desireddir      = normalize(e.origin - self.origin);\r
58         olddir          = normalize(self.velocity);                                         // get my current direction\r
59         dist            = vlen(e.origin - self.origin);\r
60 \r
61         // Do evasive maneuvers for world objects? ( this should be a cpu hog. :P )\r
62         if (cvar("g_balance_seeker_missile_smart") && (dist > cvar("g_balance_seeker_missile_smart_mindist")))\r
63         {\r
64             // Is it a better idea (shorter distance) to trace to the target itself?\r
65             if ( vlen(self.origin + olddir * self.wait) < dist)\r
66                 traceline(self.origin, self.origin + olddir * self.wait, FALSE, self);\r
67             else\r
68                 traceline(self.origin, self.enemy.origin, FALSE, self);\r
69 \r
70             // Setup adaptive tracelength\r
71             self.wait = vlen(self.origin - trace_endpos);\r
72             if (self.wait < cvar("g_balance_seeker_missile_smart_trace_min")) self.wait = cvar("g_balance_seeker_missile_smart_trace_min");\r
73             if (self.wait > cvar("g_balance_seeker_missile_smart_trace_max")) self.wait = cvar("g_balance_seeker_missile_smart_trace_max");\r
74 \r
75             // Calc how important it is that we turn and add this to the desierd (enemy) dir.\r
76             desireddir  = normalize(((trace_plane_normal * (1 - trace_fraction)) + (desireddir * trace_fraction)) * 0.5);\r
77         }\r
78 \r
79         //newdir = normalize((olddir + desireddir * turnrate) * 0.5);// take the average of the 2 directions; not the best method but simple & easy\r
80         newdir = normalize(olddir + desireddir * turnrate);// take the average of the 2 directions; not the best method but simple & easy\r
81 \r
82         self.velocity = newdir * self.switchweapon;                                         // make me fly in the new direction at my flight speed\r
83     }\r
84 \r
85     // Proxy\r
86     if (cvar("g_balance_seeker_missile_proxy"))\r
87     {\r
88         if ( dist <= cvar("g_balance_seeker_missile_proxy_maxrange"))\r
89         {\r
90             if (self.autoswitch == 0)\r
91             {\r
92                 self.autoswitch = time + cvar("g_balance_seeker_missile_proxy_delay");\r
93             }\r
94             else\r
95             {\r
96                 if (self.autoswitch <= time)\r
97                 {\r
98                     Seeker_Missile_Explode();\r
99                     self.autoswitch = 0;\r
100                 }\r
101             }\r
102         }\r
103         else\r
104         {\r
105             if (self.autoswitch != 0)\r
106                 self.autoswitch = 0;\r
107         }\r
108     }\r
109     ///////////////\r
110 \r
111     if (self.enemy.deadflag != DEAD_NO)\r
112     {\r
113         self.enemy = world;\r
114         self.cnt = time + 1 + (random() * 4);\r
115         self.nextthink = self.cnt;\r
116         return;\r
117     }\r
118 \r
119     self.angles = vectoangles(self.velocity);                   // turn model in the new flight direction\r
120     self.nextthink = time + 0.05;\r
121 \r
122 }\r
123 \r
124 \r
125 \r
126 void Seeker_Missile_Damage (entity inflictor, entity attacker, float damage, float deathtype, vector hitloc, vector force)\r
127 {\r
128 \r
129     float d;\r
130     d = damage;\r
131 \r
132     if (self.owner == attacker)\r
133         d = d * 0.25;\r
134 \r
135     self.health = self.health - d;\r
136 \r
137     if (self.health <= 0)\r
138         Seeker_Missile_Explode();\r
139 }\r
140 \r
141 #define EF_ROCKET                               16777216\r
142 void Seeker_Missile_Animate()\r
143 {\r
144     self.frame = self.frame +1;\r
145     self.nextthink = time + 0.05;\r
146 \r
147     if(self.frame == 5)\r
148     {   self.think           = Seeker_Missile_Think;\r
149         self.nextthink       = time + cvar("g_balance_seeker_missile_activate_delay");\r
150         self.effects         = EF_LOWPRECISION | EF_NOSHADOW | EF_ROCKET;\r
151 \r
152         if (cvar("g_balance_seeker_guided_proxy"))\r
153             self.movetype    = MOVETYPE_BOUNCEMISSILE;\r
154         else\r
155             self.movetype    = MOVETYPE_FLYMISSILE;\r
156 \r
157     }\r
158 }\r
159 \r
160 void Seeker_Fire_Missile(vector f_org)\r
161 {\r
162     local entity missile;\r
163 \r
164     if not(self.items & IT_UNLIMITED_AMMO)\r
165         self.ammo_rockets = self.ammo_rockets - cvar("g_balance_seeker_missile_ammo");\r
166 \r
167     makevectors(self.v_angle);\r
168     W_SetupShot (self, f_org, FALSE, 2, "weapons/hagar_fire.ogg");\r
169     pointparticles(particleeffectnum("hagar_muzzleflash"), w_shotorg, w_shotdir * 1000, 1);\r
170 \r
171     //self.detornator         = FALSE;\r
172 \r
173     missile                 = spawn();\r
174     missile.owner           = self;\r
175     missile.classname       = "seeker_missile";\r
176     missile.bot_dodge       = TRUE;\r
177     missile.bot_dodgerating = cvar("g_balance_seeker_missile_damage");\r
178 \r
179     missile.think           = Seeker_Missile_Animate;\r
180 \r
181     //if (!cvar("g_balance_seeker_missile_proxy"))\r
182     missile.touch           = Seeker_Missile_Touch;\r
183 \r
184     missile.event_damage    = Seeker_Missile_Damage;\r
185     missile.nextthink       = time + 0.2;// + cvar("g_balance_seeker_missile_activate_delay");\r
186     missile.cnt             = time + cvar("g_balance_seeker_missile_lifetime");\r
187     missile.enemy           = self.enemy;\r
188     missile.switchweapon           = cvar("g_balance_seeker_missile_speed");\r
189     missile.effects         = EF_LOWPRECISION | EF_NOSHADOW;// | EF_ROCKET;\r
190     missile.solid           = SOLID_BBOX;\r
191     missile.scale           = 2;\r
192     missile.takedamage          = DAMAGE_YES;\r
193     missile.damageforcescale    = 4;\r
194     missile.health              = 5;\r
195 \r
196     setorigin (missile, w_shotorg);\r
197     setmodel  (missile, "models/tagrocket.md3");\r
198     setsize (missile, '0 0 0', '0 0 0');\r
199 \r
200 \r
201     missile.movetype    = MOVETYPE_FLYMISSILE;// MOVETYPE_TOSS;\r
202 \r
203     missile.flags       = FL_PROJECTILE;\r
204 \r
205     missile.velocity    = (w_shotdir + '0 0 0.45') * missile.switchweapon;\r
206     W_SetupProjectileVelocity(missile);\r
207 \r
208     missile.switchweapon = vlen(missile.velocity);\r
209     missile.angles = vectoangles (missile.velocity);\r
210 \r
211 }\r
212 \r
213 void Seeker_Vollycontroler_Think()\r
214 {\r
215     entity oldself,oldenemy;\r
216     self.cnt = self.cnt - 1;\r
217     if ((self.owner.ammo_rockets < cvar("g_balance_seeker_missile_ammo")) || (self.cnt <= -1))\r
218     {\r
219         remove(self);\r
220         return;\r
221 \r
222     }\r
223 \r
224     self.nextthink = time + cvar("g_balance_seeker_missile_delay");\r
225 \r
226     oldself = self;\r
227     self = self.owner;\r
228 \r
229     oldenemy = self.enemy;\r
230     self.enemy = oldself.enemy;\r
231 \r
232     switch(oldself.cnt)\r
233     {\r
234         case 0:\r
235             Seeker_Fire_Missile('37.5 17 -22');\r
236             break;\r
237         case 1:\r
238             Seeker_Fire_Missile('37.5 9.5 -22');\r
239             break;\r
240         case 2:\r
241             Seeker_Fire_Missile('40 17 -29');\r
242             break;\r
243         case 3:\r
244             Seeker_Fire_Missile('40 09.5 -29');\r
245             break;\r
246     }\r
247 \r
248 \r
249 \r
250     self.enemy = oldenemy;\r
251     self = oldself;\r
252 }\r
253 \r
254 void Seeker_Tag_Damage (entity inflictor, entity attacker, float damage, float deathtype, vector hitloc, vector force)\r
255 {\r
256     self.health = self.health - damage;\r
257     if (self.health <= 0)\r
258         self.think ();\r
259 }\r
260 \r
261 void Seeker_Tag_Explode ()\r
262 {\r
263     vector      org2;\r
264     float b;\r
265 \r
266     //if(other==self.owner)\r
267     //    return;\r
268 \r
269     org2 = findbetterlocation (self.origin, 12);\r
270     te_explosion (org2);\r
271 \r
272     b = crandom();\r
273     if (b<-0.7)\r
274         sound (self, CHAN_PROJECTILE, "weapons/hagexp1.ogg", 1, ATTN_NORM);\r
275     else if (b<0.4)\r
276         sound (self, CHAN_PROJECTILE, "weapons/hagexp2.ogg", 1, ATTN_NORM);\r
277     else if (b<1)\r
278         sound (self, CHAN_PROJECTILE, "weapons/hagexp3.ogg", 1, ATTN_NORM);\r
279 \r
280     remove (self);\r
281 }\r
282 \r
283 void Seeker_Tag_Think()\r
284 {\r
285     remove(self);\r
286     return;\r
287 }\r
288 \r
289 void Seeker_Tag_Touch()\r
290 {\r
291     vector dir;\r
292     vector org2;\r
293 \r
294         if (trace_dphitq3surfaceflags & Q3SURFACEFLAG_NOIMPACT)\r
295         {\r
296                 remove(self);\r
297                 return;\r
298         }\r
299 \r
300     if (other == self.owner)\r
301         return;\r
302 \r
303     dir     = normalize (self.owner.origin - self.origin);\r
304     org2    = findbetterlocation (self.origin, 8);\r
305 \r
306     te_knightspike(org2);\r
307 \r
308     self.event_damage = SUB_Null;\r
309     sound (self, CHAN_PROJECTILE, "weapons/laserimpact.ogg", 1, ATTN_NORM);\r
310 \r
311     if (other.classname == "player")\r
312     {\r
313 \r
314         entity e;\r
315         e           = spawn();\r
316         e.cnt       = cvar("g_balance_seeker_missile_count");\r
317         e.owner     = self.owner;\r
318         e.enemy     = other;\r
319         e.think     = Seeker_Vollycontroler_Think;\r
320         e.nextthink = time;\r
321 \r
322         //sprint(self.owner, "^1Target lock ^3[^7 ",other.netname, " ^3]^1 acquired - autofire activated.\n");\r
323         //sprint(other,"^1You are targeted!\n");\r
324 \r
325         // stuffcmd(other,"play2 weapons/zany-alarm4.ogg\n");\r
326         // stuffcmd(self.owner, "play2 weapons/zany-lock4.ogg\n");\r
327     }\r
328 \r
329     remove(self);\r
330     return;\r
331 }\r
332 \r
333 \r
334 \r
335 void Seeker_Fire_Tag()\r
336 {\r
337     local entity missile;\r
338     if (cvar("g_use_ammunition"))\r
339         self.ammo_rockets = self.ammo_rockets - cvar("g_balance_seeker_tag_ammo");\r
340 \r
341     W_SetupShot (self, '56 13 -15', FALSE, 2, "weapons/hagar_fire.ogg");\r
342 \r
343     missile                 = spawn();\r
344     missile.owner           = self;\r
345     missile.classname       = "seeker_tag";\r
346     missile.bot_dodge       = TRUE;\r
347     missile.bot_dodgerating = 50;\r
348     missile.touch           = Seeker_Tag_Touch;\r
349     missile.think           = Seeker_Tag_Think;\r
350     missile.nextthink       = time + 15;\r
351     missile.movetype        = MOVETYPE_FLY;\r
352     missile.solid           = SOLID_BBOX;\r
353     missile.owner           = self;\r
354 \r
355     missile.takedamage       = DAMAGE_YES;\r
356     missile.event_damage    = Seeker_Tag_Explode;\r
357     missile.health          = 5;\r
358 \r
359     setmodel (missile, "models/laser.mdl");\r
360     setorigin (missile, w_shotorg);\r
361 \r
362     missile.effects     = EF_BRIGHTFIELD | EF_FULLBRIGHT | EF_NOSHADOW | EF_LOWPRECISION;\r
363     missile.flags       = FL_PROJECTILE;\r
364 \r
365     missile.velocity    = w_shotdir  * cvar("g_balance_seeker_tag_speed");\r
366     missile.movetype    = MOVETYPE_BOUNCEMISSILE;\r
367     W_SetupProjectileVelocity(missile);\r
368     missile.angles = vectoangles (missile.velocity);\r
369 }\r
370 \r
371 \r
372 void Seeker_Flac_Explode ()\r
373 {\r
374     vector      org2;\r
375     float b;\r
376     org2 = findbetterlocation (self.origin, 12);\r
377     te_explosion (org2);\r
378 \r
379     b = crandom();\r
380     if (b<-0.7)\r
381         sound (self, CHAN_PROJECTILE, "weapons/hagexp1.ogg", 1, ATTN_NORM);\r
382     else if (b<0.4)\r
383         sound (self, CHAN_PROJECTILE, "weapons/hagexp2.ogg", 1, ATTN_NORM);\r
384     else if (b<1)\r
385         sound (self, CHAN_PROJECTILE, "weapons/hagexp3.ogg", 1, ATTN_NORM);\r
386 \r
387     self.event_damage = SUB_Null;\r
388 \r
389     RadiusDamage (self, self.owner, cvar("g_balance_seeker_flac_damage"), cvar("g_balance_seeker_flac_edgedamage"), cvar("g_balance_seeker_flac_radius"), world, cvar("g_balance_seeker_flac_force"), self.projectiledeathtype, other);\r
390 \r
391     remove (self);\r
392 }\r
393 \r
394 void Seeker_Flac_Touch()\r
395 {\r
396     if (other == self.owner)\r
397         return;\r
398 \r
399     Seeker_Flac_Explode();\r
400 }\r
401 \r
402 void Seeker_Fire_Flac()\r
403 {\r
404         local entity missile;\r
405     vector f_org;\r
406 \r
407         if not(self.items & IT_UNLIMITED_AMMO)\r
408                 self.ammo_rockets = self.ammo_rockets - cvar("g_balance_seeker_flac_ammo");\r
409 \r
410     switch(floor(random() * 3))\r
411     {\r
412         case 0:\r
413             f_org = '37.5 17 -22';\r
414             break;\r
415         case 1:\r
416             f_org = '37.5 9.5 -22';\r
417             break;\r
418         case 2:\r
419             f_org = '40 17 -29';\r
420             break;\r
421         case 3:\r
422             f_org = '40 09.5 -29';\r
423             break;\r
424     }\r
425     W_SetupShot (self, f_org, FALSE, 2, "weapons/hagar_fire.wav");\r
426 \r
427         pointparticles(particleeffectnum("hagar_muzzleflash"), w_shotorg, w_shotdir * 1000, 1);\r
428 \r
429         missile = spawn ();\r
430         missile.owner = missile.realowner = self;\r
431         missile.classname = "missile";\r
432         missile.bot_dodge = TRUE;\r
433         missile.bot_dodgerating = cvar("g_balance_seeker_flac_damage");\r
434         missile.touch = Seeker_Flac_Explode;\r
435         missile.use = Seeker_Flac_Explode;\r
436         missile.think = Seeker_Flac_Explode;\r
437         missile.nextthink = time + cvar("g_balance_seeker_flac_lifetime") + cvar("g_balance_seeker_flac_lifetime_rand");\r
438         missile.solid = SOLID_BBOX;\r
439         missile.scale = 0.4; // BUG: the model is too big\r
440         missile.projectiledeathtype = WEP_SEEKER;\r
441         setorigin (missile, w_shotorg);\r
442         setmodel (missile, "models/hagarmissile.mdl"); // precision set below\r
443         setsize (missile, '0 0 0', '0 0 0');\r
444 \r
445         missile.effects = EF_LOWPRECISION;\r
446         missile.modelflags = MF_GRENADE;\r
447 \r
448         missile.movetype = MOVETYPE_FLY;\r
449         w_shotdir = w_shotdir + '0 0 0.3';\r
450         missile.velocity    = (w_shotdir  + randomvec() * cvar("g_balance_seeker_flac_spread")) * cvar("g_balance_seeker_flac_speed");\r
451 \r
452         W_SetupProjectileVelocity(missile);\r
453 \r
454         missile.angles = vectoangles (missile.velocity);\r
455         missile.flags = FL_PROJECTILE;\r
456 }\r
457 \r
458 float w_seeker(float req)\r
459 {\r
460     if (req == WR_AIM)\r
461         self.BUTTON_ATCK = bot_aim(cvar("g_balance_seeker_tag_speed"), 0, 20, FALSE);\r
462 \r
463     else if (req == WR_THINK)\r
464     {\r
465         if (self.BUTTON_ATCK)\r
466             if (weapon_prepareattack(0, cvar("g_balance_seeker_tag_refire")))\r
467             {\r
468                 Seeker_Fire_Tag();\r
469                 weapon_thinkf(WFRAME_FIRE1, cvar("g_balance_seeker_tag_animtime"), w_ready);\r
470             }\r
471 \r
472         if (self.BUTTON_ATCK2)\r
473             if (weapon_prepareattack(1, cvar("g_balance_seeker_flac_refire")))\r
474             {\r
475                 Seeker_Fire_Flac();\r
476                 weapon_thinkf(WFRAME_FIRE1, cvar("g_balance_seeker_flac_animtime"), w_ready);\r
477             }\r
478 \r
479     }\r
480     else if (req == WR_PRECACHE)\r
481     {\r
482         precache_model ("models/tagrocket.md3");\r
483         precache_model ("models/weapons/g_seeker.md3");\r
484         precache_model ("models/weapons/v_seeker.md3");\r
485         precache_model ("models/weapons/w_seeker.zym");\r
486         precache_sound ("weapons/hagar_fire.wav");\r
487         precache_sound ("weapons/hagexp1.wav");\r
488         precache_sound ("weapons/hagexp2.wav");\r
489         precache_sound ("weapons/hagexp3.wav");\r
490     }\r
491     else if (req == WR_SETUP)\r
492         weapon_setup(WEP_SEEKER);\r
493     else if (req == WR_CHECKAMMO1)\r
494         return self.ammo_rockets >= cvar("g_balance_seeker_tag_ammo");\r
495     else if (req == WR_CHECKAMMO2)\r
496         return self.ammo_rockets >= cvar("g_balance_seeker_secondary_ammo");\r
497     else if (req == WR_SUICIDEMESSAGE)\r
498         w_deathtypestring = "played with tiny rockets";\r
499     else if (req == WR_KILLMESSAGE)\r
500             w_deathtypestring = "was taged by";\r
501     return TRUE;\r
502 };\r
503 \r