From: Mircea Kitsune Date: Sat, 25 Sep 2010 21:20:48 +0000 (+0300) Subject: First step toward a mine layer weapon; Make a copy of the Rocket Launcher gun, and... X-Git-Tag: xonotic-v0.1.0preview~307^2~33^2~51 X-Git-Url: https://git.rm.cloudns.org/?a=commitdiff_plain;h=4ab47ae52e69afa22af48d2d583a918c00c1b643;p=xonotic%2Fxonotic-data.pk3dir.git First step toward a mine layer weapon; Make a copy of the Rocket Launcher gun, and a test map to test the gun on (to be removed when this branch is merged). Next step is a proper mine projectile. --- diff --git a/balanceXonotic.cfg b/balanceXonotic.cfg index 076a4935b6..61a5db0b4e 100644 --- a/balanceXonotic.cfg +++ b/balanceXonotic.cfg @@ -303,6 +303,36 @@ set g_balance_grenadelauncher_secondary_damageforcescale 0 set g_balance_grenadelauncher_secondary_bouncefactor 0.7 set g_balance_grenadelauncher_secondary_bouncestop 0.12 // }}} +// {{{ rocketlauncher // TODO +set g_balance_minelayer_damage 100 +set g_balance_minelayer_edgedamage 33 +set g_balance_minelayer_force 350 +set g_balance_minelayer_radius 125 +set g_balance_minelayer_speed 1000 +set g_balance_minelayer_speedaccel 0 +set g_balance_minelayer_speedstart 1000 +set g_balance_minelayer_lifetime 5 +set g_balance_minelayer_refire 1 +set g_balance_minelayer_animtime 0.4 +set g_balance_minelayer_ammo 3 +set g_balance_minelayer_health 0 +set g_balance_minelayer_damageforcescale 0 +set g_balance_minelayer_detonatedelay -1 // positive: timer till detonation is allowed, negative: "security device" that prevents ANY remote detonation if it could hurt its owner, zero: detonatable at any time +set g_balance_minelayer_guiderate 45 // max degrees per second +set g_balance_minelayer_guideratedelay 0.01 // immediate +set g_balance_minelayer_guidegoal 512 // goal distance for (non-laser) guiding (higher = less control, lower = erratic) +set g_balance_minelayer_guidedelay 0.15 // delay before guiding kicks in +set g_balance_minelayer_guidestop 0 // stop guiding when firing again +set g_balance_minelayer_laserguided_speed 1000 //650 +set g_balance_minelayer_laserguided_speedaccel 0 +set g_balance_minelayer_laserguided_speedstart 1000 +set g_balance_minelayer_laserguided_turnrate 0.75 //0.5 +set g_balance_minelayer_laserguided_allow_steal 1 +set g_balance_minelayer_remote_damage 50 +set g_balance_minelayer_remote_edgedamage 16.5 +set g_balance_minelayer_remote_radius 120 +set g_balance_minelayer_remote_force 350 +// }}} // {{{ electro // TODO set g_balance_electro_lightning 1 set g_balance_electro_primary_damage 90 diff --git a/maps/minelayer_test.bsp b/maps/minelayer_test.bsp new file mode 100644 index 0000000000..8d778ddae1 Binary files /dev/null and b/maps/minelayer_test.bsp differ diff --git a/maps/minelayer_test.map b/maps/minelayer_test.map new file mode 100644 index 0000000000..ee20966c04 --- /dev/null +++ b/maps/minelayer_test.map @@ -0,0 +1,79 @@ + +// entity 0 +{ +"classname" "worldspawn" +// brush 0 +{ +( 1024 1024 1152 ) ( 1024 -1024 1152 ) ( -1024 1024 1152 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( 1024 1024 1152 ) ( -1024 1024 1152 ) ( 1024 1024 1024 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( 1024 1024 1152 ) ( 1024 1024 1024 ) ( 1024 -1024 1152 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( -1024 -1024 1024 ) ( 1024 -1024 1024 ) ( -1024 1024 1024 ) skies/exosystem 0 0 0 0.5 0.5 0 0 0 +( -1024 -1024 1024 ) ( -1024 -1024 1152 ) ( 1024 -1024 1024 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( -1024 -1024 1024 ) ( -1024 1024 1024 ) ( -1024 -1024 1152 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +} +// brush 1 +{ +( -1024 1024 1024 ) ( -1024 -1024 1024 ) ( -1152 1024 1024 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( -1024 1024 1024 ) ( -1152 1024 1024 ) ( -1024 1024 0 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( -1024 1024 1024 ) ( -1024 1024 0 ) ( -1024 -1024 1024 ) eX/eXmetalrib01 0 0 0 1 1 0 0 0 +( -1152 -1024 0 ) ( -1024 -1024 0 ) ( -1152 1024 0 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( -1152 -1024 0 ) ( -1152 -1024 1024 ) ( -1024 -1024 0 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( -1152 -1024 0 ) ( -1152 1024 0 ) ( -1152 -1024 1024 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +} +// brush 2 +{ +( 1152 1024 1024 ) ( 1152 -1024 1024 ) ( 1024 1024 1024 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( 1152 1024 1024 ) ( 1024 1024 1024 ) ( 1152 1024 0 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( 1152 1024 1024 ) ( 1152 1024 0 ) ( 1152 -1024 1024 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( 1024 -1024 0 ) ( 1152 -1024 0 ) ( 1024 1024 0 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( 1024 -1024 0 ) ( 1024 -1024 1024 ) ( 1152 -1024 0 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( 1024 -1024 0 ) ( 1024 1024 0 ) ( 1024 -1024 1024 ) eX/eXmetalrib01 0 0 0 1 1 0 0 0 +} +// brush 3 +{ +( -1024 -1024 1024 ) ( 1024 -1024 1024 ) ( -1024 -1152 1024 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( -1024 -1024 1024 ) ( -1024 -1152 1024 ) ( -1024 -1024 0 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( -1024 -1024 1024 ) ( -1024 -1024 0 ) ( 1024 -1024 1024 ) eX/eXmetalrib01 0 0 0 1 1 0 0 0 +( 1024 -1152 0 ) ( 1024 -1024 0 ) ( -1024 -1152 0 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( 1024 -1152 0 ) ( 1024 -1152 1024 ) ( 1024 -1024 0 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( 1024 -1152 0 ) ( -1024 -1152 0 ) ( 1024 -1152 1024 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +} +// brush 4 +{ +( -1024 1152 1024 ) ( 1024 1152 1024 ) ( -1024 1024 1024 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( -1024 1152 1024 ) ( -1024 1024 1024 ) ( -1024 1152 0 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( -1024 1152 1024 ) ( -1024 1152 0 ) ( 1024 1152 1024 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( 1024 1024 0 ) ( 1024 1152 0 ) ( -1024 1024 0 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( 1024 1024 0 ) ( 1024 1024 1024 ) ( 1024 1152 0 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( 1024 1024 0 ) ( -1024 1024 0 ) ( 1024 1024 1024 ) eX/eXmetalrib01 0 0 0 1 1 0 0 0 +} +// brush 5 +{ +( 1024 1024 0 ) ( 1024 -1024 0 ) ( -1024 1024 0 ) eX/eXmetalFloor02 0 0 0 1 1 0 0 0 +( 1024 1024 0 ) ( -1024 1024 0 ) ( 1024 1024 -128 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( 1024 1024 0 ) ( 1024 1024 -128 ) ( 1024 -1024 0 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( -1024 -1024 -128 ) ( 1024 -1024 -128 ) ( -1024 1024 -128 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( -1024 -1024 -128 ) ( -1024 -1024 0 ) ( 1024 -1024 -128 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +( -1024 -1024 -128 ) ( -1024 1024 -128 ) ( -1024 -1024 0 ) common/caulk 0 0 0 0.25 0.25 0 0 0 +} +} +// entity 1 +{ +"classname" "info_player_deathmatch" +"origin" "0.000000 0.000000 32.000000" +} +// entity 2 +{ +"classname" "item_rockets" +"origin" "256.000000 128.000000 32.000000" +} +// entity 3 +{ +"classname" "item_rockets" +"origin" "256.000000 -128.000000 32.000000" +} +// entity 4 +{ +"classname" "weapon_minelayer" +"origin" "256.000000 0.000000 32.000000" +} diff --git a/qcsrc/client/hud.qc b/qcsrc/client/hud.qc index c1dbede921..0f54a2857f 100644 --- a/qcsrc/client/hud.qc +++ b/qcsrc/client/hud.qc @@ -1478,6 +1478,7 @@ float GetAmmoTypeForWep(float i) case WEP_UZI: return 1; case WEP_CAMPINGRIFLE: return 1; case WEP_GRENADE_LAUNCHER: return 2; + case WEP_MINE_LAYER: return 2; case WEP_ELECTRO: return 3; case WEP_CRYLINK: return 3; case WEP_HLAC: return 3; diff --git a/qcsrc/menu/xonotic/dialog_multiplayer_create_mutators.c b/qcsrc/menu/xonotic/dialog_multiplayer_create_mutators.c index 517f68bd31..c0d16a9597 100644 --- a/qcsrc/menu/xonotic/dialog_multiplayer_create_mutators.c +++ b/qcsrc/menu/xonotic/dialog_multiplayer_create_mutators.c @@ -268,7 +268,7 @@ void XonoticMutatorsDialog_fill(entity me) me.TDempty(me, 0.2); me.TD(me, 1, 2, e = makeXonoticRadioButton(1, "g_start_weapon_laser", "0", "No start weapons")); e.cvarOffValue = "-1"; - makeMulti(e, "g_start_weapon_shotgun g_start_weapon_uzi g_start_weapon_grenadelauncher g_start_weapon_electro g_start_weapon_crylink g_start_weapon_nex g_start_weapon_hagar g_start_weapon_rocketlauncher g_start_weapon_campingrifle g_start_weapon_hlac g_start_weapon_seeker g_start_weapon_minstanex g_start_weapon_hook g_start_weapon_porto g_start_weapon_tuba"); + makeMulti(e, "g_start_weapon_shotgun g_start_weapon_uzi g_start_weapon_grenadelauncher g_start_weapon_minelayer g_start_weapon_electro g_start_weapon_crylink g_start_weapon_nex g_start_weapon_hagar g_start_weapon_rocketlauncher g_start_weapon_campingrifle g_start_weapon_hlac g_start_weapon_seeker g_start_weapon_minstanex g_start_weapon_hook g_start_weapon_porto g_start_weapon_tuba"); me.gotoRC(me, me.rows - 1, 0); me.TD(me, 1, me.columns, e = makeXonoticButton("OK", '0 0 0')); diff --git a/qcsrc/qc-server.cbp b/qcsrc/qc-server.cbp index 00912bea33..eaa30d2d46 100644 --- a/qcsrc/qc-server.cbp +++ b/qcsrc/qc-server.cbp @@ -192,6 +192,7 @@ + diff --git a/qcsrc/server/miscfunctions.qc b/qcsrc/server/miscfunctions.qc index 3db4e47be8..d423242407 100644 --- a/qcsrc/server/miscfunctions.qc +++ b/qcsrc/server/miscfunctions.qc @@ -989,7 +989,7 @@ void readplayerstartcvars() if (g_weaponarena) { start_weapons = g_weaponarena; - if (g_weaponarena & (WEPBIT_GRENADE_LAUNCHER | WEPBIT_HAGAR | WEPBIT_ROCKET_LAUNCHER)) + if (g_weaponarena & (WEPBIT_GRENADE_LAUNCHER | WEPBIT_MINE_LAYER | WEPBIT_HAGAR | WEPBIT_ROCKET_LAUNCHER)) start_ammo_rockets = 999; if (g_weaponarena & WEPBIT_SHOTGUN) start_ammo_shells = 999; diff --git a/qcsrc/server/w_all.qc b/qcsrc/server/w_all.qc index 9a9b383024..83fc8acb5d 100644 --- a/qcsrc/server/w_all.qc +++ b/qcsrc/server/w_all.qc @@ -2,6 +2,7 @@ #include "w_shotgun.qc" #include "w_uzi.qc" #include "w_grenadelauncher.qc" +#include "w_minelayer.qc" #include "w_electro.qc" #include "w_crylink.qc" #include "w_nex.qc" diff --git a/qcsrc/server/w_minelayer.qc b/qcsrc/server/w_minelayer.qc new file mode 100644 index 0000000000..c1b80af1c6 --- /dev/null +++ b/qcsrc/server/w_minelayer.qc @@ -0,0 +1,525 @@ +#ifdef REGISTER_WEAPON +REGISTER_WEAPON(MINE_LAYER, w_minelayer, IT_ROCKETS, 9, WEP_FLAG_NORMAL | WEP_TYPE_SPLASH, BOT_PICKUP_RATING_HIGH, "minelayer", "minelayer", "Mine Layer"); +#else +#ifdef SVQC +.float rl_release; +.float rl_detonate_later; + +void spawnfunc_weapon_minelayer (void) +{ + weapon_defaultspawnfunc(WEP_MINE_LAYER); +} + +void W_Mine_Unregister() +{ + if(self.owner && self.owner.lastrocket == self) + { + self.owner.lastrocket = world; + // self.owner.rl_release = 1; + } +} + +void W_Mine_Explode () +{ + W_Mine_Unregister(); + + if(other.takedamage == DAMAGE_AIM) + if(other.classname == "player") + if(IsDifferentTeam(self.owner, other)) + if(IsFlying(other)) + AnnounceTo(self.owner, "airshot"); + + self.event_damage = SUB_Null; + self.takedamage = DAMAGE_NO; + + RadiusDamage (self, self.owner, cvar("g_balance_minelayer_damage"), cvar("g_balance_minelayer_edgedamage"), cvar("g_balance_minelayer_radius"), world, cvar("g_balance_minelayer_force"), self.projectiledeathtype, other); + + if (self.owner.weapon == WEP_MINE_LAYER) + { + if(self.owner.ammo_rockets < cvar("g_balance_minelayer_ammo")) + { + self.owner.cnt = WEP_MINE_LAYER; + ATTACK_FINISHED(self.owner) = time; + self.owner.switchweapon = w_getbestweapon(self.owner); + } + if(g_laserguided_missile) + ATTACK_FINISHED(self.owner) = time + cvar("g_balance_minelayer_refire") * W_WeaponRateFactor(); + } + remove (self); +} + +void W_Mine_DoRemoteExplode () +{ + W_Mine_Unregister(); + + self.event_damage = SUB_Null; + self.takedamage = DAMAGE_NO; + + RadiusDamage (self, self.owner, cvar("g_balance_minelayer_remote_damage"), cvar("g_balance_minelayer_remote_edgedamage"), cvar("g_balance_minelayer_remote_radius"), world, cvar("g_balance_minelayer_remote_force"), self.projectiledeathtype | HITTYPE_BOUNCE, world); + + if (self.owner.weapon == WEP_MINE_LAYER) + { + if(self.owner.ammo_rockets < cvar("g_balance_minelayer_ammo")) + { + self.owner.cnt = WEP_MINE_LAYER; + ATTACK_FINISHED(self.owner) = time; + self.owner.switchweapon = w_getbestweapon(self.owner); + } + if(g_laserguided_missile) + ATTACK_FINISHED(self.owner) = time + cvar("g_balance_minelayer_refire") * W_WeaponRateFactor(); + } + remove (self); +} + +void W_Mine_RemoteExplode() +{ + if(self.owner.deadflag == DEAD_NO) + if(self.owner.lastrocket) + { + if((self.spawnshieldtime >= 0) + ? (time >= self.spawnshieldtime) // timer + : (vlen(NearestPointOnBox(self.owner, self.origin) - self.origin) > cvar("g_balance_minelayer_radius")) // safety device + ) + { + W_Mine_DoRemoteExplode(); + } + } +} + +void W_Mine_Think (void) +{ + entity e; + vector desireddir, olddir, newdir, desiredorigin, goal; +#if 0 + float cosminang, cosmaxang, cosang; +#endif + float turnrate, velspeed, f; + self.nextthink = time; + if (time > self.cnt) + { + other = world; + self.projectiledeathtype |= HITTYPE_BOUNCE; + W_Mine_Explode (); + return; + } + + if(g_laserguided_missile) + { + // accelerate + makevectors(self.angles_x * '-1 0 0' + self.angles_y * '0 1 0'); + velspeed = cvar("g_balance_minelayer_laserguided_speed") * g_weaponspeedfactor - (self.velocity * v_forward); + if (velspeed > 0) + self.velocity = self.velocity + v_forward * min(cvar("g_balance_minelayer_laserguided_speedaccel") * frametime, velspeed); + } + else + { + // accelerate + makevectors(self.angles_x * '-1 0 0' + self.angles_y * '0 1 0'); + velspeed = cvar("g_balance_minelayer_speed") * g_weaponspeedfactor - (self.velocity * v_forward); + if (velspeed > 0) + self.velocity = self.velocity + v_forward * min(cvar("g_balance_minelayer_speedaccel") * g_weaponspeedfactor * frametime, velspeed); + } + + // laser guided, or remote detonation + if (self.owner.weapon == WEP_MINE_LAYER) + { + if(g_laserguided_missile) + { + if(self.rl_detonate_later) + W_Mine_RemoteExplode(); + + if(cvar("g_balance_minelayer_laserguided_allow_steal")) + { + if(self.owner.laser_on) + { + if(self.attack_finished_single < time) + { + self.attack_finished_single = time + 0.2 + random()*0.3; + //self.enemy = FindLaserTarget(self, 0.7, 0.7); + } + + if(!self.enemy) + self.enemy = self.owner.weaponentity.lasertarget; + } + else self.enemy = world; + } + else // don't allow stealing: always target my owner's laser (if it exists) + self.enemy = self.owner.weaponentity.lasertarget; + + if(self.enemy != world) + { + //bprint("Targeting ", self.enemy.owner.netname, "'s laser\n"); + velspeed = vlen(self.velocity); + e = self.enemy;//self.owner.weaponentity.lasertarget; + turnrate = cvar("g_balance_minelayer_laserguided_turnrate");//0.65; // how fast to turn + desireddir = normalize(e.origin - self.origin); // get direction from my position to the laser target + olddir = normalize(self.velocity); // get my current direction + newdir = normalize(olddir + desireddir * turnrate); // take the average of the 2 directions; not the best method but simple & easy + self.velocity = newdir * velspeed; // make me fly in the new direction at my flight speed + self.angles = vectoangles(self.velocity); // turn model in the new flight direction + + ATTACK_FINISHED(self.owner) = time + 0.2 * W_WeaponRateFactor(); + } + } + else + { + if(self == self.owner.lastrocket) + if not(self.owner.rl_release) + if not(self.BUTTON_ATCK2) + if(cvar("g_balance_minelayer_guiderate")) + if(time > self.pushltime) + if(self.owner.deadflag == DEAD_NO) + { + f = cvar("g_balance_minelayer_guideratedelay"); + if(f) + f = bound(0, (time - self.pushltime) / f, 1); + else + f = 1; + + velspeed = vlen(self.velocity); + + makevectors(self.owner.v_angle); + desireddir = WarpZone_RefSys_TransformVelocity(self.owner, self, v_forward); + desiredorigin = WarpZone_RefSys_TransformOrigin(self.owner, self, self.owner.origin + self.owner.view_ofs); + olddir = normalize(self.velocity); + +#if 0 + // disabled this code because it doesn't do what I want it to do :P + cosminang = cos(cvar("g_balance_minelayer_guidefadeangle") * DEG2RAD); + cosmaxang = cos(cvar("g_balance_minelayer_guidemaxangle") * DEG2RAD); + cosang = desireddir * normalize(self.origin - desiredorigin); + if(cosminang == cosmaxang) + f *= (cosang >= cosminang); + else + f *= bound(0, (cosang - cosmaxang) / (cosminang - cosmaxang), 1); +#endif + + if(!self.count) + { + pointparticles(particleeffectnum("rocket_guide"), self.origin, self.velocity, 1); + // TODO add a better sound here + sound (self.owner, CHAN_WEAPON2, "weapons/rocket_mode.wav", VOL_BASE, ATTN_NORM); + self.count = 1; + } + } + + if(self.rl_detonate_later) + W_Mine_RemoteExplode(); + } + } + + if(self.csqcprojectile_clientanimate == 0) + UpdateCSQCProjectile(self); +} + +void W_Mine_Touch (void) +{ + W_Mine_Unregister(); + + PROJECTILE_TOUCH; + W_Mine_Explode (); +} + +void W_Mine_Damage (entity inflictor, entity attacker, float damage, float deathtype, vector hitloc, vector force) +{ + if (self.health <= 0) + return; + self.health = self.health - damage; + self.angles = vectoangles(self.velocity); + if (self.health <= 0) + W_PrepareExplosionByDamage(attacker, W_Mine_Explode); +} + +void W_Mine_Attack (void) +{ + local entity missile; + local entity flash; + + if not(self.items & IT_UNLIMITED_WEAPON_AMMO) + self.ammo_rockets = self.ammo_rockets - cvar("g_balance_minelayer_ammo"); + + W_SetupShot_ProjectileSize (self, '-3 -3 -3', '3 3 3', FALSE, 5, "weapons/rocket_fire.wav", cvar("g_balance_minelayer_damage")); + pointparticles(particleeffectnum("minelayer_muzzleflash"), w_shotorg, w_shotdir * 1000, 1); + + missile = WarpZone_RefSys_SpawnSameRefSys(self); + missile.owner = self; + self.lastrocket = missile; + if(cvar("g_balance_minelayer_detonatedelay") >= 0) + missile.spawnshieldtime = time + cvar("g_balance_minelayer_detonatedelay"); + else + missile.spawnshieldtime = -1; + missile.pushltime = time + cvar("g_balance_minelayer_guidedelay"); + missile.classname = "rocket"; + missile.bot_dodge = TRUE; + missile.bot_dodgerating = cvar("g_balance_minelayer_damage") * 2; // * 2 because it can be detonated inflight which makes it even more dangerous + + missile.takedamage = DAMAGE_YES; + missile.damageforcescale = cvar("g_balance_minelayer_damageforcescale"); + missile.health = cvar("g_balance_minelayer_health"); + missile.event_damage = W_Mine_Damage; + + missile.movetype = MOVETYPE_FLY; + PROJECTILE_MAKETRIGGER(missile); + missile.projectiledeathtype = WEP_MINE_LAYER; + setsize (missile, '-3 -3 -3', '3 3 3'); // give it some size so it can be shot + + setorigin (missile, w_shotorg - v_forward * 3); // move it back so it hits the wall at the right point + if(g_laserguided_missile && self.laser_on) + W_SetupProjectileVelocity(missile, cvar("g_balance_minelayer_laserguided_speedstart"), 0); + else + W_SetupProjectileVelocity(missile, cvar("g_balance_minelayer_speedstart"), 0); + missile.angles = vectoangles (missile.velocity); + + missile.touch = W_Mine_Touch; + missile.think = W_Mine_Think; + missile.nextthink = time; + missile.cnt = time + cvar("g_balance_minelayer_lifetime"); + missile.flags = FL_PROJECTILE; + + CSQCProjectile(missile, cvar("g_balance_minelayer_guiderate") == 0 && cvar("g_balance_minelayer_speedaccel") == 0 && !g_laserguided_missile, PROJECTILE_ROCKET, FALSE); // because of fly sound + + // muzzle flash for 1st person view + flash = spawn (); + setmodel (flash, "models/flash.md3"); // precision set below + SUB_SetFade (flash, time, 0.1); + flash.effects = EF_ADDITIVE | EF_FULLBRIGHT | EF_LOWPRECISION; + W_AttachToShotorg(flash, '5 0 0'); + + // common properties +} + +void spawnfunc_weapon_minelayer (void); // defined in t_items.qc + +float w_minelayer(float req) +{ + entity rock; + float rockfound; + if (req == WR_AIM) + { + // aim and decide to fire if appropriate + self.BUTTON_ATCK = bot_aim(cvar("g_balance_minelayer_speed"), 0, cvar("g_balance_minelayer_lifetime"), FALSE); + if(skill >= 2) // skill 0 and 1 bots won't detonate rockets! + { + // decide whether to detonate rockets + local entity missile, targetlist, targ; + local float edgedamage, coredamage, edgeradius, recipricoledgeradius, d; + local float selfdamage, teamdamage, enemydamage; + edgedamage = cvar("g_balance_minelayer_edgedamage"); + coredamage = cvar("g_balance_minelayer_damage"); + edgeradius = cvar("g_balance_minelayer_radius"); + recipricoledgeradius = 1 / edgeradius; + selfdamage = 0; + teamdamage = 0; + enemydamage = 0; + targetlist = findchainfloat(bot_attack, TRUE); + missile = find(world, classname, "rocket"); + while (missile) + { + if (missile.owner != self) + { + missile = find(missile, classname, "rocket"); + continue; + } + targ = targetlist; + while (targ) + { + d = vlen(targ.origin + (targ.mins + targ.maxs) * 0.5 - missile.origin); + d = bound(0, edgedamage + (coredamage - edgedamage) * sqrt(1 - d * recipricoledgeradius), 10000); + // count potential damage according to type of target + if (targ == self) + selfdamage = selfdamage + d; + else if (targ.team == self.team && teams_matter) + teamdamage = teamdamage + d; + else if (bot_shouldattack(targ)) + enemydamage = enemydamage + d; + targ = targ.chain; + } + missile = find(missile, classname, "rocket"); + } + local float desirabledamage; + desirabledamage = enemydamage; + if (teamplay != 1 && time > self.invincible_finished && time > self.spawnshieldtime) + desirabledamage = desirabledamage - selfdamage * cvar("g_balance_selfdamagepercent"); + if (self.team && teamplay != 1) + desirabledamage = desirabledamage - teamdamage; + + missile = find(world, classname, "rocket"); + while (missile) + { + if (missile.owner != self) + { + missile = find(missile, classname, "rocket"); + continue; + } + makevectors(missile.v_angle); + targ = targetlist; + if (skill > 9) // normal players only do this for the target they are tracking + { + targ = targetlist; + while (targ) + { + if ( + (v_forward * normalize(missile.origin - targ.origin)< 0.1) + && desirabledamage > 0.1*coredamage + )self.BUTTON_ATCK2 = TRUE; + targ = targ.chain; + } + }else{ + local float distance; distance= bound(300,vlen(self.origin-self.enemy.origin),30000); + //As the distance gets larger, a correct detonation gets near imposible + //Bots are assumed to use the rocket spawnfunc_light to see if the rocket gets near a player + if(v_forward * normalize(missile.origin - self.enemy.origin)< 0.1) + if(self.enemy.classname == "player") + if(desirabledamage >= 0.1*coredamage) + if(random()/distance*300 > frametime*bound(0,(10-skill)*0.2,1)) + self.BUTTON_ATCK2 = TRUE; + // dprint(ftos(random()/distance*300),">");dprint(ftos(frametime*bound(0,(10-skill)*0.2,1)),"\n"); + } + + missile = find(missile, classname, "rocket"); + } + // if we would be doing at X percent of the core damage, detonate it + // but don't fire a new shot at the same time! + if (desirabledamage >= 0.75 * coredamage) //this should do group damage in rare fortunate events + self.BUTTON_ATCK2 = TRUE; + if ((skill > 6.5) && (selfdamage > self.health)) + self.BUTTON_ATCK2 = FALSE; + //if(self.BUTTON_ATCK2 == TRUE) + // dprint(ftos(desirabledamage),"\n"); + if (self.BUTTON_ATCK2 == TRUE) self.BUTTON_ATCK = FALSE; + } + } + else if (req == WR_THINK) + { + if(g_laserguided_missile) + { + if (self.BUTTON_ATCK && self.rl_release) + { + rockfound = 0; + for(rock = world; (rock = find(rock, classname, "rocket")); ) if(rock.owner == self) + { + if(!rock.rl_detonate_later) + { + rock.rl_detonate_later = TRUE; + rockfound = 1; + } + } + if(rockfound) + sound (self, CHAN_WEAPON2, "weapons/rocket_det.wav", VOL_BASE, ATTN_NORM); + else + { + if (weapon_prepareattack(0, cvar("g_balance_minelayer_refire"))) + { + W_Mine_Attack(); + weapon_thinkf(WFRAME_FIRE1, cvar("g_balance_minelayer_animtime"), w_ready); + } + } + self.rl_release = 0; + } + if (!self.BUTTON_ATCK) + self.rl_release = 1; + if (self.BUTTON_ATCK2) + if(self.exteriorweaponentity.attack_finished_single < time) + { + self.exteriorweaponentity.attack_finished_single = time + 0.4; + self.laser_on = !self.laser_on; + // UGLY WORKAROUND: play this on CHAN_WEAPON2 so it can't cut off fire sounds + sound (self, CHAN_WEAPON2, "weapons/rocket_mode.wav", VOL_BASE, ATTN_NORM); + } + } + else + { + if (self.BUTTON_ATCK) + { + if(self.rl_release || cvar("g_balance_minelayer_guidestop")) + if(weapon_prepareattack(0, cvar("g_balance_minelayer_refire"))) + { + W_Mine_Attack(); + weapon_thinkf(WFRAME_FIRE1, cvar("g_balance_minelayer_animtime"), w_ready); + self.rl_release = 0; + } + } + else + self.rl_release = 1; + + if (self.BUTTON_ATCK2) + { + rockfound = 0; + for(rock = world; (rock = find(rock, classname, "rocket")); ) if(rock.owner == self) + { + if(!rock.rl_detonate_later) + { + rock.rl_detonate_later = TRUE; + rockfound = 1; + } + } + if(rockfound) + sound (self, CHAN_WEAPON2, "weapons/rocket_det.wav", VOL_BASE, ATTN_NORM); + } + } + } + else if (req == WR_PRECACHE) + { + precache_model ("models/flash.md3"); + precache_model ("models/weapons/g_rl.md3"); + precache_model ("models/weapons/v_rl.md3"); + precache_model ("models/weapons/h_rl.iqm"); + precache_sound ("weapons/rocket_det.wav"); + precache_sound ("weapons/rocket_fire.wav"); + precache_sound ("weapons/rocket_mode.wav"); + if (g_laserguided_missile) + { + precache_model ("models/laser_dot.mdl"); // mine layer + } + } + else if (req == WR_SETUP) + { + weapon_setup(WEP_MINE_LAYER); + self.rl_release = 1; + } + else if (req == WR_CHECKAMMO1) + { + // don't switch while guiding a missile + if ((ATTACK_FINISHED(self) <= time || self.weapon != WEP_MINE_LAYER) + && self.ammo_rockets < cvar("g_balance_minelayer_ammo")) + return FALSE; + } + else if (req == WR_CHECKAMMO2) + return FALSE; + else if (req == WR_RESETPLAYER) + { + self.rl_release = 0; + } + return TRUE; +}; +#endif +#ifdef CSQC +float w_minelayer(float req) +{ + if(req == WR_IMPACTEFFECT) + { + vector org2; + org2 = w_org + w_backoff * 12; + pointparticles(particleeffectnum("rocket_explode"), org2, '0 0 0', 1); + if(!w_issilent) + sound(self, CHAN_PROJECTILE, "weapons/rocket_impact.wav", VOL_BASE, ATTN_NORM); + } + else if(req == WR_PRECACHE) + { + precache_sound("weapons/rocket_impact.wav"); + } + else if (req == WR_SUICIDEMESSAGE) + w_deathtypestring = "%s exploded"; + else if (req == WR_KILLMESSAGE) + { + if(w_deathtype & HITTYPE_BOUNCE) // (remote detonation) + w_deathtypestring = "%s got too close to %s's rocket"; + else if(w_deathtype & HITTYPE_SPLASH) + w_deathtypestring = "%s almost dodged %s's rocket"; + else + w_deathtypestring = "%s ate %s's rocket"; + } + return TRUE; +} +#endif +#endif