]> de.git.xonotic.org Git - xonotic/xonotic-data.pk3dir.git/blob - qcsrc/client/tturrets.qc
Smoke effects for damage turrets
[xonotic/xonotic-data.pk3dir.git] / qcsrc / client / tturrets.qc
1 string tid2info_base;
2 string tid2info_head;
3 string tid2info_name;
4 vector  tid2info_min;
5 vector  tid2info_max;
6
7 void turret_tid2info(float _tid);
8 void turret_precache(float _tid);
9 float turret_is_precache[TID_LAST];
10
11 void turrets_precache()
12 {
13     turret_precache(TID_COMMON);
14 }
15
16 void turret_precache(float _tid)
17 {    
18     if (!turret_is_precache[TID_COMMON])
19     {
20         dprint("turret_precache TID_COMMON\n");
21         precache_sound ("weapons/rocket_impact.wav");
22         precache_model ("models/turrets/base-gib1.md3");
23         precache_model ("models/turrets/base-gib2.md3");
24         precache_model ("models/turrets/base-gib3.md3");
25         precache_model ("models/turrets/base-gib4.md3");
26         precache_model ("models/turrets/head-gib1.md3");
27         precache_model ("models/turrets/head-gib2.md3");
28         precache_model ("models/turrets/head-gib3.md3");
29         precache_model ("models/turrets/head-gib4.md3");
30         precache_model ("models/turrets/terrainbase.md3");
31         precache_model ("models/turrets/base.md3");
32         precache_model ("models/turrets/rocket.md3");
33     }
34     turret_tid2info(_tid);
35     if(turret_is_precache[_tid])
36     {
37         dprint("turret_precache: ", tid2info_name, " allready precachd, skipping.\n");
38         return;
39     }
40     else
41         dprint("turret_precache: ", tid2info_name, " precaching.\n");
42
43     switch(_tid)
44     {
45         case TID_EWHEEL:
46             precache_model ("models/turrets/ewheel-base2.md3");
47             precache_model ("models/turrets/ewheel-gun1.md3");            
48             break;
49         case TID_FLAC:
50             precache_model ("models/turrets/flac.md3");
51             break;
52         case TID_FUSION:
53             precache_model ("models/turrets/reactor.md3");    
54             break;
55         case TID_HELLION:
56             precache_model ("models/turrets/hellion.md3");
57             break;
58         case TID_HK:
59             precache_model ("models/turrets/hk.md3");
60             break;
61         case TID_MACHINEGUN:
62             precache_model ("models/turrets/machinegun.md3");
63             precache_sound ("weapons/uzi_fire.wav");
64             break;
65         case TID_MLRS:
66             precache_model ("models/turrets/mlrs.md3");
67             break;
68         case TID_PHASER:
69             precache_model ("models/turrets/phaser.md3");
70             precache_model ("models/turrets/phaser_beam.md3");
71             precache_sound ("turrets/phaser.wav");
72             break;
73         case TID_PLASMA:
74             precache_model ("models/turrets/plasma.md3");
75             break;
76         case TID_PLASMA_DUAL:
77             precache_model ("models/turrets/plasmad.md3");
78             break;
79         case TID_TESLA:
80             precache_model ("models/turrets/tesla_head.md3");
81             precache_model ("models/turrets/tesla_base.md3");
82             break;
83         case TID_WALKER:
84             precache_model ("models/turrets/walker_head_minigun.md3");
85             precache_model ("models/turrets/walker_body.md3");
86             precache_sound ("weapons/uzi_fire.wav");
87             break;
88     }    
89     turret_is_precache[_tid] = TRUE;
90 }
91
92 void turret_tid2info(float _tid)
93 {
94     tid2info_base = "models/turrets/base.md3";
95     tid2info_min = '-32 -32 0';
96     tid2info_max = '32 32 64';
97
98     switch(_tid)
99     {
100         case TID_EWHEEL:
101             tid2info_base = "models/turrets/ewheel-base2.md3";
102             tid2info_head = "models/turrets/ewheel-gun1.md3";
103             tid2info_name = "eWheel";
104             break;
105         case TID_FLAC:
106             tid2info_head = "models/turrets/flac.md3";
107             tid2info_name = "Flac Cannon";
108             break;
109         case TID_FUSION:
110             tid2info_head = "models/turrets/reactor.md3";
111             tid2info_name = "Fusion Reactor";
112             tid2info_min = '-34 -34 0';
113             tid2info_max = '34 34 90';
114             break;
115         case TID_HELLION:
116             tid2info_head = "models/turrets/hellion.md3";
117             tid2info_name = "Hellion";
118             break;
119         case TID_HK:
120             tid2info_head = "models/turrets/hk.md3";
121             tid2info_name = "Hunter-Killer";
122             break;
123         case TID_MACHINEGUN:
124             tid2info_head = "models/turrets/machinegun.md3";
125             tid2info_name = "Machinegun";
126             break;
127         case TID_MLRS:
128             tid2info_head = "models/turrets/mlrs.md3";
129             tid2info_name = "MLRS";
130             break;
131         case TID_PHASER:
132             tid2info_head = "models/turrets/phaser.md3";
133             tid2info_name = "Phaser";
134             break;
135         case TID_PLASMA:
136             tid2info_head = "models/turrets/plasma.md3";
137             tid2info_name = "Plasma";
138             break;
139         case TID_PLASMA_DUAL:
140             tid2info_head = "models/turrets/plasmad.md3";
141             tid2info_name = "Dual Plasma";
142             break;
143         case TID_TESLA:
144             tid2info_base = "models/turrets/tesla_base.md3";
145             tid2info_head = "models/turrets/tesla_head.md3";
146             tid2info_name = "Tesla coil";
147             tid2info_min = '-60 -60 0';
148             tid2info_max  ='60 60 128';
149             break;
150         case TID_WALKER:
151             tid2info_base = "models/turrets/walker_body.md3";
152             tid2info_head = "models/turrets/walker_head_minigun.md3";
153             tid2info_name = "Walker";
154             tid2info_min = '-70 -70 0';
155             tid2info_max = '70 70 95';
156             break;
157     }    
158 }
159
160 void turret_remove()
161 {
162     turret_tid2info(self.turret_type);
163     dprint("Removing ", tid2info_name, " turrret.\n");
164     
165     remove(self.tur_head);
166     self.tur_head = world;    
167 }
168
169 void turret_changeteam()
170 {
171         self.colormod = '0 0 0';
172         
173         switch(self.team)
174         {
175         case COLOR_TEAM1: // Red
176             self.colormod = '2 0.5 0.5';
177             break;
178
179         case COLOR_TEAM2: // Blue
180             self.colormod = '0.5 0.5 2';
181             break;
182
183         case COLOR_TEAM3: // Yellow
184             self.colormod = '1.4 1.4 0.6';
185             break;
186
187         case COLOR_TEAM4: // Pink
188             self.colormod = '1.4 0.6 1.4';
189             break;
190         }
191         
192         self.tur_head.colormod = self.colormod;    
193 }
194
195 void turret_head_draw()
196 {    
197     self.drawmask = MASK_NORMAL;
198 }
199
200 void turret_draw()
201 {        
202     float dt;
203
204     dt = time - self.move_time;
205     self.move_time = time;
206     if(dt <= 0)
207         return;
208     
209     self.tur_head.angles += dt * self.tur_head.move_avelocity;
210
211     if (self.health < 127)
212     {
213         dt = random();
214         
215         if(dt < 0.25)
216             te_spark(self.origin + '0 0 40', randomvec() * 256 + '0 0 256', 16);    
217         
218     }
219     
220     if(self.health < 85)    
221     if(dt < 0.1)
222         pointparticles(particleeffectnum("smoke_small"), (self.origin + (randomvec() * 80)), '0 0 0', 1);        
223     
224     if(self.health < 32)
225     if(dt < 0.1)
226         pointparticles(particleeffectnum("smoke_large"), (self.origin + (randomvec() * 80)), '0 0 0', 1);        
227     
228 }
229
230
231 void turret_walker_draw()
232 {        
233     float dt;
234             
235     dt = time - self.move_time;
236     self.move_time = time;
237     if(dt <= 0)
238         return;
239     
240     fixedmakevectors(self.angles);
241     movelib_groundalign4point(300, 100, 0.25);
242     
243     setorigin(self, self.origin + self.velocity * dt);
244     setorigin(self.tur_head, gettaginfo(self, gettagindex(self, "tag_head")));
245     
246     self.tur_head.angles += dt * self.tur_head.move_avelocity;
247     
248     self.angles_y = self.move_angles_y;
249     
250     if (self.health < 127)
251     if(random() < 0.25)
252         te_spark(self.origin + '0 0 40', randomvec() * 256 + '0 0 256', 16);    
253 }
254
255 void turret_ewheel_draw()
256 {        
257     float dt;
258             
259     dt = time - self.move_time;
260     self.move_time = time;
261     if(dt <= 0)
262         return;
263     
264     fixedmakevectors(self.angles);
265     movelib_groundalign4point(300, 100, 0.25);
266     
267     setorigin(self, self.origin + self.velocity * dt);
268     setorigin(self.tur_head, self.origin);
269     
270     self.tur_head.angles += dt * self.tur_head.move_avelocity;
271     
272     // Simulate banking
273     //self.angles_z -= self.angles_z * dt * 2;
274     //self.angles_z = bound(-45, self.angles_z  + ((self.move_angles_y - self.angles_y * -25) * dt), 45);
275     
276     self.angles_y = self.move_angles_y;
277     
278     if (self.health < 127)
279     if(random() < 0.25)
280         te_spark(self.origin + '0 0 40', randomvec() * 256 + '0 0 256', 16);    
281 }
282
283 void turret_construct()
284 {    
285     if(self.tur_head == world)
286         self.tur_head = spawn();
287     
288     turret_tid2info(self.turret_type);
289     dprint("Constructing ", tid2info_name , " turret (", ftos(self.turret_type), ")\n");
290
291     setorigin(self, self.origin);
292     setmodel(self, tid2info_base);
293     setmodel(self.tur_head, tid2info_head);
294     setsize(self, tid2info_min, tid2info_max);
295     setsize(self.tur_head, '0 0 0', '0 0 0');
296     setorigin(self.tur_head, gettaginfo(self, gettagindex(self, "tag_head")));
297
298     self.tur_head.classname     = "turret_head";
299     self.tur_head.owner         = self;
300     self.tur_head.move_movetype = MOVETYPE_NOCLIP;
301     self.move_movetype          = MOVETYPE_NOCLIP;
302     self.tur_head.angles        = self.angles;
303     self.health                 = 255;
304     self.solid                  = SOLID_BBOX;
305     self.tur_head.solid         = SOLID_NOT;
306     self.movetype               = MOVETYPE_NOCLIP;
307     self.tur_head.movetype      = MOVETYPE_NOCLIP;    
308     self.draw                   = turret_draw;
309     self.entremove              = turret_remove;
310     self.drawmask = MASK_NORMAL;
311     self.tur_head.drawmask = MASK_NORMAL;
312
313     if(self.turret_type == TID_EWHEEL || self.turret_type == TID_WALKER)
314     {
315         self.gravity            = 1;
316         self.movetype           = MOVETYPE_BOUNCE;
317         self.move_movetype      = MOVETYPE_BOUNCE;
318         self.move_origin        = self.origin;                
319         self.move_time          = time;
320         switch(self.turret_type)
321         {
322             case TID_EWHEEL:
323                 self.draw               = turret_ewheel_draw;
324                 break;
325             case TID_WALKER:
326                 self.draw               = turret_walker_draw;
327         }        
328     }
329 }
330
331 entity turret_gibtoss(string _model, vector _from, vector _to, vector _cmod, float _explode);
332 void turret_gibboom();
333 void turret_gib_draw()
334 {
335     Movetype_Physics_MatchTicrate(autocvar_cl_gibs_ticrate, autocvar_cl_gibs_sloppy);
336     
337     self.drawmask = MASK_NORMAL;
338         
339         if(self.cnt)
340         {
341             if(time >= self.nextthink)
342             {
343             turret_gibboom();
344             remove(self);
345             }
346         }
347         else
348         {
349         self.alpha = bound(0, self.nextthink - time, 1);
350         if(self.alpha < ALPHA_MIN_VISIBLE)
351             remove(self);           
352         }
353 }
354
355 void turret_gibboom()
356 {
357     float i;
358     
359     sound (self, CHAN_PROJECTILE, "weapons/rocket_impact.wav", VOL_BASE, ATTN_NORM);
360     pointparticles(particleeffectnum("rocket_explode"), self.origin, '0 0 0', 1);
361
362     for (i = 1; i < 5; i = i + 1)
363         turret_gibtoss(strcat("models/turrets/head-gib", ftos(i), ".md3"), self.origin, self.velocity + randomvec() * 700, '0 0 0', FALSE);
364 }
365
366 entity turret_gibtoss(string _model, vector _from, vector _to, vector _cmod, float _explode)
367 {
368     entity gib;
369
370     gib = spawn();
371     setorigin(gib, _from);
372     setmodel(gib, _model);
373     gib.colormod    = _cmod;
374         gib.solid       = SOLID_CORPSE;
375     gib.draw        = turret_gib_draw;    
376     gib.cnt         = _explode;
377     
378     if(_explode)
379     {
380         gib.nextthink = time + 0.2 * (autocvar_cl_gibs_lifetime * (1 + prandom() * 0.15));
381         gib.effects = EF_FLAME;
382     }        
383     else
384         gib.nextthink = time + autocvar_cl_gibs_lifetime * (1 + prandom() * 0.15);
385     
386     gib.gravity         = 1;
387         gib.move_movetype   = MOVETYPE_BOUNCE;
388         gib.move_origin     = gib.origin = _from;
389         gib.move_velocity   = _to;      
390         gib.move_avelocity  = prandomvec() * 32;
391         gib.move_time       = time;
392         gib.damageforcescale = 1;
393         
394         return gib;
395 }
396
397 void turret_die()
398 {    
399     entity headgib;
400     
401     setmodel(self, "");
402     setmodel(self.tur_head, "");
403     sound (self, CHAN_PROJECTILE, "weapons/rocket_impact.wav", VOL_BASE, ATTN_NORM);
404     pointparticles(particleeffectnum("rocket_explode"), self.origin, '0 0 0', 1);
405     turret_tid2info(self.turret_type);
406     dprint("Turret ", tid2info_name, " dies.\n");
407     
408     // Base
409     if(self.turret_type == TID_EWHEEL)
410         turret_gibtoss(tid2info_base, self.origin, self.velocity, '-1 -1 -1', FALSE);
411     else if (self.turret_type == TID_WALKER)
412         turret_gibtoss(tid2info_base, self.origin, self.velocity, '-1 -1 -1', FALSE);
413     else if (self.turret_type == TID_TESLA)
414         turret_gibtoss(tid2info_base, self.origin, self.velocity, '-1 -1 -1', FALSE);
415     else
416     {        
417         if (random() > 0.5)
418         {
419             turret_gibtoss("models/turrets/base-gib2.md3", self.origin + '0 0 2', '0 0 50' + randomvec() * 150, '0 0 0', FALSE);
420             turret_gibtoss("models/turrets/base-gib3.md3", self.origin + '0 0 2', '0 0 50' + randomvec() * 150, '0 0 0', FALSE);
421             turret_gibtoss("models/turrets/base-gib4.md3", self.origin + '0 0 2', '0 0 50' + randomvec() * 150, '0 0 0', FALSE);
422         }
423         else
424             turret_gibtoss("models/turrets/base-gib1.md3", self.origin + '0 0 4', '0 0 0', '0 0 0', TRUE);
425     }
426     
427     headgib = turret_gibtoss(tid2info_head, self.origin + '0 0 32', '0 0 200' + randomvec() * 200, '-1 -1 -1', TRUE);
428     headgib.angles = headgib.move_angles = self.tur_head.angles;
429     headgib.avelocity = headgib.move_avelocity = self.tur_head.move_avelocity + randomvec() * 45;
430     headgib.avelocity_y = headgib.move_avelocity_y = headgib.move_avelocity_y * 5;
431     headgib.gravity = 0.5;
432 }
433
434 void ent_turret()
435 {
436     float sf;
437     sf = ReadByte();
438
439         if(sf & TNSF_SETUP)
440         {           
441             self.turret_type = ReadByte();
442                     
443             self.origin_x = ReadCoord();
444             self.origin_y = ReadCoord();
445             self.origin_z = ReadCoord();
446             
447             self.angles_x = ReadAngle();
448             self.angles_y = ReadAngle();
449             
450             turret_precache(self.turret_type);
451             turret_construct();
452     }
453     
454     if(sf & TNSF_ANG)
455     {
456         self.tur_head.move_angles_x = ReadShort();
457         self.tur_head.move_angles_y = ReadShort();
458         self.tur_head.angles = self.angles + self.tur_head.move_angles;
459     }
460     
461     if(sf & TNSF_AVEL)
462     {
463         self.tur_head.move_avelocity_x = ReadShort();
464         self.tur_head.move_avelocity_y = ReadShort();            
465     }
466     
467     if(sf & TNSF_MOVE)
468     {
469         self.origin_x = ReadShort();
470         self.origin_y = ReadShort();
471         self.origin_z = ReadShort();
472         setorigin(self, self.origin);
473         
474         self.velocity_x = ReadShort();
475         self.velocity_y = ReadShort();
476         self.velocity_z = ReadShort();
477         
478         self.move_angles_y = ReadShort();
479             
480         self.move_time     = time;
481         self.move_velocity = self.velocity;
482         self.move_origin   = self.origin;
483     }
484         
485     if(sf & TNSF_ANIM)
486     {
487         self.frame1time = ReadCoord();
488         self.frame      = ReadByte();
489     }
490
491     if(sf & TNSF_STATUS)
492     {
493         float _team;
494         _team = ReadByte();
495         _team -= 1; // /&)=(%&#)&%)/#&)=½!!!½!!". thanks.
496         self.health = ReadByte();
497         if(_team != self.team)
498         {
499             self.team = _team;
500             turret_changeteam();        
501         }
502     } 
503     
504     if(self.health == 0)
505         turret_die();
506 }