4 void turret_gib_damage (entity inflictor, entity attacker, float damage, float deathtype, vector hitloc, vector vforce)
\r
6 self.velocity += vforce;
\r
10 vector v_from, vector v_to, vector v_colormod,
\r
12 float f_lifetime, float f_fadetime, float b_burn)
\r
19 gib.classname = "turret_gib";
\r
20 setmodel(gib,smodel);
\r
21 setorigin(gib,v_from);
\r
22 SUB_SetFade(gib,time + f_lifetime,2);
\r
24 gib.solid = SOLID_BBOX;
\r
26 gib.movetype = MOVETYPE_BOUNCE;
\r
27 gib.takedamage = DAMAGE_YES;
\r
28 gib.event_damage = turret_gib_damage;
\r
30 gib.effects = EF_LOWPRECISION;
\r
31 gib.flags = FL_NOTARGET;
\r
32 gib.colormod = v_colormod;
\r
33 gib.velocity = v_to;
\r
38 burn.effects = EF_LOWPRECISION|EF_FLAME;
\r
39 setattachment(burn,gib,"");
\r
40 setorigin(burn,(gib.mins + gib.maxs) * 0.5);
\r
41 SUB_SetFade(burn,time + (f_lifetime * 0.5) ,2);
\r
45 void turret_gib_boom()
\r
51 for (i = 1; i < 5; i = i +1)
\r
54 gib.classname = "turret_gib";
\r
56 s = strcat("models/turrets/head-gib",ftos(i));
\r
57 s = strcat(s,".md3");
\r
58 // bprint("s:",s,"\n");
\r
61 setorigin(gib,self.origin);
\r
63 SUB_SetFade(gib,time + 5,2);
\r
65 gib.solid = SOLID_BBOX;
\r
67 gib.movetype = MOVETYPE_BOUNCE;
\r
69 gib.damageforcescale = 2;
\r
70 gib.takedamage = DAMAGE_YES;
\r
71 gib.event_damage = turret_gib_damage;
\r
73 gib.effects = EF_LOWPRECISION;
\r
74 gib.flags = FL_NOTARGET;
\r
75 gib.velocity = self.velocity + (randomvec() * 700);
\r
76 gib.avelocity = randomvec() * 64;
\r
79 WriteByte (MSG_BROADCAST, SVC_TEMPENTITY);
\r
80 WriteByte (MSG_BROADCAST, 78);
\r
81 WriteCoord (MSG_BROADCAST, self.origin_x);
\r
82 WriteCoord (MSG_BROADCAST, self.origin_y);
\r
83 WriteCoord (MSG_BROADCAST, self.origin_z);
\r
88 void turret_trowgib2(
\r
89 vector v_from, vector v_to, vector v_colormod,
\r
90 entity e_mimic, float boomtime)
\r
96 gib.classname = "turret_gib";
\r
97 setmodel(gib,e_mimic.model);
\r
98 setorigin(gib,v_from);
\r
100 gib.solid = SOLID_BBOX;
\r
102 gib.movetype = MOVETYPE_BOUNCE;
\r
103 gib.gravity = 0.75;
\r
104 gib.damageforcescale = 2;
\r
105 gib.takedamage = DAMAGE_YES;
\r
106 gib.event_damage = turret_gib_damage;
\r
108 gib.effects = EF_LOWPRECISION;
\r
109 gib.flags = FL_NOTARGET;
\r
110 gib.colormod = v_colormod;
\r
111 gib.velocity = v_to;
\r
112 gib.avelocity = randomvec() * 32;
\r
113 gib.think = turret_gib_boom;
\r
114 gib.nextthink = boomtime;
\r
115 gib.effects = EF_FLAME;
\r
120 * Spawn a boom, trow fake bits arround
\r
121 * and hide the real ones.
\r
123 void turret_stdproc_die()
\r
128 self.deadflag = DEAD_DEAD;
\r
129 self.tur_head.deadflag = self.deadflag;
\r
131 sound (self, CHAN_PLAYER, "weapons/rocket_impact.wav", VOL_BASE, ATTN_NORM);
\r
132 org2 = self.origin + '0 0 40';
\r
134 // Explotion grafix
\r
135 WriteByte (MSG_BROADCAST, SVC_TEMPENTITY);
\r
136 WriteByte (MSG_BROADCAST, 78);
\r
137 WriteCoord (MSG_BROADCAST, org2_x);
\r
138 WriteCoord (MSG_BROADCAST, org2_y);
\r
139 WriteCoord (MSG_BROADCAST, org2_z);
\r
141 // Unsolidify and hide real parts
\r
142 self.solid = SOLID_NOT;
\r
143 self.tur_head.solid = self.solid;
\r
146 self.tur_head.alpha = self.alpha;
\r
148 self.event_damage = SUB_Null;
\r
149 self.tur_head.event_damage = SUB_Null;
\r
150 self.takedamage = DAMAGE_NO;
\r
151 self.tur_head.takedamage = self.takedamage;
\r
154 self.tur_head.effects = self.effects;
\r
157 // Trow fake parts arround
\r
159 if not(self.damage_flags & TFL_DMG_DEATH_NOGIBS)
\r
161 makevectors(self.angles);
\r
162 if (random() > 0.5)
\r
164 turret_trowgib(self.origin, '0 0 0','1 1 1',"models/turrets/base-gib2.md3",min(self.respawntime,20),1,1);
\r
165 t_dir = (v_up * 700) + (randomvec() * 300);
\r
166 turret_trowgib(self.origin, t_dir,'1 1 1',"models/turrets/base-gib3.md3",min(self.respawntime,10),1,1);
\r
167 t_dir = (v_up * 700) + (randomvec() * 300);
\r
168 turret_trowgib(self.origin, t_dir,'1 1 1',"models/turrets/base-gib4.md3",min(self.respawntime,10),1,1);
\r
172 turret_trowgib(self.origin, '0 0 0','1 1 1',"models/turrets/base-gib1.md3",min(self.respawntime,20),1,1);
\r
175 // Blow the top part up into the air
\r
176 turret_trowgib2( self.origin + (v_up * 50),
\r
177 v_up * 150 + randomvec() * 50,
\r
179 self.tur_head,time + 0.5 + (random() * 0.5));
\r
183 RadiusDamage (self,self, min(self.ammo,50),min(self.ammo,50) * 0.25,250,world,min(self.ammo,50)*5,DEATH_TURRET,world);
\r
185 if(self.damage_flags & TFL_DMG_DEATH_NORESPAWN)
\r
187 if (self.turret_diehook)
\r
188 self.turret_diehook();
\r
190 remove(self.tur_head);
\r
196 self.nextthink = time + self.respawntime;
\r
197 //self.think = self.turret_spawnfunc;
\r
198 self.think = turret_stdproc_respawn;
\r
199 if (self.turret_diehook)
\r
200 self.turret_diehook();
\r
205 void turret_stdproc_respawn()
\r
207 // Make sure all parts belong to the same team since
\r
208 // this function doubles as "teamchange" function.
\r
210 self.tur_head.team = self.team;
\r
213 COLOR_TEAM1 = 4; // red
\r
214 COLOR_TEAM2 = 13; // blue
\r
215 COLOR_TEAM3 = 12; // yellow
\r
216 COLOR_TEAM4 = 9; // pink
\r
219 self.colormod = '0 0 0';
\r
223 case COLOR_TEAM1: // Red
\r
224 self.colormod = '1.4 0.8 0.8';
\r
227 case COLOR_TEAM2: // Blue
\r
228 self.colormod = '0.8 0.8 1.4';
\r
231 case COLOR_TEAM3: // Yellow
\r
232 self.colormod = '1.4 1.4 0.6';
\r
235 case COLOR_TEAM4: // Pink
\r
236 self.colormod = '1.4 0.6 1.4';
\r
240 self.deadflag = DEAD_NO;
\r
241 self.tur_head.deadflag = self.deadflag;
\r
243 self.tur_head.effects = self.effects;
\r
245 self.solid = SOLID_BBOX;
\r
246 self.tur_head.solid = self.solid;
\r
249 self.tur_head.alpha = self.alpha;
\r
251 self.takedamage = DAMAGE_AIM;
\r
252 self.tur_head.takedamage = self.takedamage;
\r
254 self.event_damage = turret_stdproc_damage;
\r
255 self.tur_head.event_damage = self.event_damage;
\r
257 self.avelocity = '0 0 0';
\r
258 self.tur_head.avelocity = self.avelocity;
\r
259 self.tur_head.angles = self.idle_aim;
\r
260 self.health = self.tur_health;
\r
261 self.tur_head.health = self.tur_health;
\r
263 self.enemy = world;
\r
264 self.volly_counter = self.shot_volly;
\r
265 self.ammo = self.ammo_max;
\r
267 self.nextthink = time + self.ticrate;
\r
268 self.think = turret_think;
\r
270 if (self.turret_respawnhook)
\r
271 self.turret_respawnhook();
\r
276 * Standard damage proc.
\r
278 void turret_stdproc_damage (entity inflictor, entity attacker, float damage, float deathtype, vector hitloc, vector vforce)
\r
283 baseent = self.owner;
\r
287 // Enougth allready!
\r
288 if (self.health <= 0)
\r
291 // Inactive turrets take no damage. (hm..)
\r
292 if not (baseent.tur_active)
\r
296 if (self.team == attacker.team)
\r
298 // This does not happen anymore. Re-enable if you fix that.
\r
299 //if(attacker.flags & FL_CLIENT)
\r
300 if(clienttype(attacker) == CLIENTTYPE_REAL)
\r
301 sprint(attacker, "\{1}Turret tells you: I'm on your team!\n");
\r
303 if(cvar("g_friendlyfire"))
\r
304 damage = damage * cvar("g_friendlyfire");
\r
309 self.health = self.health - damage;
\r
311 // thorw head slightly off aim when hit?
\r
312 if (self.classname == "turret_head")
\r
313 if (baseent.damage_flags & TFL_DMG_HEADSHAKE)
\r
315 //baseent.tur_aimoff_x += (random() * damage);
\r
316 //baseent.tur_aimoff_y += ((random()*0.75) * damage);
\r
317 self.angles_x = self.angles_x + (-0.5 + random()) * damage;
\r
318 self.angles_y = self.angles_y + (-0.5 + random()) * damage;
\r
321 if (baseent.turrcaps_flags & TFL_TURRCAPS_MOVE)
\r
322 baseent.velocity = baseent.velocity + vforce;
\r
324 // FIXME: Better damage feedback
\r
325 // Start burning when we have 10% or less health left
\r
326 //if (self.health < (self.tur_health * 0.1))
\r
327 // self.effects = EF_FLAME;
\r
329 if (self.health <= 0)
\r
331 baseent.event_damage = SUB_Null;
\r
332 baseent.tur_head.event_damage = SUB_Null;
\r
333 baseent.takedamage = DAMAGE_NO;
\r
334 baseent.tur_head.takedamage = baseent.takedamage;
\r
335 baseent.nextthink = time;
\r
336 baseent.think = turret_stdproc_die;
\r