2024-09-04 23:46:54 +10:00

711 lines
18 KiB
Plaintext

#include maps\mp\_utility;
#include maps\mp\gametypes\_hud_util;
#include common_scripts\utility;
init()
{
precacheString( &"MP_LASE_TARGET_FOR_GUIDED_MORTAR" );
precacheString( &"MP_WAIT_FOR_MORTAR_READY" );
precacheString( &"MP_MORTAR_ROUNDS_DEPLETED" );
precacheString( &"SPLASHES_DESTROYED_REMOTE_MORTAR" );
precacheModel( "vehicle_predator_b" );
precacheItem( "remote_mortar_missile_mp" );
precacheItem( "mortar_remote_mp" );
precacheItem( "mortar_remote_zoom_mp" );
precacheShader( "compass_waypoint_bomb" );
precacheShader( "viper_locked_box" );
PrecacheMiniMapIcon( "compass_objpoint_reaper_friendly" );
PrecacheMiniMapIcon( "compass_objpoint_reaper_enemy" );
level._remote_mortar_fx["laserTarget"] = loadfx("nx/misc/nx_laser_glow");
level._killstreakFuncs["reaper"] = ::tryUseRemoteMortar;
/#
SetDevDvarIfUninitialized( "scr_remote_mortar_timeout", 40.0 );
#/
}
tryUseRemoteMortar( lifeId )
{
if ( isDefined( self.lastStand ) && !self _hasPerk( "specialty_finalstand" ) )
{
self iPrintLnBold( &"MP_UNAVILABLE_IN_LASTSTAND" );
return false;
}
self setUsingRemote( "remote_mortar" );
/* result = self maps\mp\killstreaks\_killstreaks::initRideKillstreak();
if ( result != "success" )
{
if ( result != "disconnect" )
self clearUsingRemote();
return false;
}
*/
self maps\mp\_matchdata::logKillstreakEvent( "remote_mortar", self.origin );
return startRemoteMortar( lifeId );
}
startRemoteMortar( lifeId )
{
remote = spawnRemote( lifeId, self );
if ( !isDefined( remote ) )
return false;
self setPlayerData( "reaperHUD", "targetDistance", -1 );
self remoteRide( remote );
self thread teamPlayerCardSplash( "used_remote_mortar", self );
return true;
}
spawnRemote( lifeId, owner )
{
remote = spawnPlane( owner, "script_model", level._UAVRig getTagOrigin( "tag_origin" ), "compass_objpoint_reaper_friendly", "compass_objpoint_reaper_enemy" );
if ( !isDefined( remote ) )
return undefined;
remote setModel( "vehicle_predator_b" );
remote.lifeId = lifeId;
remote.team = owner.team;
remote.owner = owner;
remote.numFlares = 1;
remote setCanDamage( true );
remote thread damageTracker();
remote.heliType = "remote_mortar";
// for target lists (javelin, stinger, sam, emp, etc)
remote.uavType = "remote_mortar";
maps\mp\killstreaks\_uav::addUAVModel( remote );
// same height and radius as the AC130 with random angle and counter rotation
zOffset = 6300;
angle = randomInt( 360 );
radiusOffset = 6100;
xOffset = cos( angle ) * radiusOffset;
yOffset = sin( angle ) * radiusOffset;
angleVector = vectorNormalize( (xOffset,yOffset,zOffset) );
angleVector = ( angleVector * 6100 );
remote linkTo( level._UAVRig, "tag_origin", angleVector, (0,angle-90,10) );
remote thread handleDeath( owner );
remote thread handleTimeout( owner );
remote thread handleOwnerChangeTeam( owner );
remote thread handleOwnerDisconnect( owner );
remote thread handleIncomingStinger();
remote thread handleIncomingSAM();
return remote;
}
lookCenter()
{
wait( 0.05 );
lookVec = vectorToAngles( level._UAVRig.origin - self GetEye() );
self setPlayerAngles( lookVec );
}
remoteRide( remote )
{
self setPlayerData( "reaperHUD", "zoomed", false );
self _giveWeapon("mortar_remote_mp");
self SwitchToWeapon("mortar_remote_mp");
self ThermalVisionFOFOverlayOn();
if ( getDvarInt( "camera_thirdPerson" ) )
self setThirdPersonDOF( false );
self PlayerLinkWeaponviewToDelta( remote, "tag_player", 1.0, 35, 35, 15, 55 );
self thread lookCenter();
self thread remoteTargeting( remote );
self thread remoteFiring( remote );
self thread remoteZoom( remote );
}
remoteTargeting( remote )
{
level endon( "game_ended" );
self endon( "disconnect" );
remote endon( "remote_done" );
remote endon( "death" );
remote.targetEnt = SpawnFx( level._remote_mortar_fx["laserTarget"], (0,0,0) );
remote.targetEnt.icon = remote.targetEnt maps\mp\_entityheadIcons::setHeadIcon( self, "viper_locked_box", (0,0,-270), 100, 100, false, 0.05, false, false, false, false );
remote.targetEnt.icon.color = (0.9, 0.7, 0.2);
/*laserPos = remote getTagOrigin( "tag_player" );
remote.laserEnt = Spawn( "script_model", laserPos );
remote.laserEnt linkTo( remote, "tag_player", (0,0,0), (0,0,0) );
remote.laserEnt LaserOn();*/
while ( true )
{
origin = self GetEye();
forward = AnglesToForward( self GetPlayerAngles() );
endpoint = origin + forward * 15000;
traceData = BulletTrace( origin, endpoint, false, remote.targetEnt );
if ( isDefined( traceData["position"] ) )
{
remote.targetEnt.origin = traceData["position"];
triggerFX( remote.targetEnt );
}
wait( 0.05 );
}
}
remoteFiring( remote )
{
level endon( "game_ended" );
self endon( "disconnect" );
remote endon( "remote_done" );
remote endon( "death" );
curTime = getTime();
lastFireTime = curTime-750;
shotsFired = 0;
while( true )
{
curTime = getTime();
if ( self attackButtonPressed() && ( curTime - lastFireTime > 3000 ) )
{
shotsFired++;
lastFireTime = curTime;
self playLocalSound( "stinger_locking" );
self PlayRumbleOnEntity( "damage_heavy" );
origin = self GetEye();
forward = AnglesToForward( self GetPlayerAngles() );
right = AnglesToRight( self GetPlayerAngles() );
offset = origin + ( forward * 100 ) + ( right * -100 );
missile = MagicBullet( "remote_mortar_missile_mp", offset, remote.targetEnt.origin, self );
missile Missile_SetTargetEnt( remote.targetEnt );
missile Missile_SetFlightmodeDirect();
missile.icon = missile maps\mp\_entityheadIcons::setHeadIcon( self, "compass_waypoint_bomb", (0,0,-10), 10, 10, false, 0.05, false, false, false, false );
missile thread remoteMissileDistance( remote );
remote.missileIcon = missile.icon;
missile waittill( "death" );
if ( isDefined( missile.icon ) )
{
missile.icon destroy();
missile.icon = undefined;
}
self setPlayerData( "reaperHUD", "targetDistance", -1 );
if ( shotsFired == 14 )
break;
}
else
wait( 0.05 );
}
self remoteEndRide( remote );
remote thread remoteLeave();
}
remoteZoom( remote )
{
level endon( "game_ended" );
self endon( "disconnect" );
remote endon( "remote_done" );
remote endon( "death" );
remote.zoomed = false;
while ( true )
{
if ( self adsButtonPressed() )
{
if ( remote.zoomed == false )
{
self _giveWeapon("mortar_remote_zoom_mp");
self SwitchToWeapon("mortar_remote_zoom_mp");
remote.zoomed = true;
self setPlayerData( "reaperHUD", "zoomed", true );
if ( isDefined( remote.targetEnt.icon ) )
{
remote.targetEnt.icon destroy();
remote.targetEnt.icon = undefined;
remote.targetEnt.icon = remote.targetEnt maps\mp\_entityheadIcons::setHeadIcon( self, "viper_locked_box", (0,0,-50), 100, 100, false, 0.05, false, false, false, false );
remote.targetEnt.icon.color = (0.9, 0.7, 0.2);
}
}
}
else
{
if ( remote.zoomed == true )
{
self _giveWeapon("mortar_remote_mp");
self SwitchToWeapon("mortar_remote_mp");
remote.zoomed = false;
self setPlayerData( "reaperHUD", "zoomed", false );
if ( isDefined( remote.targetEnt.icon ) )
{
remote.targetEnt.icon destroy();
remote.targetEnt.icon = undefined;
remote.targetEnt.icon = remote.targetEnt maps\mp\_entityheadIcons::setHeadIcon( self, "viper_locked_box", (0,0,-270), 100, 100, false, 0.05, false, false, false, false );
remote.targetEnt.icon.color = (0.9, 0.7, 0.2);
}
}
}
wait( 0.05 );
}
}
remoteMissileDistance( remote )
{
level endon( "game_ended" );
remote endon( "death" );
remote endon( "remote_done" );
self endon( "death" );
while( true )
{
targetDist = distance( self.origin, remote.targetent.origin );
remote.owner setPlayerData( "reaperHUD", "targetDistance", int( targetDist / 12 ) );
wait( 0.05 );
}
}
remoteEndRide( remote )
{
if ( isDefined( remote ) )
{
remote notify( "helicopter_done" );
if ( isDefined( remote.laserEnt ) )
remote.laserEnt LaserOff();
if ( isDefined( remote.targetEnt ) && isDefined( remote.targetEnt.icon ) )
{
remote.targetEnt.icon destroy();
remote.targetEnt.icon = undefined;
}
if ( isDefined( remote.missileIcon ) )
{
remote.missileIcon destroy();
remote.missileIcon = undefined;
}
}
self ThermalVisionFOFOverlayOff();
self unlink();
self clearUsingRemote();
if ( getDvarInt( "camera_thirdPerson" ) )
self setThirdPersonDOF( true );
self switchToWeapon( self getLastWeapon() );
weaponList = self GetWeaponsListExclusives();
foreach ( weapon in weaponList )
self takeWeapon( weapon );
}
handleTimeout( owner )
{
level endon( "game_ended" );
owner endon( "disconnect" );
self endon( "death" );
lifeSpan = 40.0;
/#
lifeSpan = GetDvarInt( "scr_remote_mortar_timeout", lifeSpan );
#/
maps\mp\gametypes\_hostmigration::waitLongDurationWithHostMigrationPause( lifeSpan );
if ( isDefined( owner ) )
owner remoteEndRide( self );
self thread remoteLeave();
}
handleDeath( owner )
{
level endon( "game_ended" );
owner endon( "disconnect" );
self endon( "remote_done" );
self waittill( "death" );
if ( isDefined( owner ) )
owner remoteEndRide( self );
level thread removeRemote( self );
}
handleOwnerChangeTeam( owner )
{
level endon( "game_ended" );
self endon( "remote_done" );
self endon( "death" );
owner endon( "disconnect" );
owner waittill_any( "joined_team", "joined_spectators" );
if ( isDefined( owner ) )
owner remoteEndRide( self );
self thread remoteLeave();
}
handleOwnerDisconnect( owner )
{
level endon( "game_ended" );
self endon( "remote_done" );
self endon( "death" );
owner waittill( "disconnect" );
self thread remoteLeave();
}
removeRemote( remote )
{
self notify( "remote_removed" );
if ( isDefined( remote.targetEnt ) )
{
if ( isDefined( remote.targetEnt.icon ) )
{
remote.targetEnt.icon destroy();
remote.targetEnt.icon = undefined;
}
remote.targetEnt delete();
}
if ( isDefined( remote.laserEnt ) )
remote.laserEnt delete();
if( IsDefined( remote ) )
{
remote delete();
maps\mp\killstreaks\_uav::removeUAVModel( remote );
}
}
remoteLeave()
{
level endon( "game_ended" );
self endon( "death" );
self notify( "remote_done" );
self unlink();
destPoint = self.origin + ( AnglesToForward( self.angles ) * 20000 );
self moveTo( destPoint, 30 );
PlayFXOnTag( level._effect[ "ac130_engineeffect" ] , self, "tag_origin" );
maps\mp\gametypes\_hostmigration::waitLongDurationWithHostMigrationPause( 3 );
self moveTo( destPoint, 4, 4, 0.0 );
maps\mp\gametypes\_hostmigration::waitLongDurationWithHostMigrationPause( 4 );
level thread removeRemote( self );
}
// Entities spawned from SpawnPlane do not respond to pre-damage callbacks
// so we have to wait until we get the post-damage event.
//
// Because the damage has already happened by the time we find out about it,
// we need to use an artificially high health value, restore it on erroneous damage
// events and track a virtual damage taken against a virtual max health.
damageTracker()
{
level endon( "game_ended" );
self.owner endon( "disconnect" );
self.health = 999999; // keep it from dying anywhere in code
self.maxHealth = 1000; // this is the health we'll check
self.damageTaken = 0; // how much damage has it taken
while( true )
{
self waittill( "damage", damage, attacker, direction_vec, point, meansOfDeath, modelName, tagName, partName, iDFlags, weapon );
// don't allow people to destroy things on their team if FF is off
if ( !maps\mp\gametypes\_weapons::friendlyFireCheck( self.owner, attacker ) )
continue;
if ( !IsDefined( self ) )
return;
if ( isDefined( iDFlags ) && ( iDFlags & level._iDFLAGS_PENETRATION ) )
self.wasDamagedFromBulletPenetration = true;
self.wasDamaged = true;
modifiedDamage = damage;
if( IsPlayer( attacker ) )
{
attacker maps\mp\gametypes\_damagefeedback::updateDamageFeedback( "" );
if( meansOfDeath == "MOD_RIFLE_BULLET" || meansOfDeath == "MOD_PISTOL_BULLET" )
{
if ( attacker _hasPerk( "specialty_armorpiercing" ) )
modifiedDamage += damage * level._armorPiercingMod;
}
}
if( IsDefined( weapon ) )
{
switch( weapon )
{
case "stinger_mp":
case "javelin_mp":
self.largeProjectileDamage = true;
modifiedDamage = self.maxhealth + 1;
break;
case "sam_projectile_mp":
self.largeProjectileDamage = true;
modifiedDamage *= 2.0; // takes about 1 burst of sam rockets
break;
}
}
self.damageTaken += modifiedDamage;
if ( self.damageTaken >= self.maxHealth )
{
if ( isPlayer( attacker ) && ( !isDefined( self.owner ) || attacker != self.owner ) )
{
self Hide();
forward = ( AnglesToRight( self.angles ) * 200 );
playFx ( level._uav_fx[ "explode" ], self.origin, forward );
attacker notify( "destroyed_killstreak", weapon );
thread teamPlayerCardSplash( "callout_destroyed_remote_mortar", attacker );
//attacker thread maps\mp\gametypes\_rank::giveRankXP( "kill", 50, weapon, meansOfDeath );
//attacker thread maps\mp\gametypes\_rank::xpEventPopup( &"SPLASHES_DESTROYED_REMOTE_MORTAR" );
//thread maps\mp\gametypes\_missions::vehicleKilled( self.owner, self, undefined, attacker, damage, meansOfDeath, weapon );
}
self notify( "death" );
return;
}
}
}
handleIncomingStinger() // self == remote mortar
{
level endon ( "game_ended" );
self endon ( "death" );
while( true )
{
level waittill ( "stinger_fired", player, missile, lockTarget );
if ( !IsDefined( lockTarget ) || (lockTarget != self) )
continue;
missile thread stingerProximityDetonate( lockTarget, player );
}
}
stingerProximityDetonate( missileTarget, player ) // self == missile
{
self endon ( "death" );
if( IsDefined( missileTarget.owner ) )
missileTarget.owner PlayLocalSound( "missile_incoming" );
self Missile_SetTargetEnt( missileTarget );
minDist = Distance( self.origin, missileTarget GetPointInBounds( 0, 0, 0 ) );
lastCenter = missileTarget GetPointInBounds( 0, 0, 0 );
while( true )
{
// already destroyed
if( !IsDefined( missileTarget ) )
center = lastCenter;
else
center = missileTarget GetPointInBounds( 0, 0, 0 );
lastCenter = center;
curDist = Distance( self.origin, center );
if( curDist < 3000 && missileTarget.numFlares > 0 )
{
missileTarget.numFlares--;
missileTarget thread maps\mp\killstreaks\_helicopter::playFlareFx();
newTarget = missileTarget maps\mp\killstreaks\_helicopter::deployFlares();
self Missile_SetTargetEnt( newTarget );
missileTarget = newTarget;
if( IsDefined( missileTarget.owner ) )
missileTarget.owner StopLocalSound( "missile_incoming" );
return;
}
if( curDist < minDist )
minDist = curDist;
if( curDist > minDist )
{
if( curDist > 1536 )
return;
if( IsDefined( missileTarget.owner ) )
{
missileTarget.owner stopLocalSound( "missile_incoming" );
if( level._teambased )
{
if( missileTarget.team != player.team )
RadiusDamage( self.origin, 1000, 1000, 1000, player, "MOD_EXPLOSIVE", "stinger_mp" );
}
else
{
RadiusDamage( self.origin, 1000, 1000, 1000, player, "MOD_EXPLOSIVE", "stinger_mp" );
}
}
self Hide();
wait( 0.05 );
self delete();
}
wait ( 0.05 );
}
}
handleIncomingSAM() // self == remote mortar
{
level endon ( "game_ended" );
self endon ( "death" );
while( true )
{
level waittill ( "sam_fired", player, missileGroup, lockTarget );
if ( !IsDefined( lockTarget ) || (lockTarget != self) )
continue;
level thread samProximityDetonate( lockTarget, player, missileGroup );
}
}
samProximityDetonate( missileTarget, player, missileGroup )
{
missileTarget endon( "death" );
if( IsDefined( missileTarget.owner ) )
missileTarget.owner PlayLocalSound( "missile_incoming" );
sam_projectile_damage = 150; // this should match the gdt entry
sam_projectile_damage_radius = 1000;
minDist = [];
for( i = 0; i < missileGroup.size; i++ )
{
if( IsDefined( missileGroup[ i ] ) )
minDist[ i ] = Distance( missileGroup[ i ].origin, missileTarget GetPointInBounds( 0, 0, 0 ) );
else
minDist[ i ] = undefined;
}
while( true )
{
center = missileTarget GetPointInBounds( 0, 0, 0 );
curDist = [];
for( i = 0; i < missileGroup.size; i++ )
{
if( IsDefined( missileGroup[ i ] ) )
curDist[ i ] = Distance( missileGroup[ i ].origin, center );
}
for( i = 0; i < curDist.size; i++ )
{
if( IsDefined( curDist[ i ] ) )
{
// if one of the missiles in the group get close, set off flares and redirect them all
if( curDist[ i ] < 3000 && missileTarget.numFlares > 0 )
{
missileTarget.numFlares--;
missileTarget thread maps\mp\killstreaks\_helicopter::playFlareFx();
newTarget = missileTarget maps\mp\killstreaks\_helicopter::deployFlares();
for( j = 0; j < missileGroup.size; j++ )
{
if( IsDefined( missileGroup[ j ] ) )
{
missileGroup[ j ] Missile_SetTargetEnt( newTarget );
}
}
if( IsDefined( missileTarget.owner ) )
missileTarget.owner StopLocalSound( "missile_incoming" );
return;
}
if( curDist[ i ] < minDist[ i ] )
minDist[ i ] = curDist[ i ];
if( curDist[ i ] > minDist[ i ] )
{
if( curDist[ i ] > 1536 )
return;
if( IsDefined( missileTarget.owner ) )
{
missileTarget.owner StopLocalSound( "missile_incoming" );
if( level._teambased )
{
if( missileTarget.team != player.team )
RadiusDamage( missileGroup[ i ].origin, sam_projectile_damage_radius, sam_projectile_damage, sam_projectile_damage, player, "MOD_EXPLOSIVE", "sam_projectile_mp" );
}
else
{
RadiusDamage( missileGroup[ i ].origin, sam_projectile_damage_radius, sam_projectile_damage, sam_projectile_damage, player, "MOD_EXPLOSIVE", "sam_projectile_mp" );
}
}
missileGroup[ i ] Hide();
wait ( 0.05 );
missileGroup[ i ] delete();
}
}
}
wait ( 0.05 );
}
}