1 .float HookType; // ENT_CLIENT_*
8 void Draw_CylindricLine(vector from, vector to, float thickness, string texture, float aspect, float shift, vector rgb, float alpha, float drawflag, vector vieworg)
10 // I want to draw a quad...
11 // from and to are MIDPOINTS.
13 vector axis, thickdir, A, B, C, D;
16 axis = normalize(to - from);
17 length_tex = aspect * vlen(to - from) / thickness;
19 // direction is perpendicular to the view normal, and perpendicular to the axis
20 thickdir = normalize(cross(axis, vieworg - from));
23 print("from ", vtos(from), "\n");
24 print("to ", vtos(to), "\n");
25 print("org ", vtos(view_origin), "\n");
26 print("dir ", vtos(thickdir), "\n");
29 A = from - thickdir * (thickness / 2);
30 B = from + thickdir * (thickness / 2);
31 C = to + thickdir * (thickness / 2);
32 D = to - thickdir * (thickness / 2);
34 R_BeginPolygon(texture, drawflag);
35 R_PolygonVertex(A, '0 0 0' + shift * '1 0 0', rgb, alpha);
36 R_PolygonVertex(B, '0 1 0' + shift * '1 0 0', rgb, alpha);
37 R_PolygonVertex(C, '0 1 0' + (shift + length_tex) * '1 0 0', rgb, alpha);
38 R_PolygonVertex(D, '0 0 0' + (shift + length_tex) * '1 0 0', rgb, alpha);
42 string Draw_GrapplingHook_trace_callback_tex;
43 float Draw_GrapplingHook_trace_callback_rnd;
44 vector Draw_GrapplingHook_trace_callback_rgb;
45 float Draw_GrapplingHook_trace_callback_a;
46 void Draw_GrapplingHook_trace_callback(vector start, vector hit, vector end)
50 vorg = WarpZone_TransformOrigin(WarpZone_trace_transform, view_origin);
51 for(i = 0; i < Draw_GrapplingHook_trace_callback_a; ++i)
52 Draw_CylindricLine(hit, start, 8, Draw_GrapplingHook_trace_callback_tex, 0.25, Draw_GrapplingHook_trace_callback_rnd, Draw_GrapplingHook_trace_callback_rgb, min(1, Draw_GrapplingHook_trace_callback_a - i), DRAWFLAG_NORMAL, vorg);
53 Draw_GrapplingHook_trace_callback_rnd += 0.25 * vlen(hit - start) / 8;
56 void Draw_GrapplingHook()
64 float intensity, offset;
66 InterpolateOrigin_Do();
68 s = cvar("cl_gunalign");
69 if(s != 1 && s != 2 && s != 4)
70 s = 3; // default value
76 vs = hook_shotorigin[s];
78 case ENT_CLIENT_LGBEAM:
79 vs = electro_shotorigin[s];
81 case ENT_CLIENT_GAUNTLET:
82 vs = gauntlet_shotorigin[s];
86 if((self.owner.sv_entnum == player_localentnum - 1))
92 a = view_origin + view_forward * vs_x + view_right * -vs_y + view_up * vs_z;
95 case ENT_CLIENT_LGBEAM:
96 case ENT_CLIENT_GAUNTLET:
98 b = view_origin + view_forward * self.HookRange;
100 b = view_origin + view_forward * vlen(self.velocity - self.origin); // honor original length of beam!
101 WarpZone_TraceLine(view_origin, b, MOVE_NORMAL, world);
102 b = WarpZone_UnTransformOrigin(WarpZone_trace_transform, trace_endpos);
103 a = view_origin + view_forward * vs_x + view_right * -vs_y + view_up * vs_z;
109 switch(self.HookType)
112 case ENT_CLIENT_HOOK:
116 case ENT_CLIENT_LGBEAM:
117 case ENT_CLIENT_GAUNTLET:
124 t = GetPlayerColorForce(self.owner.sv_entnum);
126 switch(self.HookType)
129 case ENT_CLIENT_HOOK:
134 tex = "particles/hook_red";
137 else if(t == COLOR_TEAM2)
139 tex = "particles/hook_blue";
142 else if(t == COLOR_TEAM3)
144 tex = "particles/hook_yellow";
147 else if(t == COLOR_TEAM4)
149 tex = "particles/hook_pink";
154 tex = "particles/hook_green";
158 case ENT_CLIENT_LGBEAM:
159 intensity = bound(0.2, 1 + Noise_Pink(self, frametime) * 1 + Noise_Burst(self, frametime, 0.03) * 0.3, 2);
160 offset = Noise_Brown(self, frametime) * 10;
161 tex = "particles/lgbeam";
164 case ENT_CLIENT_GAUNTLET:
166 offset = Noise_White(self, frametime);
167 tex = "particles/gauntletbeam";
172 Draw_GrapplingHook_trace_callback_tex = tex;
173 Draw_GrapplingHook_trace_callback_rnd = offset;
174 Draw_GrapplingHook_trace_callback_rgb = rgb;
175 Draw_GrapplingHook_trace_callback_a = intensity;
176 WarpZone_TraceBox_ThroughZone(a, '0 0 0', '0 0 0', b, ((self.HookType == ENT_CLIENT_HOOK) ? MOVE_NOTHING : MOVE_NORMAL), world, world, Draw_GrapplingHook_trace_callback);
177 Draw_GrapplingHook_trace_callback_tex = string_null;
179 atrans = WarpZone_TransformOrigin(WarpZone_trace_transform, a);
181 switch(self.HookType)
184 case ENT_CLIENT_HOOK:
185 setorigin(self, trace_endpos); // hook endpoint!
186 self.angles = vectoangles(trace_endpos - atrans);
188 case ENT_CLIENT_LGBEAM:
189 case ENT_CLIENT_GAUNTLET:
190 setorigin(self, a); // beam origin!
194 switch(self.HookType)
197 case ENT_CLIENT_HOOK:
199 case ENT_CLIENT_LGBEAM:
200 pointparticles(particleeffectnum("electro_lightning"), trace_endpos, normalize(atrans - trace_endpos), frametime * intensity);
202 case ENT_CLIENT_GAUNTLET:
203 pointparticles(particleeffectnum("gauntlet_lightning"), b, normalize(a - b), frametime * intensity);
208 void Remove_GrapplingHook()
210 sound (self, CHAN_PROJECTILE, "misc/null.wav", VOL_BASE, ATTN_NORM);
213 void Ent_ReadHook(float bIsNew, float type)
215 self.HookType = type;
220 self.HookSilent = (sf & 0x80);
221 self.iflags = IFLAG_VELOCITY;
223 InterpolateOrigin_Undo();
227 self.owner = playerslots[ReadByte() - 1];
228 switch(self.HookType)
231 case ENT_CLIENT_HOOK:
232 case ENT_CLIENT_GAUNTLET:
235 case ENT_CLIENT_LGBEAM:
236 self.HookRange = ReadCoord();
242 self.origin_x = ReadCoord();
243 self.origin_y = ReadCoord();
244 self.origin_z = ReadCoord();
245 setorigin(self, self.origin);
249 self.velocity_x = ReadCoord();
250 self.velocity_y = ReadCoord();
251 self.velocity_z = ReadCoord();
254 InterpolateOrigin_Note();
258 self.draw = Draw_GrapplingHook;
259 self.entremove = Remove_GrapplingHook;
261 switch(self.HookType)
264 case ENT_CLIENT_HOOK:
266 setmodel(self, "models/hook.md3");
267 self.drawmask = MASK_NORMAL;
269 case ENT_CLIENT_LGBEAM:
270 sound (self, CHAN_PROJECTILE, "weapons/lgbeam_fly.wav", VOL_BASE, ATTN_NORM);
272 case ENT_CLIENT_GAUNTLET:
273 sound (self, CHAN_PROJECTILE, "weapons/gauntletbeam_fly.wav", VOL_BASE, ATTN_NORM);
281 precache_sound("weapons/lgbeam_fly.wav");
282 precache_sound("weapons/gauntletbeam_fly.wav");
283 precache_model("models/hook.md3");
286 // TODO: hook: temporarily transform self.origin for drawing the model along warpzones!