2 REGISTER_WEAPON(HOOK, w_hook, IT_CELLS|IT_FUEL, 0, WEP_FLAG_CANCLIMB | WEP_TYPE_SPLASH, 0, "hookgun", "hook", _("Grappling Hook"))
13 .float hook_time_hooked;
14 .float hook_time_fueldecrease;
16 void W_Hook_ExplodeThink (void)
18 float dt, dmg_remaining_next, f;
20 dt = time - self.teleport_time;
21 dmg_remaining_next = pow(bound(0, 1 - dt / self.dmg_duration, 1), self.dmg_power);
23 f = self.dmg_last - dmg_remaining_next;
24 self.dmg_last = dmg_remaining_next;
26 RadiusDamage (self, self.realowner, self.dmg * f, self.dmg_edge * f, self.dmg_radius, self.realowner, self.dmg_force * f, self.projectiledeathtype, world);
27 self.projectiledeathtype |= HITTYPE_BOUNCE;
28 //RadiusDamage (self, world, self.dmg * f, self.dmg_edge * f, self.dmg_radius, world, self.dmg_force * f, self.projectiledeathtype, world);
30 if(dt < self.dmg_duration)
31 self.nextthink = time + 0.05; // soon
36 void W_Hook_Explode2 (void)
38 self.event_damage = SUB_Null;
39 self.touch = SUB_Null;
40 self.effects |= EF_NODRAW;
42 self.think = W_Hook_ExplodeThink;
43 self.nextthink = time;
44 self.dmg = autocvar_g_balance_hook_secondary_damage;
45 self.dmg_edge = autocvar_g_balance_hook_secondary_edgedamage;
46 self.dmg_radius = autocvar_g_balance_hook_secondary_radius;
47 self.dmg_force = autocvar_g_balance_hook_secondary_force;
48 self.dmg_power = autocvar_g_balance_hook_secondary_power;
49 self.dmg_duration = autocvar_g_balance_hook_secondary_duration;
50 self.teleport_time = time;
52 self.movetype = MOVETYPE_NONE;
55 void W_Hook_Damage (entity inflictor, entity attacker, float damage, float deathtype, vector hitloc, vector force)
59 self.health = self.health - damage;
61 print(strcat("hookbomb health ", ftos(self.health), " after ", ftos(damage), " damage... (at time: ", ftos(time), ")\n"));
64 W_PrepareExplosionByDamage(attacker, W_Hook_Explode2);
67 void W_Hook_Touch2 (void)
77 W_DecreaseAmmo(ammo_cells, autocvar_g_balance_hook_secondary_ammo, FALSE);
78 W_SetupShot (self, FALSE, 4, "weapons/hookbomb_fire.wav", CH_WEAPON_A, autocvar_g_balance_hook_secondary_damage);
81 gren.owner = gren.realowner = self;
82 gren.classname = "hookbomb";
83 gren.bot_dodge = TRUE;
84 gren.bot_dodgerating = autocvar_g_balance_hook_secondary_damage;
85 gren.movetype = MOVETYPE_TOSS;
86 PROJECTILE_MAKETRIGGER(gren);
87 gren.projectiledeathtype = WEP_HOOK | HITTYPE_SECONDARY;
88 setorigin(gren, w_shotorg);
89 setsize(gren, '0 0 0', '0 0 0');
91 gren.nextthink = time + autocvar_g_balance_hook_secondary_lifetime;
92 gren.think = adaptor_think2use_hittype_splash;
93 gren.use = W_Hook_Explode2;
94 gren.touch = W_Hook_Touch2;
96 gren.takedamage = DAMAGE_YES;
97 gren.health = autocvar_g_balance_hook_secondary_health;
98 gren.damageforcescale = autocvar_g_balance_hook_secondary_damageforcescale;
99 gren.event_damage = W_Hook_Damage;
100 gren.damagedbycontents = TRUE;
102 gren.velocity = '0 0 1' * autocvar_g_balance_hook_secondary_speed;
103 if(autocvar_g_projectiles_newton_style)
104 gren.velocity = gren.velocity + self.velocity;
106 gren.gravity = autocvar_g_balance_hook_secondary_gravity;
107 //W_SetupProjectileVelocity(gren); // just falling down!
109 gren.angles = '0 0 0';
110 gren.flags = FL_PROJECTILE;
112 CSQCProjectile(gren, TRUE, PROJECTILE_HOOKBOMB, TRUE);
114 other = gren; MUTATOR_CALLHOOK(EditProjectile);
117 void spawnfunc_weapon_hook (void)
119 if(g_grappling_hook) // offhand hook
121 startitem_failed = TRUE;
125 weapon_defaultspawnfunc(WEP_HOOK);
128 float w_hook(float req)
130 float hooked_time_max, hooked_fuel;
136 else if (req == WR_THINK)
138 if (self.BUTTON_ATCK || (!(self.items & IT_JETPACK) && self.BUTTON_HOOK))
141 if not(self.hook_state & HOOK_WAITING_FOR_RELEASE)
142 if not(self.hook_state & HOOK_FIRING)
143 if (time > self.hook_refire)
144 if (weapon_prepareattack(0, -1))
146 W_DecreaseAmmo(ammo_fuel, autocvar_g_balance_hook_primary_fuel, FALSE);
147 self.hook_state |= HOOK_FIRING;
148 weapon_thinkf(WFRAME_FIRE1, autocvar_g_balance_hook_primary_animtime, w_ready);
152 if (self.BUTTON_ATCK2)
154 if (weapon_prepareattack(1, autocvar_g_balance_hook_secondary_refire))
157 weapon_thinkf(WFRAME_FIRE2, autocvar_g_balance_hook_secondary_animtime, w_ready);
163 // if hooked, no bombs, and increase the timer
164 self.hook_refire = max(self.hook_refire, time + autocvar_g_balance_hook_primary_refire * W_WeaponRateFactor());
166 // hook also inhibits health regeneration, but only for 1 second
167 if not(self.items & IT_UNLIMITED_WEAPON_AMMO)
168 self.pauseregen_finished = max(self.pauseregen_finished, time + autocvar_g_balance_pause_fuel_regen);
171 if(self.hook && self.hook.state == 1)
173 hooked_time_max = autocvar_g_balance_hook_primary_hooked_time_max;
174 if (hooked_time_max > 0)
176 if ( time > self.hook_time_hooked + hooked_time_max )
177 self.hook_state |= HOOK_REMOVING;
180 hooked_fuel = autocvar_g_balance_hook_primary_hooked_fuel;
183 if ( time > self.hook_time_fueldecrease )
185 if not(self.items & IT_UNLIMITED_WEAPON_AMMO)
187 if ( self.ammo_fuel >= (time - self.hook_time_fueldecrease) * hooked_fuel )
189 W_DecreaseAmmo(ammo_fuel, (time - self.hook_time_fueldecrease) * hooked_fuel, FALSE);
190 self.hook_time_fueldecrease = time;
191 // decrease next frame again
196 self.hook_state |= HOOK_REMOVING;
197 W_SwitchWeapon_Force(self, w_getbestweapon(self));
205 self.hook_time_hooked = time;
206 self.hook_time_fueldecrease = time + autocvar_g_balance_hook_primary_hooked_time_free;
209 if (self.BUTTON_CROUCH)
211 self.hook_state &~= HOOK_PULLING;
212 if (self.BUTTON_ATCK || (!(self.items & IT_JETPACK) && self.BUTTON_HOOK))
213 self.hook_state &~= HOOK_RELEASING;
215 self.hook_state |= HOOK_RELEASING;
219 self.hook_state |= HOOK_PULLING;
220 self.hook_state &~= HOOK_RELEASING;
222 if (self.BUTTON_ATCK || (!(self.items & IT_JETPACK) && self.BUTTON_HOOK))
226 self.hook_state |= HOOK_WAITING_FOR_RELEASE;
230 self.hook_state |= HOOK_REMOVING;
231 self.hook_state &~= HOOK_WAITING_FOR_RELEASE;
235 else if (req == WR_PRECACHE)
237 precache_model ("models/weapons/g_hookgun.md3");
238 precache_model ("models/weapons/v_hookgun.md3");
239 precache_model ("models/weapons/h_hookgun.iqm");
240 precache_sound ("weapons/hook_impact.wav"); // done by g_hook.qc
241 precache_sound ("weapons/hook_fire.wav");
242 precache_sound ("weapons/hookbomb_fire.wav");
244 else if (req == WR_SETUP)
246 weapon_setup(WEP_HOOK);
247 self.current_ammo = ammo_fuel;
248 self.hook_state &~= HOOK_WAITING_FOR_RELEASE;
250 else if (req == WR_CHECKAMMO1)
253 return self.ammo_fuel > 0;
255 return self.ammo_fuel >= autocvar_g_balance_hook_primary_fuel;
257 else if (req == WR_CHECKAMMO2)
259 return self.ammo_cells >= autocvar_g_balance_hook_secondary_ammo;
261 else if (req == WR_RESETPLAYER)
263 self.hook_refire = time;
269 float w_hook(float req)
271 if(req == WR_IMPACTEFFECT)
274 org2 = w_org + w_backoff * 2;
275 pointparticles(particleeffectnum("hookbomb_explode"), org2, '0 0 0', 1);
277 sound(self, CH_SHOTS, "weapons/hookbomb_impact.wav", VOL_BASE, ATTN_NORM);
279 else if(req == WR_PRECACHE)
281 precache_sound("weapons/hookbomb_impact.wav");
283 else if (req == WR_SUICIDEMESSAGE)
284 w_deathtypestring = _("%s did the impossible");
285 else if (req == WR_KILLMESSAGE)
286 w_deathtypestring = _("%s has run into %s's gravity bomb");