]> de.git.xonotic.org Git - xonotic/xonotic-data.pk3dir.git/blobdiff - qcsrc/client/hook.qc
particle effects on gauntlet too, make crylink secondary force negative again
[xonotic/xonotic-data.pk3dir.git] / qcsrc / client / hook.qc
index 9f690d5bc884dc3692b1ebb172cdd6089e74775f..513c7926334dd139a66dc322a7dcca2bf7fca2dd 100644 (file)
@@ -1,17 +1,8 @@
-.vector HookStart;
-.vector HookEnd;
-.float HookKillTime;
-.vector LGBeamStart;
-.vector LGBeamEnd;
-.float LGBeamKillTime;
-.float LGBeamSound;
-.float LGBeamSilent;
-.vector GauntletBeamStart;
-.vector GauntletBeamEnd;
-.float GauntletBeamKillTime;
-.float GauntletBeamSound;
-.float GauntletBeamSilent;
-
+.float HookType; // ENT_CLIENT_*
+.vector origin;
+.vector velocity;
+.float HookSound;
+.float HookSilent;
 
 void Draw_CylindricLine(vector from, vector to, float thickness, string texture, float aspect, float shift, vector rgb, float alpha, float drawflag)
 {
@@ -58,211 +49,198 @@ void Draw_GrapplingHook_trace_callback(vector start, vector hit, vector end)
 void Draw_GrapplingHook()
 {
        vector a, b;
-       string tex;
+       string tex, snd;
        vector rgb;
        float t;
        float s;
        vector vs;
 
-       if(time < self.HookKillTime)
-       {
-               s = cvar("cl_gunalign");
-               if(s != 1 && s != 2 && s != 4)
-                       s = 3; // default value
-               --s;
-               vs = hook_shotorigin[s];
-
-               if(self.sv_entnum == player_localentnum - 1)
-                       a = view_origin + view_forward * vs_x + view_right * -vs_y + view_up * vs_z;
-               else
-                       a = self.HookStart;
-               b = self.HookEnd;
+       InterpolateOrigin_Do();
 
-               t = GetPlayerColorForce(self.sv_entnum);
-
-               if(t == COLOR_TEAM1)
-               {
-                       tex = "particles/hook_red";
-                       rgb = '1 .3 .3';
-               }
-               else if(t == COLOR_TEAM2)
-               {
-                       tex = "particles/hook_blue";
-                       rgb = '.3 .3 1';
-               }
-               else if(t == COLOR_TEAM3)
-               {
-                       tex = "particles/hook_yellow";
-                       rgb = '1 1 .3';
-               }
-               else if(t == COLOR_TEAM4)
-               {
-                       tex = "particles/hook_pink";
-                       rgb = '1 .3 1';
-               }
-               else
-               {
-                       tex = "particles/hook_green";
-                       rgb = '.3 1 .3';
-               }
-
-               Draw_GrapplingHook_trace_callback_tex = tex;
-               Draw_GrapplingHook_trace_callback_rnd = random();
-               WarpZone_TraceBox_ThroughZone(a, '0 0 0', '0 0 0', b, MOVE_NOMONSTERS, world, world, Draw_GrapplingHook_trace_callback);
-               Draw_GrapplingHook_trace_callback_tex = string_null;
+       s = cvar("cl_gunalign");
+       if(s != 1 && s != 2 && s != 4)
+               s = 3; // default value
+       --s;
+       switch(self.HookType)
+       {
+               default:
+               case ENT_CLIENT_HOOK:
+                       vs = hook_shotorigin[s];
+                       break;
+               case ENT_CLIENT_LGBEAM:
+                       vs = electro_shotorigin[s];
+                       break;
+               case ENT_CLIENT_GAUNTLET:
+                       vs = gauntlet_shotorigin[s];
+                       break;
        }
 
-       if(time < self.LGBeamKillTime)
+       if((self.owner.sv_entnum == player_localentnum - 1))
        {
-               s = cvar("cl_gunalign");
-               if(s != 1 && s != 2 && s != 4)
-                       s = 3; // default value
-               --s;
-               vs = electro_shotorigin[s];
-
-               if(self.sv_entnum == player_localentnum - 1)
+               switch(self.HookType)
                {
-                       b = view_origin + view_forward * MAX_SHOT_DISTANCE;
-                       WarpZone_TraceLine(view_origin, b, MOVE_NORMAL, world);
-                       a = view_origin + view_forward * vs_x + view_right * -vs_y + view_up * vs_z;
+                       default:
+                       case ENT_CLIENT_HOOK:
+                               a = view_origin + view_forward * vs_x + view_right * -vs_y + view_up * vs_z;
+                               b = self.origin;
+                               break;
+                       case ENT_CLIENT_LGBEAM:
+                       case ENT_CLIENT_GAUNTLET:
+                               b = view_origin + view_forward * vlen(self.velocity - self.origin); // honor original length of beam!
+                               WarpZone_TraceLine(view_origin, b, MOVE_NORMAL, world);
+                               b = WarpZone_UnTransformOrigin(WarpZone_trace_transform, trace_endpos);
+                               a = view_origin + view_forward * vs_x + view_right * -vs_y + view_up * vs_z;
+                               break;
                }
-               else
+       }
+       else
+       {
+               switch(self.HookType)
                {
-                       a = self.LGBeamStart;
-                       b = self.LGBeamEnd;
+                       default:
+                       case ENT_CLIENT_HOOK:
+                               a = self.velocity;
+                               b = self.origin;
+                               break;
+                       case ENT_CLIENT_LGBEAM:
+                       case ENT_CLIENT_GAUNTLET:
+                               a = self.origin;
+                               b = self.velocity;
+                               break;
                }
+       }
 
-               tex = "particles/lgbeam";
-               rgb = '1 1 1';
+       t = GetPlayerColorForce(self.owner.sv_entnum);
+
+       switch(self.HookType)
+       {
+               default:
+               case ENT_CLIENT_HOOK:
+                       if(t == COLOR_TEAM1)
+                       {
+                               tex = "particles/hook_red";
+                               rgb = '1 .3 .3';
+                       }
+                       else if(t == COLOR_TEAM2)
+                       {
+                               tex = "particles/hook_blue";
+                               rgb = '.3 .3 1';
+                       }
+                       else if(t == COLOR_TEAM3)
+                       {
+                               tex = "particles/hook_yellow";
+                               rgb = '1 1 .3';
+                       }
+                       else if(t == COLOR_TEAM4)
+                       {
+                               tex = "particles/hook_pink";
+                               rgb = '1 .3 1';
+                       }
+                       else
+                       {
+                               tex = "particles/hook_green";
+                               rgb = '.3 1 .3';
+                       }
+                       break;
+               case ENT_CLIENT_LGBEAM:
+                       tex = "particles/lgbeam";
+                       rgb = '1 1 1';
+                       break;
+               case ENT_CLIENT_GAUNTLET:
+                       tex = "particles/gauntletbeam";
+                       rgb = '1 1 1';
+                       break;
+       }
 
-               Draw_GrapplingHook_trace_callback_tex = tex;
-               Draw_GrapplingHook_trace_callback_rnd = random();
-               WarpZone_TraceBox_ThroughZone(a, '0 0 0', '0 0 0', b, MOVE_NORMAL, world, world, Draw_GrapplingHook_trace_callback);
-               Draw_GrapplingHook_trace_callback_tex = string_null;
+       Draw_GrapplingHook_trace_callback_tex = tex;
+       Draw_GrapplingHook_trace_callback_rnd = random();
+       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);
+       Draw_GrapplingHook_trace_callback_tex = string_null;
 
-               // helps the sound
-               setorigin(self, a);
+       switch(self.HookType)
+       {
+               default:
+               case ENT_CLIENT_HOOK:
+                       setorigin(self, trace_endpos); // hook endpoint!
+                       self.angles = vectoangles(trace_endpos - WarpZone_TransformOrigin(WarpZone_trace_transform, a));
+                       break;
+               case ENT_CLIENT_LGBEAM:
+               case ENT_CLIENT_GAUNTLET:
+                       setorigin(self, a); // beam origin!
+                       break;
        }
 
-       if(time < self.GauntletBeamKillTime)
+       switch(self.HookType)
        {
-               s = cvar("cl_gunalign");
-               if(s != 1 && s != 2 && s != 4)
-                       s = 3; // default value
-               --s;
-               vs = gauntlet_shotorigin[s];
+               default:
+               case ENT_CLIENT_HOOK:
+                       break;
+               case ENT_CLIENT_LGBEAM:
+                       pointparticles(particleeffectnum("electro_lightning"), b, normalize(a - b), frametime);
+                       break;
+               case ENT_CLIENT_GAUNTLET:
+                       pointparticles(particleeffectnum("gauntlet_lightning"), b, normalize(a - b), frametime);
+                       break;
+       }
+}
 
-               if(self.sv_entnum == player_localentnum - 1)
-               {
-                       b = view_origin + view_forward * MAX_SHOT_DISTANCE;
-                       WarpZone_TraceLine(view_origin, b, MOVE_NORMAL, world);
-                       a = view_origin + view_forward * vs_x + view_right * -vs_y + view_up * vs_z;
-               }
-               else
-               {
-                       a = self.GauntletBeamStart;
-                       b = self.GauntletBeamEnd;
-               }
+void Remove_GrapplingHook()
+{
+       sound (self, CHAN_PROJECTILE, "misc/null.wav", VOL_BASE, ATTN_NORM);
+}
 
-               tex = "particles/gauntletbeam";
-               rgb = '1 1 1';
+void Ent_ReadHook(float bIsNew, float type)
+{
+       self.HookType = type;
 
-               Draw_GrapplingHook_trace_callback_tex = tex;
-               Draw_GrapplingHook_trace_callback_rnd = random();
-               WarpZone_TraceBox_ThroughZone(a, '0 0 0', '0 0 0', b, MOVE_NORMAL, world, world, Draw_GrapplingHook_trace_callback);
-               Draw_GrapplingHook_trace_callback_tex = string_null;
+       float sf;
+       sf = ReadByte();
 
-               // helps the sound
-               setorigin(self, a);
-       }
+       self.HookSilent = (sf & 0x80);
+       self.iflags = IFLAG_VELOCITY;
 
-       if(time < self.LGBeamKillTime && !self.LGBeamSilent)
-       {
-               if(!self.LGBeamSound)
-               {
-                       sound (self, CHAN_PROJECTILE, "weapons/lgbeam_fly.wav", VOL_BASE, ATTN_NORM);
-                       self.LGBeamSound = 1;
-               }
-       }
-       else
+       InterpolateOrigin_Undo();
+
+       if(sf & 1)
        {
-               if(self.LGBeamSound)
-               {
-                       sound (self, CHAN_PROJECTILE, "misc/null.wav", VOL_BASE, ATTN_NORM);
-                       self.LGBeamSound = 0;
-               }
+               self.owner = playerslots[ReadByte() - 1];
        }
-
-       if(time < self.GauntletBeamKillTime && !self.GauntletBeamSilent)
+       if(sf & 2)
        {
-               if(!self.GauntletBeamSound)
-               {
-                       sound (self, CHAN_PROJECTILE, "weapons/gauntletbeam_fly.wav", VOL_BASE, ATTN_NORM);
-                       self.GauntletBeamSound = 1;
-               }
+               self.origin_x = ReadCoord();
+               self.origin_y = ReadCoord();
+               self.origin_z = ReadCoord();
+               setorigin(self, self.origin);
        }
-       else
+       if(sf & 4)
        {
-               if(self.GauntletBeamSound)
-               {
-                       sound (self, CHAN_PROJECTILE, "misc/null.wav", VOL_BASE, ATTN_NORM);
-                       self.GauntletBeamSound = 0;
-               }
+               self.velocity_x = ReadCoord();
+               self.velocity_y = ReadCoord();
+               self.velocity_z = ReadCoord();
        }
-}
-
-void Net_GrapplingHook()
-{
-       float i, t;
-       vector start, end;
-       entity p;
-
-       i = ReadByte();
-       t = ReadByte();
-       end_x = ReadCoord();
-       end_y = ReadCoord();
-       end_z = ReadCoord();
-       start_x = ReadCoord();
-       start_y = ReadCoord();
-       start_z = ReadCoord();
 
-       if(i <= 0 || i >= 256) // not owned by a client
-               return;
-       --i;
+       InterpolateOrigin_Note();
 
-       p = playerslots[i];
-       if(!p)
-               return;
-
-       switch(t)
+       if(bIsNew)
        {
-               case 0: // hook beam
-                       p.HookKillTime = time + 0.1;
-                       p.HookStart = start;
-                       p.HookEnd = end;
-                       p.draw = Draw_GrapplingHook;
-                       break;
-               case 1: // electro lgbeam
-                       p.LGBeamKillTime = time + 0.1;
-                       p.LGBeamStart = start;
-                       p.LGBeamEnd = end;
-                       p.LGBeamSilent = 0;
-                       p.draw = Draw_GrapplingHook;
-                       break;
-               case 2: // silent electro lgbeam
-                       p.LGBeamKillTime = time + 0.1;
-                       p.LGBeamStart = start;
-                       p.LGBeamEnd = end;
-                       p.LGBeamSilent = 1;
-                       p.draw = Draw_GrapplingHook;
-                       break;
-               case 3: // gauntlet beam
-                       p.GauntletBeamKillTime = time + 0.1;
-                       p.GauntletBeamStart = start;
-                       p.GauntletBeamEnd = end;
-                       p.GauntletBeamSilent = 0;
-                       p.draw = Draw_GrapplingHook;
-                       break;
+               self.draw = Draw_GrapplingHook;
+               self.entremove = Remove_GrapplingHook;
+
+               switch(self.HookType)
+               {
+                       default:
+                       case ENT_CLIENT_HOOK:
+                               // for the model
+                               setmodel(self, "models/hook.md3");
+                               self.drawmask = MASK_NORMAL;
+                               break;
+                       case ENT_CLIENT_LGBEAM:
+                               sound (self, CHAN_PROJECTILE, "weapons/lgbeam_fly.wav", VOL_BASE, ATTN_NORM);
+                               break;
+                       case ENT_CLIENT_GAUNTLET:
+                               sound (self, CHAN_PROJECTILE, "weapons/gauntletbeam_fly.wav", VOL_BASE, ATTN_NORM);
+                               break;
+               }
        }
 }
 
@@ -270,4 +248,7 @@ void Hook_Precache()
 {
        precache_sound("weapons/lgbeam_fly.wav");
        precache_sound("weapons/gauntletbeam_fly.wav");
+       precache_model("models/hook.md3");
 }
+
+// TODO: hook: temporarily transform self.origin for drawing the model along warpzones!