cod5-sdk/raw/maps/_vehicle.gsc

6739 lines
198 KiB
Plaintext

/*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * *
VEHICLE script
This handles playing the various effects and animations on a vehicle.
It handles initializing a vehicle( giving it life, turrets, machine guns, treads and things )
It also handles spawning of vehicles in a very ugly way for now, we're getting code to make it pretty
Most things you see in the vehicle menu in Radiant are handled here. There's all sorts of properties
that you can set on a trigger to access some of this functionality. A trigger can spawn a vehicle,
toggle different behaviors,
HIGH LEVEL FUNCTIONS
// vehicle_init( vehicle )
this give the vehicle life, treads, turrets, machine guns, all that good stuff
// main()
this is setup, sets up spawners, trigger associations etc is ran on first frame by _load
// trigger_process( trigger, vehicles )
since triggers are multifunction I made them all happen in the same thread so that
the sequencing would be easy to handle
// vehicle_paths()
This makes the nodes get notified trigger when they are hit by a vehicle, we hope
to move this functionality to CODE side because we have to use a lot of wrappers for
attaching a vehicle to a path
* * * * * * * * * * * * * * * * * * * * * * * * * * * * *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * *
*/
#include maps\_utility;
#include common_scripts\utility;
#using_animtree( "vehicles" );
init_vehicles()
{
// CODER_MOD: Bryce (05/08/08): Useful output for debugging replay system
/#
if( getdebugdvar( "replay_debug" ) == "1" )
println("File: _vehicle.gsc. Function: init_vehicles()\n");
#/
precachemodel( "fx" );
if(isdefined(level.bypassVehicleScripts))
return;
level.heli_default_decel = 10; // will likely go back to 10 in the near future
thread dump_handle();// press the hotkey and xenon will export a map with all the current vehicle positions
// MikeD (06/14/07): Removed, and converted it into level.aircraft_list
// init_helicopter_list();
// MikeD (06/14/07): Added aircraft
init_aircraft_list();
// MikeD (4/3/2008): Added boats
init_boat_list();
// this removes the requirement of putting vehicletype on vehicles
maps\_vehicletypes::setup_types();
// This exports script_vehicles for script_model based vehicles in the level
generate_colmaps_vehicles();
// put all the vehicles with targetnames into an array so we can spawn vehicles from
// a string instead of their vehicle group #
setup_targetname_spawners();
// vehicle related dvar initializing goes here
setup_dvars();
// initialize all the level wide vehicle system variables
setup_levelvars();
array_thread( getentarray( "truckjunk", "targetname" ), ::truckjunk );
// drives vehicle through level
// helpful on levels where vehicles don't always make it through the whole path
/# vclogin_vehicles(); #/
// pre - associate ai and spawners with their vehicles
setup_ai();
// pre - associate vehicle triggers and vehicle nodes with stuff.
setup_triggers();
// check precacheing of vehicle scripts.
allvehiclesprespawn = precache_scripts();
// setup spawners and non - spawning vehicles
setup_vehicles( allvehiclesprespawn );
// send the setup triggers to be processed
array_levelthread( level.vehicle_processtriggers, ::trigger_process, allvehiclesprespawn );
// CHECKME
level.vehicle_processtriggers = undefined;
init_level_has_vehicles();
// CODER_MOD: Bryce (05/08/08): Useful output for debugging replay system
/#
if( getdebugdvar( "replay_debug" ) == "1" )
println("File: _vehicle.gsc. Function: init_vehicles() - COMPLETE\n");
#/
// CODER_MOD: Tommy K
level.vehicle_enemy_tanks = [];
level.vehicle_enemy_tanks[ "vehicle_ger_tracked_king_tiger" ] = true;
level.vehicle_enemy_tanks[ "vehicle_ger_tracked_panther" ] = true;
level.vehicle_enemy_tanks[ "vehicle_ger_tracked_panzer4" ] = true;
level.vehicle_enemy_tanks[ "vehicle_jap_tracked_type97shinhoto" ] = true;
}
init_aircraft_list()
{
level.aircraft_list = [];
level.aircraft_list[ "player_corsair" ] = true;
level.aircraft_list[ "rufe" ] = true;
level.aircraft_list[ "corsair" ] = true;
//-- GavinL (11/12/2007): Added planes specific for use in PBY (aka The Flying Level)
level.aircraft_list[ "zero" ] = true;
level.aircraft_list[ "pby" ] = true;
level.aircraft_list[ "pby_blackcat" ] = true;
// MikeD (8/13/2007): Removed until we set it up better, currently this is intended only for the FLY level.
//level.aircraft_list[ "stuka" ] = true;
level.aircraft_list[ "jap_gunboat" ] = true;
level.aircraft_list[ "il2" ] = true;
}
init_boat_list()
{
level.boat_list = [];
level.boat_list["rubber_raft"] = true;
level.boat_list[ "jap_ptboat" ] = true;
level.boat_list[ "jap_shinyo" ] = true;
level.boat_list[ "jap_merchant_ship" ] = true;
}
init_level_has_vehicles()
{
level.levelHasVehicles = false;
// if there is a vehicle spawner then the level has vehicles
if ( level.vehicle_spawngroup.size > 0 )
level.levelHasVehicles = true;
if ( level.vehicle_spawners.size > 0 )
level.levelHasVehicles = true;
// if there are any vehicles prespawned of any team in the level then there are vehicles
if ( level.vehicles[ "allies" ].size > 0 )
level.levelHasVehicles = true;
if ( level.vehicles[ "axis" ].size > 0 )
level.levelHasVehicles = true;
if ( level.vehicles[ "neutral" ].size > 0 )
level.levelHasVehicles = true;
// if there are any script_vehicle classname there are vehicles
classname = getentarray( "script_vehicle", "classname" );
if ( classname.size > 0 )
level.levelHasVehicles = true;
}
delete_group( vehicle, group )
{
println( "delete group is just delete() now! please update your scripts" );
vehicle delete();
return;
}
trigger_getlinkmap( trigger )
{
linkMap = [];
if( isdefined( trigger.script_linkTo ) )
{
links = strtok( trigger.script_linkTo, " " );
for( i = 0; i < links.size; i ++ )
linkMap[ links[ i ] ] = true;
links = undefined;
}
return linkMap;
}
// setup_script_gatetrigger( trigger, linkMap )
setup_script_gatetrigger( trigger )
{
gates = [];
if( isdefined( trigger.script_gatetrigger ) )
return level.vehicle_gatetrigger[ trigger.script_gatetrigger ];
return gates;
}
// setup_script_vehiclespawngroup( trigger, vehicles, linkMap )
setup_script_vehiclespawngroup( trigger, vehicles )
{
script_vehiclespawngroup = false;
if( isdefined( trigger.script_vehiclespawngroup ) )
script_vehiclespawngroup = true;
return script_vehiclespawngroup;
}
trigger_process( trigger, vehicles )
{
// these triggers only trigger once where vehicle paths trigger everytime a vehicle crosses them
if( isdefined( trigger.classname ) && ( trigger.classname == "trigger_multiple" || trigger.classname == "trigger_radius" || trigger.classname == "trigger_lookat" ) )
bTriggeronce = true;
else
bTriggeronce = false;
trigger.processed_trigger = undefined; // clear out this flag that was used to get the trigger to this point.
// override to make a trigger loop
if( isdefined( trigger.script_noteworthy ) && trigger.script_noteworthy == "trigger_multiple" )
bTriggeronce = false;
gates = setup_script_gatetrigger( trigger );
script_vehiclespawngroup = isdefined( trigger.script_vehiclespawngroup );
// origin paths and script struct paths get this value
script_vehicledetour = isdefined( trigger.script_vehicledetour ) && ( is_node_script_origin( trigger ) || is_node_script_struct( trigger ) ) ;
// ground paths get this value
detoured = isdefined( trigger.detoured ) && !( is_node_script_origin( trigger ) || is_node_script_struct( trigger ) );
gotrigger = true;
vehicles = undefined;
while( gotrigger )
{
trigger waittill( "trigger", other );
if ( isdefined( trigger.script_vehicletriggergroup ) )
{
if( !isdefined( other.script_vehicletriggergroup ) )
continue;
if( other.script_vehicletriggergroup != trigger.script_vehicletriggergroup )
continue;
}
if( isdefined( trigger.enabled ) && !trigger.enabled )
trigger waittill( "enable" );
if ( isdefined( trigger.script_flag_set ) )
{
if ( isdefined( other.vehicle_flags ) )
other.vehicle_flags[ trigger.script_flag_set ] = true;
other notify( "vehicle_flag_arrived", trigger.script_flag_set );
flag_set( trigger.script_flag_set );
}
if ( isdefined( trigger.script_flag_clear ) )
{
if ( isdefined( other.vehicle_flags ) )
other.vehicle_flags[ trigger.script_flag_clear ] = false;
flag_clear( trigger.script_flag_clear );
}
if( script_vehicledetour )
other thread path_detour_script_origin( trigger );
else if ( detoured && isdefined( other ) )
other thread path_detour( trigger );
if ( isdefined( trigger.script_delay ) )
wait trigger.script_delay;
targs = [];
if( bTriggeronce )
{
if( isdefined( trigger.target ) && isdefined( level.vehicle_target[ trigger.target ] ) )
targs = level.vehicle_target[ trigger.target ];
gotrigger = false;
}
if ( isdefined( trigger.script_vehicleGroupDelete ) )
{
if( !isdefined( level.vehicle_DeleteGroup[ trigger.script_vehicleGroupDelete ] ) )
{
println( "failed to find deleteable vehicle with script_vehicleGroupDelete group number: ", trigger.script_vehicleGroupDelete );
level.vehicle_DeleteGroup[ trigger.script_vehicleGroupDelete ] = [];
}
array_levelthread( level.vehicle_DeleteGroup[ trigger.script_vehicleGroupDelete ], ::deleteEnt );
}
if( script_vehiclespawngroup )
{
level notify( "spawnvehiclegroup" + trigger.script_vehiclespawngroup );
waittillframeend;
}
if ( gates.size > 0 && bTriggeronce )
array_levelthread( gates, ::path_gate_open );
if ( isdefined( trigger.script_VehicleStartMove ) )
{
if ( !isdefined( level.vehicle_StartMoveGroup[ trigger.script_VehicleStartMove ] ) )
{
println( "^3Vehicle start trigger is: ", trigger.script_VehicleStartMove );
return;
}
array_levelthread( level.vehicle_StartMoveGroup[ trigger.script_VehicleStartMove ], ::gopath );
}
}
}
path_detour_get_detourpath( detournode )
{
detourpath = undefined;
for( j = 0; j < level.vehicle_detourpaths[ detournode.script_vehicledetour ].size; j ++ )
{
if( level.vehicle_detourpaths[ detournode.script_vehicledetour ][ j ] != detournode )
if( !islastnode( level.vehicle_detourpaths[ detournode.script_vehicledetour ][ j ] ) )
detourpath = level.vehicle_detourpaths[ detournode.script_vehicledetour ][ j ];
}
return detourpath;
}
path_detour_script_origin( detournode )
{
detourpath = path_detour_get_detourpath( detournode );
if( isdefined( detourpath ) )
self thread vehicle_paths( detourpath );
}
crash_detour_check( detourpath )
{
// long somewhat complex set of conditions on which a vehicle will detour through a crashpath.
return
(
isdefined( detourpath.script_crashtype )
&&
(
isdefined( self.deaddriver )
|| self.health <= 0
|| detourpath.script_crashtype == "forced"
|| ( level.vclogin_vehicles )
)
&&
(
!isdefined( detourpath.derailed )
|| ( isdefined( detourpath.script_crashtype ) && detourpath.script_crashtype == "plane" )
)
);
}
crash_derailed_check( detourpath )
{
return isdefined( detourpath.derailed ) && detourpath.derailed;
}
path_detour( node )
{
detournode = getvehiclenode( node.target, "targetname" );
detourpath = path_detour_get_detourpath( detournode );
// be more aggressive with this maybe?
if( ! isdefined( detourpath ) )
return;
if( node.detoured && !isdefined( detourpath.script_vehicledetourgroup ) )
return;
if( crash_detour_check( detourpath ) )
{
self notify( "crashpath", detourpath );
detourpath.derailed = 1;
self notify( "newpath" );
self setSwitchNode( node, detourpath );
return;
}
else
{
if( crash_derailed_check( detourpath ) )
return; // .derailed crashpaths fail crash check. this keeps other vehicles from following.
// detour paths specific to grouped vehicles. So they can share a lane and detour when they need to be exciting.
if( isdefined( detourpath.script_vehicledetourgroup ) )
{
if( !isdefined( self.script_vehicledetourgroup ) )
return;
if( detourpath.script_vehicledetourgroup != self.script_vehicledetourgroup )
return ;
}
// SCRIPTER_MOD: JesseS (3/14/2008): Took out this buggy script which breaks crashpath variations
// self notify( "newpath" );
// self setswitchnode( detournode, detourpath );
// thread detour_flag( detourpath );
// if ( !islastnode( detournode ) && !( isdefined( node.scriptdetour_persist ) && node.scriptdetour_persist ) )
// node.detoured = 1;
// self.attachedpath = detourpath;
// thread vehicle_paths();
// return;
}
}
detour_flag( detourpath )
{
self endon( "death" );
self.detouringpath = detourpath;
detourpath waittillmatch( "trigger", self );
self.detouringpath = undefined;
}
vehicle_Levelstuff( vehicle, trigger )
{
// associate with links. false
if( isdefined( vehicle.script_linkname ) )
level.vehicle_link = array_2dadd( level.vehicle_link, vehicle.script_linkname, vehicle );
// associate with targets
if( isdefined( vehicle.targetname ) )
level.vehicle_target = array_2dadd( level.vehicle_target, vehicle.targetname, vehicle );
if( isdefined( vehicle.script_VehicleSpawngroup ) )
level.vehicle_SpawnGroup = array_2dadd( level.vehicle_SpawnGroup, vehicle.script_VehicleSpawngroup, vehicle );
if( isdefined( vehicle.script_VehicleStartMove ) )
level.vehicle_StartMoveGroup = array_2dadd( level.vehicle_StartMoveGroup, vehicle.script_VehicleStartMove, vehicle );
if( isdefined( vehicle.script_vehicleGroupDelete ) )
level.vehicle_DeleteGroup = array_2dadd( level.vehicle_DeleteGroup, vehicle.script_vehicleGroupDelete, vehicle );
// thread vehicle_level_unstuff( vehicle );
}
// vehicle_level_unstuff( vehicle )
// {
// targetname = vehicle.targetname;
// vehicle waittill( "death" );
// // dis - associate with links. false
// if ( isdefined( vehicle.script_linkname ) )
// level.vehicle_link[ vehicle.script_linkname ] = array_remove( level.vehicle_link[ vehicle.script_linkname ], vehicle );
// // dis - associate with targets
//
// if ( isdefined( targetname ) )
// level.vehicle_target[ targetname ] = array_remove( level.vehicle_target[ targetname ], vehicle );;
//
// if ( isdefined( vehicle.script_VehicleSpawngroup ) )
// level.vehicle_SpawnGroup[ vehicle.script_VehicleSpawngroup ] = array_remove( level.vehicle_SpawnGroup[ vehicle.script_VehicleSpawngroup ], vehicle );
//
// if ( isdefined( vehicle.script_VehicleStartMove ) )
// level.vehicle_StartMoveGroup[ vehicle.script_VehicleStartMove ] = array_remove( level.vehicle_StartMoveGroup[ vehicle.script_VehicleStartMove ], vehicle );
//
// if ( isdefined( vehicle.script_vehicleGroupDelete ) )
// level.vehicle_DeleteGroup[ vehicle.script_vehicleGroupDelete ] = array_remove( level.vehicle_DeleteGroup[ vehicle.script_vehicleGroupDelete ], vehicle );
// }
spawn_array( spawners )
{
ai = [];
for( i = 0; i < spawners.size; i ++ )
{
spawners[ i ].count = 1;
dronespawn = false;
if( isdefined( spawners[ i ].script_drone ) )
{
dronespawn = true;
spawned = dronespawn( spawners[ i ] );
assert( isdefined( spawned ) );
}
else if( isdefined( spawners[ i ].script_forcespawn ) )
spawned = spawners[ i ] stalingradSpawn();
else
spawned = spawners[ i ] doSpawn();
if ( !dronespawn && !isalive( spawned ) )
continue;
assert( isdefined( spawned ) );
ai[ ai.size ] = spawned;
}
ai = remove_non_riders_from_array( ai );
return ai;
}
remove_non_riders_from_array( ai )
{
living_ai = [];
for( i = 0; i < ai.size; i ++ )
{
if ( !ai_should_be_added( ai[ i ] ) )
continue;
living_ai[ living_ai.size ] = ai[ i ];
}
return living_ai;
}
ai_should_be_added( ai )
{
if( isalive( ai ) )
return true;
if ( !isdefined( ai ) )
return false;
if ( !isdefined( ai.classname ) )
return false;
return ai.classname == "script_model";
}
spawn_group()
{
HasRiders = ( isdefined( self.script_vehicleride ) );
HasWalkers = ( isdefined( self.script_vehiclewalk ) );
if( !( HasRiders || HasWalkers ) )
{
return;
}
spawners = [];
riderspawners = [];
walkerspawners = [];
if( HasRiders )
riderspawners = level.vehicle_RideSpawners[ self.script_vehicleride ];
if( !isdefined( riderspawners ) )
riderspawners = [];
if( HasWalkers )
walkerspawners = level.vehicle_walkspawners[ self.script_vehiclewalk ];
if( !isdefined( walkerspawners ) )
walkerspawners = [];
spawners = array_combine( riderspawners, walkerspawners );
startinvehicles = [];
ai = spawn_array( spawners );
if( HasRiders )
{
if( isdefined( level.vehicle_RideAI[ self.script_vehicleride ] ) )
{
ai = array_combine( ai, level.vehicle_RideAI[ self.script_vehicleride ] );
}
}
if( HasWalkers )
{
if( isdefined( level.vehicle_WalkAI[ self.script_vehiclewalk ] ) )
{
ai = array_combine( ai, level.vehicle_WalkAI[ self.script_vehiclewalk ] );
ai vehicle_rider_walk_setup (self);
}
}
ai = sort_by_startingpos( ai );
for ( i = 0; i < ai.size; i++ )
self thread maps\_vehicle_aianim::guy_enter( ai[ i ], self );
// disabling the array_levelthread because it threads them in reverse. I don't really want to be the one to mess with that right now.
// array_levelthread( ai, maps\_vehicle_aianim::guy_enter, self );
}
sort_by_startingpos( guysarray )
{
firstarray = [];
secondarray = [];
for ( i = 0 ; i < guysarray.size ; i++ )
{
if ( isdefined( guysarray[ i ].script_startingposition ) )
firstarray[ firstarray.size ] = guysarray[ i ];
else
secondarray[ secondarray.size ] = guysarray[ i ];
}
return array_combine( firstarray, secondarray );
}
vehicle_rider_walk_setup( vehicle )
{
if ( !isdefined( self.script_vehiclewalk ) )
return;
if ( isdefined( self.script_followmode ) )
self.FollowMode = self.script_followmode;
else
self.FollowMode = "cover nodes";
// check if the AI should go to a node after walking with the vehicle
if ( !isdefined( self.target ) )
return;
node = getnode( self.target, "targetname" );
if( isdefined( node ) )
self.NodeAftervehicleWalk = node;
}
runtovehicle( guy )
{
guyarray = [];
climbinnode = self.climbnode;
climbinanim = self.climbanim;
closenode = climbinnode[ 0 ];
currentdist = 5000;
thenode = undefined;
for( i = 0; i < climbinnode.size; i ++ )
{
climborg = self gettagorigin( climbinnode[ i ] );
climbang = self gettagangles( climbinnode[ i ] );
org = getstartorigin( climborg, climbang, climbinanim[ i ] );
distance = distance( guy.origin, climborg );
if( distance < currentdist )
{
currentdist = distance;
closenode = climbinnode[ i ];
thenode = i;
}
}
climbang = undefined;
climborg = undefined;
thread runtovehicle_setgoal( guy );
while( !guy.vehicle_goal )
{
climborg = self gettagorigin( climbinnode[ thenode ] );
climbang = self gettagangles( climbinnode[ thenode ] );
org = getStartOrigin( climborg, climbang, climbinanim[ thenode ] );
guy set_forcegoal();
guy setgoalpos( org );
guy.goalradius = 64;
wait .25;
}
guy unset_forcegoal();
if( self getspeedmph() < 1 )
{
guy linkto( self );
guy animscripted( "hopinend", climborg, climbang, climbinanim[ thenode ] );
guy waittillmatch( "hopinend", "end" );
guy_enter_vehicle( guy, self );
}
}
runtovehicle_setgoal( guy )
{
guy.vehicle_goal = false;
self endon( "death" );
guy endon( "death" );
guy waittill( "goal" );
guy.vehicle_goal = true;
}
setup_groundnode_detour( node )
{
realdetournode = getvehiclenode( node.targetname, "target" );
if( !isdefined( realdetournode ) )
return;
realdetournode.detoured = 0;
add_proccess_trigger( realdetournode );
}
add_proccess_trigger( trigger )
{
// TODO: next game. stop trying to make everything a trigger. remove trigger process. I'd do it this game but there is too much complexity in Detour nodes.
// .processedtrigger is a flag that I set to keep a trigger from getting added twice.
if( isdefined( trigger.processed_trigger ) )
return;
level.vehicle_processtriggers[ level.vehicle_processtriggers.size ] = trigger;
trigger.processed_trigger = true;
}
islastnode( node )
{
if( !isdefined( node.target ) )
return true;
if( !isdefined( getvehiclenode( node.target, "targetname" ) ) && !isdefined( get_vehiclenode_any_dynamic( node.target ) ) )
return true;
return false;
}
get_path_getfunc( pathpoint )
{
get_func = ::get_from_vehicle_node;
// get_func is differnt for struct types and script_origin types of paths
if ( is_aircraft() && isdefined( pathpoint.target ) )
{
if ( isdefined( get_from_entity( pathpoint.target ) ) )
get_func = ::get_from_entity;
if ( isdefined( get_from_spawnstruct( pathpoint.target ) ) )
get_func = ::get_from_spawnstruct;
}
return get_func;
}
path_array_setup( pathpoint )
{
get_func = ::get_from_vehicle_node;
// get_func is differnt for struct types and script_origin types of paths
if( is_aircraft() && isdefined( pathpoint.target ) )
{
if( isdefined( get_from_entity( pathpoint.target ) ) )
get_func = ::get_from_entity;
if( isdefined( get_from_spawnstruct( pathpoint.target ) ) )
get_func = ::get_from_spawnstruct;
}
arraycount = 0;
pathpoints = [];
while( isdefined( pathpoint ) )
{
pathpoints[ arraycount ] = pathpoint;
arraycount ++ ;
if( isdefined( pathpoint.target ) )
pathpoint = [[ get_func ]]( pathpoint.target );
else
break;
}
return pathpoints;
}
node_wait( nextpoint, lastpoint )
{
if ( self.attachedpath == nextpoint )
{
waittillframeend;
return;
}
if( !isdefined( self ) )
return;
self setWaitNode( nextpoint );
self waittill( "reached_wait_node" );
}
vehicle_paths( node , baircraftwaitforstart )
{
assertex( isdefined( node ) || isdefined( self.attachedpath ), "vehicle_path() called without a path" );
self notify( "newpath" );
if( !isdefined( baircraftwaitforstart ) )
baircraftwaitforstart = false; // aircrafts emulate startpath() function waiting for a special scripted notify before going
// SCRIPTER_MOD: JesseS (10/1/2007): Took this out for now to fix planes
//if( baircraftwaitforstart )
// self endon( "death" );
// dynamicpaths unique. node isn't defined by info vehicle node calls to this function
if( isdefined( node ) )
self.attachedpath = node;
pathstart = self.attachedpath;
self.currentNode = self.attachedpath;
if( !isdefined( pathstart ) )
return;
self endon( "newpath" );
// this converts chains of whatever types of vehicle nodes and turns them into this array,
// from this point most of the script won't care what kind of path it is.
// pathpoints = path_array_setup( pathpoint );
pathpoint = pathstart;
// dynamic paths / struct path unique
//if( baircraftwaitforstart )
// self waittill( "start_dynamicpath" );
// default type is for vehicle nodes
wait_func = ::node_wait;
lastpoint = undefined;
nextpoint = pathstart;
get_func = get_path_getfunc( pathstart );
// for ( i = 0; i < pathpoints.size; i++ )
while ( isdefined( nextpoint ) )
{
// nextpoint = pathpoints[ i ];
// lastpoint = pathpoints[ i - 1 ];
[[ wait_func ]]( nextpoint, lastpoint );
if( !isdefined( self ) )
return;
self.currentNode = nextpoint;
if( isdefined( nextpoint.gateopen ) && !nextpoint.gateopen )
{
// threaded because vehicle may setspeed( 0, 15 ) and run into the next node
self thread path_gate_wait_till_open( nextpoint );
}
// the sweet stuff! Pathpoints handled in script as triggers!
nextpoint notify( "trigger", self );
// SRS 05/03/07: added for _planeweapons to drop bombs
// amount, delay, delay trace
if( IsDefined( nextpoint.script_dropbombs ) && nextpoint.script_dropbombs > 0 )
{
amount = nextpoint.script_dropbombs;
delay = 0;
delaytrace = 0;
if( IsDefined( nextpoint.script_dropbombs_delay ) && nextpoint.script_dropbombs_delay > 0 )
{
delay = nextpoint.script_dropbombs_delay;
}
if( IsDefined( nextpoint.script_dropbombs_delaytrace ) && nextpoint.script_dropbombs_delaytrace > 0 )
{
delaytrace = nextpoint.script_dropbombs_delaytrace;
}
self notify( "drop_bombs", amount, delay, delaytrace );
}
if ( isdefined( nextpoint.script_volumedown ) )
{
assertex( !isdefined( nextpoint.script_volumedown ), "Tried to volume down while voluming up, or vice versa" );
self thread volume_down( nextpoint.script_volumedown );
}
if ( isdefined( nextpoint.script_volumeup ) )
{
assertex( !isdefined( nextpoint.script_volumeup ), "Tried to volume down while voluming up, or vice versa" );
self thread volume_up( nextpoint.script_volumeup );
}
if ( isdefined( nextpoint.script_noteworthy ) )
{
self notify( nextpoint.script_noteworthy );
self notify( "noteworthy", nextpoint.script_noteworthy );
}
waittillframeend; // this lets other scripts interupt
if( !isdefined( self ) )
return;
if ( isdefined( nextpoint.script_noteworthy ) )
{
if ( nextpoint.script_noteworthy == "godon" )
self godon();
if ( nextpoint.script_noteworthy == "godoff" )
self godoff();
if ( nextpoint.script_noteworthy == "deleteme" )
level thread deleteent( self );
}
if ( isdefined( nextpoint.script_crashtypeoverride ) )
self.script_crashtypeoverride = nextpoint.script_crashtypeoverride;
if ( isdefined( nextpoint.script_badplace ) )
self.script_badplace = nextpoint.script_badplace;
if ( isdefined( nextpoint.script_turretmg ) )
self.script_turretmg = nextpoint.script_turretmg;
if ( isdefined( nextpoint.script_team ) )
self.script_team = nextpoint.script_team;
if ( isdefined( nextpoint.script_turningdir ) )
self notify( "turning", nextpoint.script_turningdir );
if ( isdefined( nextpoint.script_deathroll ) )
if ( nextpoint.script_deathroll == 0 )
self thread deathrolloff();
else
self thread deathrollon();
if ( isdefined( nextpoint.script_vehicleaianim ) )
{
if ( isdefined( nextpoint.script_parameters ) && nextpoint.script_parameters == "queue" )
self.queueanim = true;
if ( isdefined( nextpoint.script_startingposition ) )
self.groupedanim_pos = nextpoint.script_startingposition;
self notify( "groupedanimevent", nextpoint.script_vehicleaianim );
}
if ( isdefined( nextpoint.script_wheeldirection ) )
self wheeldirectionchange( nextpoint.script_wheeldirection );
if ( isdefined( nextpoint.script_exploder ) )
level exploder( nextpoint.script_exploder );
if ( isdefined( nextpoint.script_flag_set ) )
{
if ( isdefined( self.vehicle_flags ) )
self.vehicle_flags[ nextpoint.script_flag_set ] = true;
self notify( "vehicle_flag_arrived", nextpoint.script_flag_set );
flag_set( nextpoint.script_flag_set );
}
if ( isdefined( nextpoint.script_flag_clear ) )
{
if ( isdefined( self.vehicle_flags ) )
self.vehicle_flags[ nextpoint.script_flag_clear ] = false;
flag_clear( nextpoint.script_flag_clear );
}
if ( isdefined( nextpoint.script_unload ) )
self thread unload_node( nextpoint );
if( isdefined( nextpoint.script_flag_wait ) )
{
if ( !isdefined( self.vehicle_flags ) )
{
self.vehicle_flags = [];
}
self.vehicle_flags[ nextpoint.script_flag_wait ] = true;
self notify( "vehicle_flag_arrived", nextpoint.script_flag_wait );
// helicopters stop on their own because they know to stop at destination for script_flag_wait
// may have to provide a smoother way to stop and go tho, this is rather arbitrary, for tanks
// in this case
if ( !flag( nextpoint.script_flag_wait ) )
{
if ( !is_aircraft() )
self setSpeed( 0, 35 );
}
// wait at the end point if it has flag wait
flag_wait( nextpoint.script_flag_wait );
}
if ( isdefined( self.set_lookat_point ) )
{
self.set_lookat_point = undefined;
self clearLookAtEnt();
}
if ( isdefined( nextpoint.script_vehicle_lights_off ) )
self thread lights_off( nextpoint.script_vehicle_lights_off );
if ( isdefined( nextpoint.script_vehicle_lights_on ) )
self thread lights_on( nextpoint.script_vehicle_lights_on );
lastpoint = nextpoint;
if ( !isdefined( nextpoint.target ) )
break;
nextpoint = [[ get_func ]]( nextpoint.target );
}
self notify( "reached_dynamic_path_end" );
if( isdefined( self.script_vehicle_selfremove ) )
self delete();
}
must_stop_at_next_point( nextpoint )
{
// gotta be able to slow down for unload nodes
if( isdefined( nextpoint.script_unload ) )
return true;
// gotta stop if it depends on a flag
return isdefined( nextpoint.script_flag_wait ) && !flag( nextpoint.script_flag_wait );
}
aircraft_wait_node( nextpoint , lastpoint )
{
self endon( "newpath" );
// this handles a single node on aircraft path. they are script_structs in radiant, or script_origins
if( isdefined( nextpoint.script_unload ) && isdefined( self.fastropeoffset ) )
{
nextpoint.radius = 2;
neworg = groundpos( nextpoint.origin ) + ( 0, 0, self.fastropeoffset );
if ( neworg[ 2 ] > nextpoint.origin[ 2 ] - 2000 )
{
// dont descend if it's going to be a huge drop, the designer may intend for it to drop guys behind a wall
// where there is no geo for it to align with
nextpoint.origin = groundpos( nextpoint.origin ) + ( 0, 0, self.fastropeoffset );
}
self sethoverparams( 0, 0, 0 );
}
if( isdefined( lastpoint ) )
{
if( isdefined( lastpoint.speed ) )
{
speed = lastpoint.speed;
accel = 25;
decel = undefined;
if( isdefined( lastpoint.script_decel ) )
{
decel = lastpoint.script_decel;
}
else
{
if( must_stop_at_next_point( nextpoint ) )
{
// decel = speed;
}
}
if( isdefined( lastpoint.script_accel ) )
{
accel = lastpoint.script_accel;
}
else
{
max_accel = speed / 4;
if( accel > max_accel )
{
accel = max_accel;
}
}
if( isdefined( decel ) )
{
self setSpeed( speed, accel, decel );
}
else
{
self setSpeed( speed, accel );
}
}
else
{
if( must_stop_at_next_point( nextpoint ) )
{
// self setSpeed( 60, 15, 60 );
}
}
}
self setvehgoalnode( nextpoint );
if( isdefined( nextpoint.radius ) )
{
self setNearGoalNotifyDist( nextpoint.radius );
assertex( nextpoint.radius > 0, "radius: " + nextpoint.radius );
self waittill_any( "near_goal", "goal" );
}
else
{
self waittill( "goal" );
}
if( isdefined( nextpoint.script_stopnode ) )
{
if( nextpoint.script_stopnode )
self notify( "reached_stop_node" );
if( isdefined( nextpoint.script_delay ) )
wait nextpoint.script_delay;
}
}
helipath( msg, maxspeed, accel )
{
// depreciated
// gets a path from the targetname that is passed
// sets the lookat for the vehicle to ents that are script_linkname'd to the path
self setairresistance( 30 );
self setSpeed( maxspeed, accel, level.heli_default_decel );
vehicle_paths( getstruct( msg, "targetname" ) );
}
setvehgoalnode( node )
{
self endon( "death" );
stop = false;
if( !isdefined( stop ) )
stop = true;
if( isdefined( node.script_stopnode ) ) // z: stop at nodes if there is a script_stopnode = 1 value
stop = node.script_stopnode;
if( isdefined( node.script_unload ) )
stop = true;
script_anglevehicle = isdefined( node.script_anglevehicle ) && node.script_anglevehicle;
script_goalyaw = isdefined( node.script_goalyaw ) && node.script_goalyaw;
if ( isdefined( node.script_anglevehicle ) || isdefined( node.script_goalyaw ) )
self forcetarget( node, script_goalyaw, script_anglevehicle );
else
self unforcetarget();
if( isdefined( node.script_flag_wait ) )
{
if( !flag( node.script_flag_wait ) )
{
// if the flag gets set during flight, we should update the setvehgoalpos to not stop
stop = true;
}
}
if( !isdefined( node.target ) )
{
// stop if this is the end of the path
stop = true;
}
self setvehgoalpos_wrap( node.origin, stop ); // Z: second param = false dont stop at each node.
}
forcetarget( node, script_goalyaw, script_anglevehicle )
{
// [ 14:45 ] [ jiesang - ?? ]: lookat entity > goalyaw > targetyaw
if ( script_goalyaw )
{
self cleartargetyaw();
self setgoalyaw( node.angles[ 1 ] );
}
else
{
self cleargoalyaw(); // clear this thing
self settargetyaw( node.angles[ 1 ] );
}
}
unforcetarget()
{
self cleargoalyaw(); // clear this thing
self cleartargetyaw(); // clear the stuff
}
deathrollon()
{
if( self.health > 0 )
self.rollingdeath = 1;
}
deathrolloff()
{
self.rollingdeath = undefined;
self notify( "deathrolloff" );
}
getonpath()
{
path_start = undefined;
type = self.vehicletype;
if( isdefined( self.target ) )
{
path_start = getvehiclenode( self.target, "targetname" );
if( !isdefined( path_start ) )
{
path_start = getent( self.target, "targetname" );
}
// else if( is_aircraft() ) // MikeD (07/23/07): Aircrafts should be able to go on splines...
// {
// println( "helicopter node targetname: " + path_start.targetname );
// println( "vehicletype: " + self.vehicletype );
// assertmsg( "helicopter on vehicle path( see console for info )" );
// }
if( !isdefined( path_start ) )
{
path_start = getstruct( self.target, "targetname" );
}
}
if( !isdefined( path_start ) )
return;
self.attachedpath = path_start;
self.origin = path_start.origin;
// SCRIPTER_MOD: JesseS (10/1/2007): Added is_aircraft since it doesnt matter anymore
if( !is_aircraft() || is_aircraft())
{
self attachpath( path_start );
}
else
{
if( isdefined( self.speed ) )
{
self setspeedimmediate( self.speed, 20 );
}
else
if( isdefined( path_start.speed ) )
{
self setspeed( path_start.speed, 20, level.heli_default_decel );
}
else
{
// default heli speed
self setspeed( 150, 20, level.heli_default_decel );
}
}
if( !isdefined( self.dontDisconnectPaths ) )
{
self disconnectpaths();
}
self thread vehicle_paths( undefined, is_aircraft() );
}
/*
=============
///ScriptDocBegin
"Name: create_vehicle_from_spawngroup_and_gopath( <spawnGroup> )"
"Summary: spawns and returns and array of the vehicles in the specified spawngroup starting them on their paths"
"Module: Vehicle"
"CallOn: An entity"
"MandatoryArg: <spawnGroup> : the script_vehiclespawngroup asigned to the vehicles in radiant"
"Example: maps\_vehicle::create_vehicle_from_spawngroup_and_gopath( spawnGroup )"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
create_vehicle_from_spawngroup_and_gopath( spawnGroup )
{
vehicleArray = maps\_vehicle::scripted_spawn( spawnGroup );
for( i = 0; i < vehicleArray.size; i ++ )
level thread maps\_vehicle::gopath( vehicleArray[ i ] );
return vehicleArray;
}
gopath( vehicle )
{
if( !isdefined( vehicle ) )
println( "go path called on non existant vehicle" );
if( isdefined( vehicle.script_vehiclestartmove ) )
level.vehicle_StartMoveGroup[ vehicle.script_vehiclestartmove ] = array_remove( level.vehicle_StartMoveGroup[ vehicle.script_vehiclestartmove ], vehicle );
vehicle endon( "death" );
vehicle endon( "stop path" );
if( isdefined( vehicle.hasstarted ) )
{
println( "vehicle already moving when triggered with a startmove" );
return;
}
else
vehicle.hasstarted = true;
if( isdefined( vehicle.script_delay ) )
wait vehicle.script_delay;
vehicle notify( "start_vehiclepath" );
// SCRIPTER_MOD: JesseS (10/1/2007): Took this out for now to fix planes
//if( vehicle is_aircraft() )
// vehicle notify( "start_dynamicpath" );
//else
vehicle startpath();
// next game, remove from here down this isn't the place to do such things.
wait .05;
vehicle connectpaths();
vehicle waittill( "reached_end_node" );
if ( !isdefined( vehicle.dontDisconnectPaths ) )
vehicle disconnectpaths();
if( isdefined( vehicle.currentnode ) && isdefined( vehicle.currentnode.script_noteworthy ) && vehicle.currentnode.script_noteworthy == "deleteme" )
return;
if ( isdefined( vehicle.dontunloadonend ) )
return;
if( isdefined( vehicle.script_unloaddelay ) )
vehicle thread dounload( vehicle.script_unloaddelay );
else
vehicle notify( "unload" );
}
dounload( delay )
{
self endon( "unload" );
if( delay <= 0 )
return;
wait delay;
self notify( "unload" );
}
path_gate_open( node )
{
node.gateopen = true;
node notify( "gate opened" );
}
path_gate_wait_till_open( pathspot )
{
self endon( "death" );
self.waitingforgate = true;
self vehicle_setspeed( 0, 15, "path gate closed" );
pathspot waittill( "gate opened" );
self.waitingforgate = false;
if( self.health > 0 )
script_resumespeed( "gate opened", level.vehicle_ResumeSpeed );
}
spawner_setup( vehicles, message, from )
{
script_vehiclespawngroup = message;
if( !isdefined( level.vehicleSpawners ) )
level.vehicleSpawners = [];
level.vehicleSpawners[ script_vehiclespawngroup ] = [];
vehicle = []; // vehicle is an array of vehiclegroup array's
for( i = 0; i < vehicles.size; i ++ )
{
if( !isdefined( vehicle[ script_vehiclespawngroup ] ) )
vehicle[ script_vehiclespawngroup ] = [];
// SCRIPTER_MOD: JesseS (8/13/200): script_struct instead of script_origin
//script_origin = spawn( "script_origin", vehicles[ i ].origin );
script_struct = spawnstruct();
// creates struct that copies certain values from the vehicle to be added when the vehicle is spawned.
script_struct setspawnervariables( vehicles[ i ], from );
vehicle[ script_vehiclespawngroup ][ vehicle[ script_vehiclespawngroup ].size ] = script_struct;
level.vehicleSpawners[ script_vehiclespawngroup ][ i ] = script_struct;
}
while( 1 )
{
spawnedvehicles = [];
level waittill( "spawnvehiclegroup" + message );
for( i = 0; i < vehicle[ script_vehiclespawngroup ].size; i ++ )
spawnedvehicles[ spawnedvehicles.size ] = vehicle_spawn( vehicle[ script_vehiclespawngroup ][ i ] );
level notify( "vehiclegroup spawned" + message, spawnedvehicles );
}
}
/*
=============
///ScriptDocBegin
"Name: scripted_spawn( <group> )"
"Summary: spawns and returns a vehiclegroup, you will need to tell it to maps\_vehicle::gopath() when you want it to go"
"Module: Vehicle"
"CallOn: An entity"
"MandatoryArg: <group> : "
"Example: bmps = maps\_vehicle::scripted_spawn( 32 );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
scripted_spawn( group )
{
thread scripted_spawn_go( group );
level waittill( "vehiclegroup spawned" + group, vehicles );
return vehicles;
}
scripted_spawn_go( group )
{
waittillframeend;
level notify( "spawnvehiclegroup" + group );
}
setspawnervariables( vehicle, from )
{
// self.spawnerclassname = vehicle.classname;
self.spawnermodel = vehicle.model;
self.angles = vehicle.angles;
self.origin = vehicle.origin;
if( isdefined( vehicle.script_delay ) )
self.script_delay = vehicle.script_delay;
if( isdefined( vehicle.script_noteworthy ) )
self.script_noteworthy = vehicle.script_noteworthy;
if( isdefined( vehicle.script_parameters ) )
self.script_parameters = vehicle.script_parameters;
if( isdefined( vehicle.script_team ) )
self.script_team = vehicle.script_team;
if( isdefined( vehicle.script_vehicleride ) )
self.script_vehicleride = vehicle.script_vehicleride;
if( isdefined( vehicle.target ) )
self.target = vehicle.target;
if( isdefined( vehicle.targetname ) )
self.targetname = vehicle.targetname;
else
self.targetname = "notdefined";
self.spawnedtargetname = self.targetname;
self.targetname = self.targetname + "_vehiclespawner";
if( isdefined( vehicle.triggeredthink ) )
self.triggeredthink = vehicle.triggeredthink;
if( isdefined( vehicle.script_sound ) )
self.script_sound = vehicle.script_sound;
if( isdefined( vehicle.script_turretmg ) )
self.script_turretmg = vehicle.script_turretmg;
if( isdefined( vehicle.script_startinghealth ) )
self.script_startinghealth = vehicle.script_startinghealth;
if( isdefined( vehicle.spawnerNum ) )
self.spawnerNum = vehicle.spawnerNum;
if( isdefined( vehicle.script_deathnotify ) )
self.script_deathnotify = vehicle.script_deathnotify;
if( isdefined( vehicle.script_turret ) )
self.script_turret = vehicle.script_turret;
if( isdefined( vehicle.script_linkTo ) )
self.script_linkTo = vehicle.script_linkTo;
if( isdefined( vehicle.script_VehicleSpawngroup ) )
self.script_VehicleSpawngroup = vehicle.script_VehicleSpawngroup;
if( isdefined( vehicle.script_VehicleStartMove ) )
self.script_VehicleStartMove = vehicle.script_VehicleStartMove;
if( isdefined( vehicle.script_vehicleGroupDelete ) )
self.script_vehicleGroupDelete = vehicle.script_vehicleGroupDelete;
if( isdefined( vehicle.script_vehicle_selfremove ) )
self.script_vehicle_selfremove = vehicle.script_vehicle_selfremove;
if( isdefined( vehicle.script_nomg ) )
self.script_nomg = vehicle.script_nomg;
if( isdefined( vehicle.script_badplace ) )
self.script_badplace = vehicle.script_badplace;
if( isdefined( vehicle.script_vehicleride ) )
self.script_vehicleride = vehicle.script_vehicleride;
if( isdefined( vehicle.script_vehiclewalk ) )
self.script_vehiclewalk = vehicle.script_vehiclewalk;
if( isdefined( vehicle.script_linkName ) )
self.script_linkName = vehicle.script_linkName;
if( isdefined( vehicle.script_crashtypeoverride ) )
self.script_crashtypeoverride = vehicle.script_crashtypeoverride;
if( isdefined( vehicle.script_unloaddelay ) )
self.script_unloaddelay = vehicle.script_unloaddelay;
if( isdefined( vehicle.script_unloadmgguy ) )
self.script_unloadmgguy = vehicle.script_unloadmgguy;
if( isdefined( vehicle.script_keepdriver ) )
self.script_keepdriver = vehicle.script_keepdriver;
if( isdefined( vehicle.script_fireondrones ) )
self.script_fireondrones = vehicle.script_fireondrones;
if( isdefined( vehicle.script_tankgroup ) )
self.script_tankgroup = vehicle.script_tankgroup;
if( isdefined( vehicle.script_avoidplayer ) )
self.script_avoidplayer = vehicle.script_avoidplayer;
if( isdefined( vehicle.script_playerconeradius ) )
self.script_playerconeradius = vehicle.script_playerconeradius;
if( isdefined( vehicle.script_cobratarget ) )
self.script_cobratarget = vehicle.script_cobratarget;
if( isdefined( vehicle.script_targettype ) )
self.script_targettype = vehicle.script_targettype;
if( isdefined( vehicle.script_targetoffset_z ) )
self.script_targetoffset_z = vehicle.script_targetoffset_z;
if( isdefined( vehicle.script_wingman ) )
self.script_wingman = vehicle.script_wingman;
if( isdefined( vehicle.script_mg_angle ) )
self.script_mg_angle = vehicle.script_mg_angle;
if( isdefined( vehicle.script_physicsjolt ) )
self.script_physicsjolt = vehicle.script_physicsjolt;
if( isdefined( vehicle.script_vehicle_lights_on ) )
self.script_vehicle_lights_on = vehicle.script_vehicle_lights_on;
if( isdefined( vehicle.script_light_toggle ) )
self.script_light_toggle = vehicle.script_light_toggle;
if( isdefined( vehicle.script_vehicledetourgroup ) )
self.script_vehicledetourgroup = vehicle.script_vehicledetourgroup;
if( isdefined( vehicle.speed ) )
self.speed = vehicle.speed;
if( isdefined( vehicle.script_vehicletriggergroup ) )
self.script_vehicletriggergroup = vehicle.script_vehicletriggergroup;
if( isdefined( vehicle.script_cheap ) )
self.script_cheap = vehicle.script_cheap;
if ( isdefined( vehicle.script_volume ) )
self.script_volume = vehicle.script_volume;
// SCRIPTER_MOD: JesseS (5/7/07) - script_flak88 added
if( isdefined( vehicle.script_flak88 ) )
self.script_flak88 = vehicle.script_flak88;
// SCRIPTER_MOD: JesseS (7/21/2007): added script_nonmovingvehicle to eventually replace script_flak88
if( isdefined( vehicle.script_nonmovingvehicle ) )
self.script_nonmovingvehicle = vehicle.script_nonmovingvehicle;
if( isdefined( vehicle.script_flag ) )
self.script_flag = vehicle.script_flag;
if ( isdefined( vehicle.script_disconnectpaths ) )
self.script_disconnectpaths = vehicle.script_disconnectpaths;
if ( isdefined( vehicle.script_bulletshield ) )
self.script_bulletshield = vehicle.script_bulletshield;
if ( isdefined( vehicle.script_volumeramp ) )
self.script_volumeramp = vehicle.script_volumeramp;
if ( isdefined( vehicle.script_godmode ) )
self.script_godmode = vehicle.script_godmode;
//// SCRIPTER_MOD: JesseS (5/12/200) - script_vehicleattackgroup added back in
if (isdefined (vehicle.script_vehicleattackgroup))
{
self.script_vehicleattackgroup = vehicle.script_vehicleattackgroup;
}
if (isdefined (vehicle.script_vehicleattackgroupwait))
{
self.script_vehicleattackgroupwait = vehicle.script_vehicleattackgroupwait;
}
// guzzo - 4/1/08 - made it so script_friendname carries over to the spawned vehicle from the spawner
if (isdefined (vehicle.script_friendname))
{
self.script_friendname = vehicle.script_friendname;
}
if( vehicle.count > 0 )
self.count = vehicle.count;
else
self.count = 1;
if( isdefined( vehicle.vehicletype ) )
self.vehicletype = vehicle.vehicletype;
else
self.vehicletype = maps\_vehicletypes::get_type( vehicle.model );
// SRS 05/03/07: added to support _planeweapons dropping bombs
if( IsDefined( vehicle.script_numbombs ) )
{
self.script_numbombs = vehicle.script_numbombs;
}
vehicle delete();
id = vehicle_spawnidgenerate( self.origin );
self.spawner_id = id;
level.vehicle_spawners[ id ] = self;
}
vehicle_spawnidgenerate( origin )
{
return "spawnid" + int( origin[ 0 ] ) + "a" + int( origin[ 1 ] ) + "a" + int( origin[ 2 ] );
}
// CODER_MOD: Tommy K
vehicleDamageAssist()
{
self endon( "death" );
self.attackers = [];
self.attackerData = [];
while( true )
{
self waittill( "damage", amount, attacker );
if( !isDefined ( attacker ) || !isPlayer(attacker) )
{
continue;
}
if ( !isdefined( self.attackerData[ attacker getEntityNumber() ] ) )
{
self.attackers[ self.attackers.size ] = attacker;
self.attackerData[ attacker getEntityNumber() ] = false;
}
}
}
vehicle_spawn( vspawner, from )
{
if( !vspawner.count )
return;
vehicle = spawnVehicle( vspawner.spawnermodel, vspawner.spawnedtargetname, vspawner.vehicletype, vspawner.origin, vspawner.angles );
if( isdefined( vspawner.script_delay ) )
vehicle.script_delay = vspawner.script_delay;
if( isdefined( vspawner.script_noteworthy ) )
vehicle.script_noteworthy = vspawner.script_noteworthy;
if( isdefined( vspawner.script_parameters ) )
vehicle.script_parameters = vspawner.script_parameters;
if( isdefined( vspawner.script_team ) )
vehicle.script_team = vspawner.script_team;
if( isdefined( vspawner.script_vehicleride ) )
vehicle.script_vehicleride = vspawner.script_vehicleride;
if( isdefined( vspawner.target ) )
vehicle.target = vspawner.target;
if( isdefined( vspawner.vehicletype ) )
vehicle.vehicletype = vspawner.vehicletype;
if( isdefined( vspawner.triggeredthink ) )
vehicle.triggeredthink = vspawner.triggeredthink;
if( isdefined( vspawner.script_sound ) )
vehicle.script_sound = vspawner.script_sound;
if( isdefined( vspawner.script_turretmg ) )
vehicle.script_turretmg = vspawner.script_turretmg;
if( isdefined( vspawner.script_startinghealth ) )
vehicle.script_startinghealth = vspawner.script_startinghealth;
if( isdefined( vspawner.script_deathnotify ) )
vehicle.script_deathnotify = vspawner.script_deathnotify;
if( isdefined( vspawner.script_turret ) )
vehicle.script_turret = vspawner.script_turret;
if( isdefined( vspawner.script_linkTo ) )
vehicle.script_linkTo = vspawner.script_linkTo;
if( isdefined( vspawner.script_VehicleSpawngroup ) )
vehicle.script_VehicleSpawngroup = vspawner.script_VehicleSpawngroup;
if( isdefined( vspawner.script_VehicleStartMove ) )
vehicle.script_VehicleStartMove = vspawner.script_VehicleStartMove;
if( isdefined( vspawner.script_vehicleGroupDelete ) )
vehicle.script_vehicleGroupDelete = vspawner.script_vehicleGroupDelete;
if( isdefined( vspawner.script_vehicle_selfremove ) )
vehicle.script_vehicle_selfremove = vspawner.script_vehicle_selfremove;
if( isdefined( vspawner.script_nomg ) )
vehicle.script_nomg = vspawner.script_nomg;
if( isdefined( vspawner.script_badplace ) )
vehicle.script_badplace = vspawner.script_badplace;
if( isdefined( vspawner.script_vehicleride ) )
vehicle.script_vehicleride = vspawner.script_vehicleride;
if( isdefined( vspawner.script_vehiclewalk ) )
vehicle.script_vehiclewalk = vspawner.script_vehiclewalk;
if( isdefined( vspawner.script_linkName ) )
vehicle.script_linkName = vspawner.script_linkName;
if( isdefined( vspawner.script_crashtypeoverride ) )
vehicle.script_crashtypeoverride = vspawner.script_crashtypeoverride;
if( isdefined( vspawner.script_unloaddelay ) )
vehicle.script_unloaddelay = vspawner.script_unloaddelay;
if( isdefined( vspawner.script_unloadmgguy ) )
vehicle.script_unloadmgguy = vspawner.script_unloadmgguy;
if( isdefined( vspawner.script_keepdriver ) )
vehicle.script_keepdriver = vspawner.script_keepdriver;
if( isdefined( vspawner.script_fireondrones ) )
vehicle.script_fireondrones = vspawner.script_fireondrones;
if( isdefined( vspawner.script_tankgroup ) )
vehicle.script_tankgroup = vspawner.script_tankgroup;
if( isdefined( vspawner.script_avoidplayer ) )
vehicle.script_avoidplayer = vspawner.script_avoidplayer;
if( isdefined( vspawner.script_playerconeradius ) )
vehicle.script_playerconeradius = vspawner.script_playerconeradius;
if( isdefined( vspawner.script_cobratarget ) )
vehicle.script_cobratarget = vspawner.script_cobratarget;
if( isdefined( vspawner.script_targettype ) )
vehicle.script_targettype = vspawner.script_targettype;
if( isdefined( vspawner.script_targetoffset_z ) )
vehicle.script_targetoffset_z = vspawner.script_targetoffset_z;
if( isdefined( vspawner.script_wingman ) )
vehicle.script_wingman = vspawner.script_wingman;
if( isdefined( vspawner.script_mg_angle ) )
vehicle.script_mg_angle = vspawner.script_mg_angle;
if( isdefined( vspawner.script_physicsjolt ) )
vehicle.script_physicsjolt = vspawner.script_physicsjolt;
if( isdefined( vspawner.script_cheap ) )
vehicle.script_cheap = vspawner.script_cheap;
if ( isdefined( vspawner.script_flag ) )
vehicle.script_flag = vspawner.script_flag;
if( isdefined( vspawner.script_vehicle_lights_on ) )
vehicle.script_vehicle_lights_on = vspawner.script_vehicle_lights_on;
if( isdefined( vspawner.script_vehicledetourgroup ) )
vehicle.script_vehicledetourgroup = vspawner.script_vehicledetourgroup;
if( isdefined( vspawner.speed ) )
vehicle.speed = vspawner.speed;
if ( isdefined( vspawner.script_volume ) )
vehicle.script_volume = vspawner.script_volume;
if( isdefined( vspawner.script_light_toggle ) )
vehicle.script_light_toggle = vspawner.script_light_toggle;
if( isdefined( vspawner.spawner_id ) )
vehicle.spawner_id = vspawner.spawner_id;
if( isdefined( vspawner.script_vehicletriggergroup ) )
vehicle.script_vehicletriggergroup = vspawner.script_vehicletriggergroup;
if ( isdefined( vspawner.script_disconnectpaths ) )
vehicle.script_disconnectpaths = vspawner.script_disconnectpaths;
if ( isdefined( vspawner.script_godmode ) )
vehicle.godmode = vspawner.script_godmode;
if ( isdefined( vspawner.script_bulletshield ) )
vehicle.script_bulletshield = vspawner.script_bulletshield;
if ( isdefined( vspawner.script_volumeramp ) )
vehicle.script_volumeramp = vspawner.script_volumeramp;
// SRS 05/03/07: added to support _planeweapons dropping bombs
if( IsDefined( vspawner.script_numbombs ) )
{
vehicle.script_numbombs = vspawner.script_numbombs;
}
// JesseS (05/08/07): script_flak88 added
if( isdefined( vspawner.script_flak88 ) )
{
vehicle.script_flak88 = vspawner.script_flak88;
}
if( isdefined( vspawner.script_flag ) )
{
vehicle.script_flag = vspawner.script_flag;
}
// SCRIPTER_MOD: JesseS (7/21/2007): script_nonmovingvehicle added to eventually replace script_flak88
if( isdefined( vspawner.script_nonmovingvehicle ) )
{
vehicle.script_nonmovingvehicle = vspawner.script_nonmovingvehicle;
}
// SCRIPTER_MOD: JesseS (5/12/200) - script_vehicleattackgroup readded
if( isdefined( vspawner.script_vehicleattackgroup ) )
{
vehicle.script_vehicleattackgroup = vspawner.script_vehicleattackgroup;
}
if( isdefined( vspawner.script_vehicleattackgroupwait ) )
{
vehicle.script_vehicleattackgroupwait = vspawner.script_vehicleattackgroupwait;
}
// guzzo - 4/1/08 - made it so script_friendname carries over to the spawned vehicle from the spawner
if (isdefined (vspawner.script_friendname))
{
vehicle.script_friendname = vspawner.script_friendname;
}
// if( initVehicle )
vehicle_init( vehicle );
if( isdefined( vehicle.targetname ) )
level notify( "new_vehicle_spawned" + vehicle.targetname, vehicle );
if( isdefined( vehicle.script_noteworthy ) )
level notify( "new_vehicle_spawned" + vehicle.script_noteworthy, vehicle );
if ( isdefined( vehicle.spawner_id ) )
level notify( "new_vehicle_spawned" + vehicle.spawner_id, vehicle );
// CODER_MOD: Tommy K
if ( vehicle.script_team == "axis" && isdefined( level.vehicle_enemy_tanks[ vspawner.spawnermodel ] ) )
{
vehicle thread vehicleDamageAssist();
}
return vehicle;
}
waittill_vehiclespawn( targetname )
{
level waittill( "new_vehicle_spawned" + targetname, vehicle );
return vehicle;
}
waittill_vehiclespawn_noteworthy( noteworthy )
{
level waittill( "new_vehicle_spawned" + noteworthy, vehicle );
return vehicle;
}
/*
=============
///ScriptDocBegin
"Name: waittill_vehiclespawn_spawner_id( <spawner_id> )"
"Summary: Waits for a specific vehicle to spawn with the specified < spawner id > "
"Module: Vehicle"
"MandatoryArg: <spawner_id> : id of the vehicle to spawn. This value can be found in the array level.vehicle_spawners in any level with spawned vehicles."
"Example: eVehicle = waittill_vehiclespawn_spawner_id( self.spawner_id );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
waittill_vehiclespawn_spawner_id( spawner_id )
{
level waittill( "new_vehicle_spawned" + spawner_id, vehicle );
return vehicle;
}
wait_vehiclespawn( targetname )
{
println( "wait_vehiclespawn() called; change to waittill_vehiclespawn()" );
level waittill( "new_vehicle_spawned" + targetname, vehicle );
return vehicle;
}
spawn_through( vehicle )
{
struct = spawnstruct();
struct setspawnervariables( vehicle );
return vehicle_spawn( struct );
}
vehicle_init( vehicle )
{
if( vehicle.classname == "script_model" )
{
vehicle = spawn_through( vehicle ); // this sends non - spawned script models through the spawner process to get vehicleinfo.
return;
}
if( vehicle.vehicletype == "bog_mortar" )
return;
if( ( isdefined( vehicle.script_noteworthy ) ) && ( vehicle.script_noteworthy == "playervehicle" ) )
return; // TODO: I really don't think we should branch off the players vehicle so early. - nate
// TODO: These shouldn't be asigned to everyvehicle
// vehicle.tanksquadfollow = false;
vehicle.zerospeed = true;
if( !isdefined( vehicle.modeldummyon ) )
vehicle.modeldummyon = false;
type = vehicle.vehicletype;
// give the vehicle health
vehicle vehicle_life();
// set the script_team value used everywhere to determine which team the vehicle belongs to
vehicle vehicle_setteam();
// init pointer is specified in the precache script( IE maps\_tiger::main() )
// only special case gag works should exist in this thread,
// I'm currently working to reduce redundancy by moving common init functions here
if( !isdefined( level.vehicleInitThread[ vehicle.vehicletype ][ vehicle.model ] ) )
{
println( "vehicle.vehicletype is: " + vehicle.vehicletype );
println( "vehicle.model is: " + vehicle.model );
}
vehicle thread [[ level.vehicleInitThread[ vehicle.vehicletype ][ vehicle.model ] ]]();
vehicle thread maingun_FX();
// Guzzo 5-19-08 This should be off now; moved to clientscripts
//vehicle thread playTankExhaust();
// if( !isdefined( vehicle.script_badplace ) )
// vehicle.script_badplace = true;
if( !isdefined( vehicle.script_avoidplayer ) )
vehicle.script_avoidplayer = false;
vehicle.riders = [];
vehicle.unloadque = []; // for ai. wait till a vehicle is unloaded all the way
vehicle.unload_group = "default";
// getoutrig means fastrope.
vehicle.getoutrig = [];
if( isdefined( level.vehicle_attachedmodels ) && isdefined( level.vehicle_attachedmodels[ type ] ) )
{
rigs = level.vehicle_attachedmodels[ type ];
strings = getarraykeys( rigs );
for( i = 0; i < strings.size; i ++ )
{
vehicle.getoutrig[ strings[ i ] ] = undefined;
vehicle.getoutriganimating[ strings[ i ] ] = false;
}
}
// make ai run way from vehicle
vehicle thread vehicle_badplace();
// toggle vehicle lights on / off
if( isdefined( vehicle.script_vehicle_lights_on ) )
vehicle thread lights_on( vehicle.script_vehicle_lights_on );
// regenerate friendly fire damage
if( !vehicle isCheap() )
vehicle thread friendlyfire_shield();
// handles guys riding and doing stuff on vehicles
vehicle thread maps\_vehicle_aianim::handle_attached_guys();
// special stuff for unloading
if( !vehicle isCheap() )
vehicle thread vehicle_handleunloadevent();
// Make the main turret think
vehicle thread turret_attack_think();
// Shellshock player on main turret fire.
if( !vehicle isCheap() )
vehicle thread vehicle_weapon_fired(); // moved to indiviual tank scripts.
// make the vehicle rumble
vehicle thread vehicle_rumble();
// make vehicle shake physics objects.
if( isdefined( vehicle.script_physicsjolt ) && vehicle.script_physicsjolt )
vehicle thread physicsjolt_proximity();
// CODER_MOD: 5/7/2008 - JamesS - This is now setup in the client scripts
// handle tread effects
//vehicle thread vehicle_treads();
// associate vehicle with living level variables.
vehicle_Levelstuff( vehicle );
if( isdefined( vehicle.script_team ) )
vehicle setvehicleteam( vehicle.script_team );
// handle the compassicon for vehicles
vehicle thread vehicle_compasshandle();
// make the wheels rotate
vehicle thread animate_drive_idle();
// handle machine guns
if( !vehicle isCheap() )
vehicle thread mginit();
if( isdefined( level.vehicleSpawnCallbackThread ) )
level thread [[ level.vehicleSpawnCallbackThread ]]( vehicle );
// this got kind of ugly and hackery but it's how I deal with player driveable vehicles in decoytown, elalamein, 88ridge and libya
if ( isdefined( vehicle.spawnflags ) && vehicle.spawnflags & 1 )
{
startinvehicle = ( isdefined( vehicle.script_noteworthy ) && vehicle.script_noteworthy == "startinside" ); // can't see making a whole new keys.txt entry for something that's only going to be used once in any given level.
vehicle maps\_vehicledrive::setup_vehicle_other();
vehicle thread maps\_vehicledrive::vehicle_wait( startinvehicle );
vehicle thread kill();
return;
}
// every vehicle that stops will disconnect its paths
if( !vehicle isCheap() )
vehicle thread disconnect_paths_whenstopped();
// squad behavior
// commenting out, nobody uses this
// if( !vehicle isCheap() )
// vehicle thread squad();
// get on path and start the path handler thread
// SCRIPTER_MOD: JesseS (5/9/07): Shouldnt need this for stationary guns
if (!isdefined (vehicle.script_flak88) && !isdefined (vehicle.script_nonmovingvehicle))
{
vehicle thread getonpath();
}
if (isdefined (vehicle.script_vehicleattackgroup))
{
vehicle thread attackgroup_think();
}
// helicopters do dust kickup fx
if( vehicle hasHelicopterDustKickup() )
{
if(!level.clientscripts)
{
vehicle thread aircraft_dust_kickup();
}
}
/#
// MikeD (8/1/2007): Debug info for vehicles
vehicle thread debug_vehicle();
#/
// spawn the vehicle and it's associated ai
vehicle spawn_group();
vehicle thread kill();
if ( !isdefined( vehicle.script_volume ) )
vehicle thread init_ramp_volume();
vehicle apply_truckjunk();
}
init_ramp_volume()
{
time = 2;
if ( self is_aircraft() )
time = 1;
if ( isdefined( self.script_volumeramp ) )
time = self.script_volumeramp;
// thread lerp_enginesound( time, 0, 1 );
self volume_up( time );
}
kill_damage( type )
{
if( !isdefined( level.vehicle_death_radiusdamage ) || !isdefined( level.vehicle_death_radiusdamage[ type ] ) )
return;
if( isdefined( self.deathdamage_max ) )
maxdamage = self.deathdamage_max;
else
maxdamage = level.vehicle_death_radiusdamage[ type ].maxdamage;
if( isdefined( self.deathdamage_min ) )
mindamage = self.deathdamage_min;
else
mindamage = level.vehicle_death_radiusdamage[ type ].mindamage;
if ( isdefined( level.vehicle_death_radiusdamage[ type ].delay ) )
wait level.vehicle_death_radiusdamage[ type ].delay;
if( !isdefined( self ) )
return; //deleted in this time.
// SCRIPTER_MOD: JesseS (8/10/2007): removed level.player
if( level.vehicle_death_radiusdamage[ type ].bKillplayer )
{
players = get_players();
for (i = 0; i < players.size; i++)
{
players[i] enableHealthShield( false );
}
}
radiusDamage( self.origin + level.vehicle_death_radiusdamage[ type ].offset, level.vehicle_death_radiusdamage[ type ].range, maxdamage, mindamage );
// SCRIPTER_MOD: JesseS (8/10/2007): removed level.player
if( level.vehicle_death_radiusdamage[ type ].bKillplayer )
{
players = get_players();
for (i = 0; i < players.size; i++)
{
players[i] enableHealthShield( true );
}
}
}
vehicle_is_tank()
{
return self.vehicletype == "sherman"
|| self.vehicletype == "panzer4"
|| self.vehicletype == "type97"
|| self.vehicletype == "t34"
|| self.vehicletype == "see2_panzeriv"
|| self.vehicletype == "see2_tiger"
|| self.vehicletype == "see2_ot34"
|| self.vehicletype == "see2_t34"
|| self.vehicletype == "see2_panther"
|| self.vehicletype == "tiger"
|| self.vehicletype == "see2_physics_t34";
}
kill()
{
self endon( "nodeath_thread" );
type = self.vehicletype;
model = self.model;
targetname = self.targetname;
attacker = undefined;
arcadeModePointsRewarded = false;
while ( 1 )
{
// waittill death twice. in some cases the vehicle dies and does a bunch of stuff. then it gets deleted. which it then needs to do more stuff
if ( isdefined( self ) )
{
self waittill( "death", attacker );
}
// CODER MOD: TOMMY K
if( arcadeMode() && false == arcadeModePointsRewarded )
{
if( self.script_team != "allies" )
{
if ( isdefined( self.attackers ) )
{
for ( i = 0; i < self.attackers.size; i++ )
{
player = self.attackers[i];
if ( !isDefined( player ) )
continue;
if ( player == attacker )
continue;
maps\_challenges_coop::doMissionCallback( "playerAssist", player );
player.assists++;
arcademode_assignpoints( "arcademode_score_tankassist", player );
}
self.attackers = [];
self.attackerData = [];
}
if( IsPlayer( attacker ) )
{
//ye killed a enemy tank
arcademode_assignpoints( "arcademode_score_tank", attacker );
// increment the player kills for tank kill
attacker.kills++;
}
}
else
{ //u killed a friendly tank
arcademode_assignpoints( "arcademode_score_tank_friendly", attacker );
}
arcadeModePointsRewarded = true;
}
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -
// some tank and turret cleanup
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -
// MIKED TODO: Do we need this since it's on the clientside??
if( isdefined( self.rumbletrigger ) )
{
self.rumbletrigger delete();
}
if( isdefined( self.mgturret ) )
{
array_levelthread( self.mgturret, ::turret_deleteme );
self.mgturret = undefined;
}
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -
if( isdefined( self.script_team ) )
{
level.vehicles[ self.script_team ] = array_remove( level.vehicles[ self.script_team ], self );
}
if ( isdefined( self.script_linkname ) )
{
level.vehicle_link[ self.script_linkname ] = array_remove( level.vehicle_link[ self.script_linkname ], self );
}
if( isdefined( targetname ) )
{
level.vehicle_target[ targetname ] = array_remove( level.vehicle_target[ targetname ], self );
}
if( isdefined( self.script_VehicleSpawngroup ) )
{
level.vehicle_SpawnGroup[ self.script_VehicleSpawngroup ] = array_remove( level.vehicle_SpawnGroup[ self.script_VehicleSpawngroup ], self );
}
if( isdefined( self.script_VehicleStartMove ) )
{
level.vehicle_StartMoveGroup[ self.script_VehicleStartMove ] = array_remove( level.vehicle_StartMoveGroup[ self.script_VehicleStartMove ], self );
}
if( isdefined( self.script_vehicleGroupDelete ) )
{
level.vehicle_DeleteGroup[ self.script_vehicleGroupDelete ] = array_remove( level.vehicle_DeleteGroup[ self.script_vehicleGroupDelete ], self );
}
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -
thread lights_kill();// only kill the lights once
// if vehicle is gone then delete the ai here.
if ( !isdefined( self ) || is_corpse() )
{
if ( isdefined( self.riders ) )
{
for ( j = 0; j < self.riders.size; j++ )
{
if ( isdefined( self.riders[ j ] ) )
{
self.riders[ j ] delete();
}
}
}
if ( is_corpse() )
{
self.riders = [];
continue;
}
self notify( "delete_destructible" );// kills some destructible fxs
return;
}
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -
if( isdefined( level.vehicle_rumble[ type ] ) )
{
self StopRumble( level.vehicle_rumble[ type ].rumble );
}
if( isdefined( level.vehicle_death_thread[ type ] ) )
{
thread [[ level.vehicle_death_thread[ type ] ]]();
}
if( !IsDefined( self.destructibledef ) )
{
if( isdefined( level.vehicle_death_earthquake[ type ] ) )
{
earthquake(
level.vehicle_death_earthquake[ type ].scale,
level.vehicle_death_earthquake[ type ].duration,
self.origin,
level.vehicle_death_earthquake[ type ].radius
);
}
thread kill_damage( type );
if( isdefined( level.vehicle_deathmodel[ model ] ) )
{
self thread set_death_model( level.vehicle_deathmodel[ model ], level.vehicle_deathmodel_delay[ model ] );
}
// Do kill_fx if it has not been mantled
if( (!IsDefined( self.mantled ) || !self.mantled ) && !IsDefined(self.nodeathfx) )
{
thread kill_fx( model );
}
if( isdefined( self.delete_on_death ) )
{
wait( 0.05 );
if ( !isdefined( self ) )
continue;
if( !isdefined( self.dontDisconnectPaths ) )
{
self disconnectpaths();
}
self freevehicle();
wait 0.05;
if ( !isdefined( self ) )
continue;
self notify( "death_finished" );
self delete();
continue;
}
}
// makes riders blow up
maps\_vehicle_aianim::blowup_riders();
thread kill_badplace( type );
// I used this for identifying when the players tank kills stuff in libya for dialog
if( isdefined( level.vehicle_deathnotify ) && isdefined( level.vehicle_deathnotify[ self.vehicletype ] ) )
{
level notify( level.vehicle_deathnotify[ self.vehicletype ], attacker );
}
// all the vehicles get the same jolt..
if ( self.classname == "script_vehicle" )
{
self thread kill_jolt( type );
}
if( !IsDefined( self.destructibledef ) )
{
if( isdefined( self.script_crashtypeoverride ) )
{
crashtype = self.script_crashtypeoverride;
}
else if( self is_aircraft() )
{
crashtype = "aircraft";
}
else if( isdefined( self.currentnode ) && crash_path_check( self.currentnode ) )
{
crashtype = "none";
}
else
{
crashtype = "tank"; // tanks used to be the only vehicle that would stop. legacy nonsense from CoD1
}
if( crashtype == "aircraft" )
{
self thread aircraft_crash();
}
if( crashtype == "tank" )
{
if( !isdefined( self.rollingdeath ) )
{
self vehicle_setspeed( 0, 25, "Dead" );
}
else
{
// dpg 3/12/08 removed this. if you need it for something, let someone know
//self vehicle_setspeed( 8, 25, "Dead rolling out of path intersection" );
self waittill( "deathrolloff" );
self vehicle_setspeed( 0, 25, "Dead, finished path intersection" );
}
wait .4;
self vehicle_setspeed( 0, 10000, "deadstop" );
self notify( "deadstop" );
if ( !isdefined( self.dontDisconnectPaths ) )
{
self disconnectpaths();
}
if( ( isdefined( self.tankgetout ) ) && ( self.tankgetout > 0 ) )
{
self waittill( "animsdone" ); // tankgetout will never get notified if there are no guys getting out
}
}
}
if( isdefined( level.vehicle_hasMainTurret[ model ] ) && level.vehicle_hasMainTurret[ model ] )
{
self clearTurretTarget();
}
if( self is_aircraft() || self is_boat())
{
if( ( isdefined( self.crashing ) ) && ( self.crashing == true ) )
{
self waittill( "crash_done" );
}
}
else
{
while( isdefined( self ) && self getspeedmph() > 0 )
{
wait( 0.1 );
}
}
self notify( "stop_looping_death_fx" );
self notify( "death_finished" );
wait( 0.5 );
if( is_corpse() )
{
continue;
}
if( isdefined( self ) )
{
while( isdefined( self.dontfreeme ) && isdefined(self) )
{
wait( .05 );
}
if( !isdefined( self ) )
{
continue;
}
self freevehicle();
if ( self.modeldummyon )
{
self hide();
}
}
if( ( isdefined( self.crashing ) ) && ( self.crashing == true ) )
{
self delete();
continue;
}
}
}
is_corpse()
{
is_corpse = false;
if ( isdefined( self ) && self.classname == "script_vehicle_corpse" )
is_corpse = true;
return is_corpse;
}
set_death_model( sModel, fDelay )
{
assert( isdefined( sModel ) );
if( isdefined( fDelay ) && ( fDelay > 0 ) )
wait fDelay;
if(!isdefined(self))
return;
eModel = get_dummy();
if( isdefined( self.clear_anims_on_death ) )
eModel clearanim( %root, 0 );
if( isdefined( self ) )
eModel setmodel( sModel );
}
// MikeD (07/26/07): Removed and put in aircraft_crash();
//helicopter_crash()
//{
// self.crashing = true;
//
// if( isdefined( self.unloading ) )
// {
// while( isdefined( self.unloading ) )
// wait 0.05;
// }
//
// if( !isdefined( self ) )
// return;
//
// // MikeD (07/26/06
// self thread helicopter_crash_move();
//}
aircraft_crash()
{
self.crashing = true;
if( isdefined( self.unloading ) )
{
while( isdefined( self.unloading ) )
{
wait( 0.05 );
}
}
if( !isdefined( self ) )
{
return;
}
self thread aircraft_crash_move();
}
_hasweapon( weapon )
{
weapons = self getweaponslist();
for ( i = 0; i < weapons.size; i++ )
{
if(issubstr( weapons[i],weapon ) )
return true;
}
return false;
}
get_unused_crash_locations()
{
unusedLocations = [];
for ( i = 0; i < level.helicopter_crash_locations.size; i++ )
{
if ( isdefined( level.helicopter_crash_locations[ i ].claimed ) )
continue;
unusedLocations[ unusedLocations.size ] = level.helicopter_crash_locations[ i ];
}
return unusedLocations;
}
detach_getoutrigs()
{
if(!isdefined(self.getoutrig))
return;
if(! self.getoutrig.size )
return;
keys = getarraykeys( self.getoutrig );
for ( i = 0; i < keys.size; i++ )
{
self.getoutrig[keys[i]] unlink();
}
}
aircraft_crash_move()
{
self endon( "death" );
// MikeD (7/30/2007): TODO: Will need something better, like splines?
forward = AnglesToForward( self.angles + ( 45, 0, 0 ) );
dest_point = self.origin + vector_multiply( forward, 10000 );
dest_point = ( dest_point[0], dest_point[1], -700 );
self SetSpeed( 300, 30 );
self SetNearGoalNotifyDist( 300 );
self SetVehGoalPos( dest_point, 1 );
self waittill_any( "goal", "near_goal" );
self notify( "crash_done" );
}
// MikeD (07/26/07): Remove, put in aircraft_crash_move() instead
//helicopter_crash_move()
//{
// // get the nearest unused crash location
// assertEx( level.helicopter_crash_locations.size > 0, "A helicopter tried to crash but you didn't have any script_origins with targetname helicopter_crash_location in the level" );
// crashLoc = undefined;
// unusedLocations = [];
// for( i = 0; i < level.helicopter_crash_locations.size; i ++ )
// {
// if( isdefined( level.helicopter_crash_locations[ i ].claimed ) )
// continue;
// unusedLocations[ unusedLocations.size ] = level.helicopter_crash_locations[ i ];
// }
// assertEx( unusedLocations.size > 0, "You dont have enough script_origins with targetname helicopter_crash_location in the level" );
// crashLoc = getclosest( self.origin, unusedLocations );
// assert( isdefined( crashLoc ) );
// crashLoc.claimed = true;
//
// // make the chopper spin around
// self thread helicopter_crash_rotate();
//
// // move chopper closer to crash point
// self setSpeed( 40, 10, 10 );
// self setNearGoalNotifyDist( 300 );
// self setvehgoalpos( ( crashLoc.origin[ 0 ], crashLoc.origin[ 1 ], self.origin[ 2 ] ), 1 );
// self waittill_any( "goal", "near_goal" );
// self setvehgoalpos( crashLoc.origin, 1 );
// self waittill( "goal" );
// crashLoc.claimed = undefined;
//
// self notify( "crash_done" );
//}
// MikeD (07/26/07): No longer needed, since we do not have helicopters
//helicopter_crash_rotate()
//{
// self endon( "crash_done" );
//
// self setyawspeed( 400, 100, 100 );
// for( ;; )
// {
// self setTargetYaw( self.angles[ 1 ] + 90 );
// wait 0.5;
// }
//}
crash_path_check( node )
{
// find a crashnode on the current path
// this only works on ground info_vehicle_node vheicles. not dynamic helicopter script_origin paths. they have their own dynamic crashing.
targ = node;
while( isdefined( targ ) )
{
if( ( isdefined( targ.detoured ) ) && ( targ.detoured == 0 ) )
{
detourpath = path_detour_get_detourpath( getvehiclenode( targ.target, "targetname" ) );
if( isdefined( detourpath ) && isdefined( detourpath.script_crashtype ) )
return true;
}
if( isdefined( targ.target ) )
{
// Edited for the case of nodes targetting eachother for looping paths 1/30/08 TFlame
targ1 = getvehiclenode( targ.target, "targetname" );
if (isdefined(targ1) && isdefined(targ1.target) && isdefined(targ.targetname) && targ1.target == targ.targetname)
{
return false;
}
else
{
targ = targ1;
}
}
else
targ = undefined;
}
return false;
}
death_firesound( sound )
{
self thread play_loop_sound_on_tag( sound, undefined, false );
self waittill_any( "fire_extinguish", "stop_crash_loop_sound" );
if ( !isdefined( self ) )
return;
self notify( "stop sound" + sound );
}
kill_fx( model )
{
// going to use vehicletypes for identifying a vehicles association with effects.
// will add new vehicle types if vehicle is different enough that it needs to use
// different effect. also handles the sound
if( self isdestructible() )
return;
level notify( "vehicle_explosion", self.origin );
self notify( "explode" );
type = self.vehicletype;
for( i = 0; i < level.vehicle_death_fx[ type ].size; i ++ )
{
struct = level.vehicle_death_fx[ type ][ i ];
thread kill_fx_thread( model, struct, type );
}
}
/*
=============
///ScriptDocBegin
"Name: vehicle_flag_arrived( <msg> )"
"Summary: Script waits until the vehicle hits the node that has script_flag_wait with this msg"
"Module: vehicle_build( vehicle.gsc )"
"CallOn: A vehicle"
"MandatoryArg: <msg> : The flag"
"Example: heli vehicle_flag_arrived( "surrender_to_me" );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
vehicle_flag_arrived( msg )
{
if ( !isdefined( self.vehicle_flags ) )
{
self.vehicle_flags = [];
}
while ( !isdefined( self.vehicle_flags[ msg ] ) )
{
self waittill( "vehicle_flag_arrived", notifymsg );
if ( msg == notifymsg )
return;
}
}
kill_fx_thread( model, struct, type )
{
assert( isdefined( struct ) );
if( isdefined( struct.waitDelay ) )
{
if( struct.waitDelay >= 0 )
wait struct.waitDelay;
else
self waittill( "death_finished" );
}
if ( !isdefined( self ) )
{
// self may have been removed during the wait
return;
}
if( isdefined( struct.notifyString ) )
self notify( struct.notifyString );
eModel = get_dummy();
if( isdefined( struct.effect ) )
{
if( ( struct.bEffectLooping ) && ( !isdefined( self.delete_on_death ) ) )
{
if( isdefined( struct.tag ) )
{
if( ( isdefined( struct.stayontag ) ) && ( struct.stayontag == true ) )
thread loop_fx_on_vehicle_tag( struct.effect, struct.delay, struct.tag );
else
thread playLoopedFxontag( struct.effect, struct.delay, struct.tag );
}
else
{
forward = ( eModel.origin + ( 0, 0, 100 ) ) - eModel.origin;
playfx( struct.effect, eModel.origin, forward );
}
}
else if( isdefined( struct.tag ) )
playfxontag( struct.effect, deathfx_ent(), struct.tag );
else
{
forward = ( eModel.origin + ( 0, 0, 100 ) ) - eModel.origin;
playfx( struct.effect, eModel.origin, forward );
}
}
if( ( isdefined( struct.sound ) ) && ( !isdefined( self.delete_on_death ) ) )
{
if( struct.bSoundlooping )
thread death_firesound( struct.sound );
else
self play_sound_in_space( struct.sound );
}
}
loop_fx_on_vehicle_tag( effect, loopTime, tag )
{
assert( isdefined( effect ) );
assert( isdefined( tag ) );
assert( isdefined( loopTime ) );
self endon( "stop_looping_death_fx" );
while( isdefined( self ) )
{
playfxontag( effect, deathfx_ent(), tag );
wait loopTime;
}
}
/*
=============
///ScriptDocBegin
"Name: build_radiusdamage( <offset> , <range> , <maxdamage> , <mindamage> , <bKillplayer> , <delay> )"
"Summary: called in individual vehicle file - define amount of radius damage to be set on each vehicle"
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"MandatoryArg: <health> : health"
"MandatoryArg: <offset> : worldspace offset vector, usually goes up"
"MandatoryArg: <range> : randomly chooses between the minhealth, maxhealth"
"MandatoryArg: <maxdamage> : randomly chooses between the minhealth, maxhealth"
"MandatoryArg: <mindamage> : randomly chooses between the minhealth, maxhealth"
"MandatoryArg: <bKillplayer> : true / false: kills player"
"OptionalArg: <delay> : delay after "death" to do the damage."
"Example: build_radiusdamage( ( 0, 0, 53 ), 512, 300, 20, false );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_radiusdamage( offset, range, maxdamage, mindamage, bKillplayer, delay )
{
if( !isdefined( level.vehicle_death_radiusdamage ) )
level.vehicle_death_radiusdamage = [];
if( !isdefined( bKillplayer ) )
bKillplayer = false;
if( !isdefined( offset ) )
offset = ( 0, 0, 0 );
struct = spawnstruct();
struct.offset = offset;
struct.range = range;
struct.maxdamage = maxdamage;
struct.mindamage = mindamage;
struct.bKillplayer = bKillplayer;
struct.delay = delay;
level.vehicle_death_radiusdamage[ level.vttype ] = struct;
}
/*
=============
///ScriptDocBegin
"Name: build_rumble( <rumble> , <scale> , <duration> , <radius> , <basetime> , <randomaditionaltime> )"
"Summary: called in individual vehicle file - define amount of radius damage to be set on each vehicle"
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"MandatoryArg: <rumble> : rumble asset"
"MandatoryArg: <scale> : scale"
"MandatoryArg: <duration> : duration"
"MandatoryArg: <radius> : radius"
"MandatoryArg: <basetime> : time to wait between rumbles"
"MandatoryArg: <randomaditionaltime> : random amount of time to add to basetime"
"Example: build_rumble( "tank_rumble", 0.15, 4.5, 600, 1, 1 );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_rumble( rumble, scale, duration, radius, basetime, randomaditionaltime )
{
if( !isdefined( level.vehicle_rumble ) )
level.vehicle_rumble = [];
struct = build_quake( scale, duration, radius, basetime, randomaditionaltime );
assert( isdefined( rumble ) );
precacherumble( rumble );
struct.rumble = rumble;
level.vehicle_rumble[ level.vttype ] = struct;
}
/*
=============
///ScriptDocBegin
"Name: build_deathquake( <scale> , <duration> , <radius> )"
"Summary: called in individual vehicle file - define amount of radius damage to be set on each vehicle"
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"MandatoryArg: <scale> : scale"
"MandatoryArg: <duration> : duration"
"MandatoryArg: <radius> : radius"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_deathquake( scale, duration, radius )
{
if( !isdefined( level.vehicle_death_earthquake ) )
level.vehicle_death_earthquake = [];
level.vehicle_death_earthquake[ level.vttype ] = build_quake( scale, duration, radius );
}
build_quake( scale, duration, radius, basetime, randomaditionaltime )
{
struct = spawnstruct();
struct.scale = scale;
struct.duration = duration;
struct.radius = radius;
if( isdefined( basetime ) )
struct.basetime = basetime;
if( isdefined( randomaditionaltime ) )
struct.randomaditionaltime = randomaditionaltime;
return struct;
}
build_fx( effect, tag, sound, bEffectLooping, delay, bSoundlooping, waitDelay, stayontag, notifyString )
{
if( !isdefined( bSoundlooping ) )
bSoundlooping = false;
if( !isdefined( bEffectLooping ) )
bEffectLooping = false;
if( !isdefined( delay ) )
delay = 1;
struct = spawnstruct();
struct.effect = loadfx( effect );
struct.tag = tag;
struct.sound = sound;
struct.bSoundlooping = bSoundlooping;
struct.delay = delay;
struct.waitDelay = waitDelay;
struct.stayontag = stayontag;
struct.notifyString = notifyString;
struct.bEffectLooping = bEffectLooping;
return struct;
}
/*
=============
///ScriptDocBegin
"Name: build_deathfx_override( <type> , <effect> , <tag> , <sound> , <bEffectLooping> , <delay> , <bSoundlooping> , <waitDelay> , <stayontag> , <notifyString> )"
"Summary: called in individual vehicle file - death effects on vehicles, usually multiple lines for multistaged / multitagged sequences"
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"MandatoryArg: <type> : vehicle type to override the effect of"
"MandatoryArg: <effect> : effect to play on death"
"OptionalArg: <tag> : tag to play the effect on"
"OptionalArg: <sound> : " sound to play with effect
"OptionalArg: <bEffectLooping> : play it old fashioned loop style"
"OptionalArg: <delay> : old fashioned loop time"
"OptionalArg: <bSoundlooping> : true / false: sound loops "
"OptionalArg: <waitDelay> : wait this long after death to start this effect sequence"
"OptionalArg: <stayontag> : playfxontag"
"OptionalArg: <notifyString> : notifies vehicle this when effect starts"
"Example: build_deathfx( "explosions / large_vehicle_explosion", undefined, "explo_metal_rand" );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_deathfx_override( type, effect, tag, sound, bEffectLooping, delay, bSoundlooping, waitDelay, stayontag, notifyString )
{
level.vttype = type;
level.vehicle_death_fx_override[ type ] = true;
if ( !isdefined( level.vehicle_death_fx ) )
level.vehicle_death_fx = [];
build_deathfx( effect, tag, sound, bEffectLooping, delay, bSoundlooping, waitDelay, stayontag, notifyString, true );
}
/*
=============
///ScriptDocBegin
"Name: build_deathfx( <effect> , <tag> , <sound> , <bEffectLooping> , <delay> , <bSoundlooping> , <waitDelay> , <stayontag> , <notifyString> )"
"Summary: called in individual vehicle file - death effects on vehicles, usually multiple lines for multistaged / multitagged sequences"
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"MandatoryArg: <effect> : effect to play on death"
"OptionalArg: <tag> : tag to play the effect on"
"OptionalArg: <sound> : sound to play with effect"
"OptionalArg: <bEffectLooping> : play it old fashioned loop style. Set this to true or undefined"
"OptionalArg: <delay> : old fashioned loop time in seconds"
"OptionalArg: <bSoundlooping> : true / false: sound loops"
"OptionalArg: <waitDelay> : wait this long after death to start this effect sequence"
"OptionalArg: <stayontag> : playfxontag"
"OptionalArg: <notifyString> : notifies vehicle this when effect starts"
"Example: build_deathfx( "explosions / large_vehicle_explosion", undefined, "explo_metal_rand" );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_deathfx( effect, tag, sound, bEffectLooping, delay, bSoundlooping, waitDelay, stayontag, notifyString, bOverride )
{
if ( !isdefined( bOverride ) )
bOverride = false;
assertEx( isdefined( effect ), "Failed to build death effect because there is no effect specified for the model used for that vehicle." );
type = level.vttype;
if(
isdefined( level.vehicle_death_fx_override )
&& isdefined( level.vehicle_death_fx_override[ type ] )
&& level.vehicle_death_fx_override[ type ]
&& !bOverride
)
return; // build_deathfx already defined by override. keeps death effects from loading twice.
if( !isdefined( level.vehicle_death_fx[ type ] ) )
level.vehicle_death_fx[ type ] = [];
level.vehicle_death_fx[ type ][ level.vehicle_death_fx [ type ].size ] = build_fx( effect, tag, sound, bEffectLooping, delay, bSoundlooping, waitDelay, stayontag, notifyString );
}
get_script_modelvehicles()
{
array = [];
models = getentarray( "script_model", "classname" );
if ( isdefined( level.modelvehicles ) )
return level.modelvehicles;
level.modelvehicles = [];
for( i = 0; i < models.size; i ++ )
{
if ( isdefined( models[ i ].targetname )
&& ( models[ i ].targetname == "destructible"
|| models[ i ].targetname == "zpu"
|| models[ i ].targetname == "exploder" )
)
continue;
if ( isdefined( models[ i ].script_noteworthy ) && models[ i ].script_noteworthy == "notvehicle" )
continue;
if( maps\_vehicletypes::is_type( models[ i ].model ) )
{
array[ array.size ] = models[ i ];
models[ i ].vehicletype = maps\_vehicletypes::get_type( models[ i ].model );
}
}
level.modelvehicles = array;
return level.modelvehicles;
}
precache_scripts()
{
// find all the vehicles in the level and initialize precaching( calling of vehicles main() mostly )
allvehiclesprespawn = [];
vehicles = getentarray( "script_vehicle", "classname" );
// vehicles = array_combine( vehicles, getentarray( "vehiclespawnmodel", "targetname" ) );
vehicles = array_combine( vehicles, get_script_modelvehicles() );
level.needsprecaching = [];
playerdrivablevehicles = [];
allvehiclesprespawn = [];
if( !isdefined( level.vehicleInitThread ) )
level.vehicleInitThread = [];
for( i = 0; i < vehicles.size; i ++ )
{
vehicles[ i ].vehicletype = tolower( vehicles[ i ].vehicletype );
if( vehicles[ i ].vehicletype == "bog_mortar" )
continue;
if( isdefined( vehicles[ i ].spawnflags ) && vehicles[ i ].spawnflags & 1 )
playerdrivablevehicles[ playerdrivablevehicles.size ] = vehicles[ i ];
allvehiclesprespawn[ allvehiclesprespawn.size ] = vehicles[ i ];
if( !isdefined( level.vehicleInitThread[ vehicles[ i ].vehicletype ] ) )
level.vehicleInitThread[ vehicles[ i ].vehicletype ] = [];
loadstring = "maps\\\_" + vehicles[ i ].vehicletype + "::main( \"" + vehicles[ i ].model + "\" );";
if( level.bScriptgened )
script_gen_dump_addline( loadstring, vehicles[ i ].model ); // adds to scriptgendump using model as signature for lookup
precachesetup( loadstring, vehicles[ i ] );
}
if( !level.bScriptgened && level.needsprecaching.size > 0 )
{
println( " -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- " );
println( " -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- " );
println( " -- -- - add these lines to your level script above maps\\\_load::main(); -- -- -- -- -- -- - " );
for( i = 0; i < level.needsprecaching.size; i ++ )
println( level.needsprecaching[ i ] );
println( " -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- " );
println( " -- -- -- -- -- -- -- -- -- -- -- -- - hint copy paste them from console.log -- -- -- -- -- -- -- -- -- -- " );
println( " -- -- if it already exists then check for unmatching vehicletypes in your map -- -- -- " );
println( " -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- " );
// CODER_MOD - JamesS - See2 is having some load problems after IW integrate
//assertEX( false, "missing vehicle scripts, see above console prints" );
//level waittill( "never" );
}
if( playerdrivablevehicles.size > 0 )
thread maps\_vehicledrive::main(); // precache driveable vehicle huds and such.
return allvehiclesprespawn;
}
precachesetup( string, vehicle )
{
if( isdefined( level.vehicleInitThread[ vehicle.vehicletype ][ vehicle.model ] ) )
return;
matched = false;
for( i = 0; i < level.needsprecaching.size; i ++ )
if( level.needsprecaching[ i ] == string )
matched = true;
if( !matched )
level.needsprecaching[ level.needsprecaching.size ] = string;
}
vehicle_modelinarray( arrayofmodels, model )
{
for( i = 0; i < arrayofmodels.size; i ++ )
if( arrayofmodels[ i ] == model )
return true;
return false;
}
vehicle_kill_disconnect_paths_forever()
{
self notify( "kill_disconnect_paths_forever" );
}
disconnect_paths_whenstopped()
{
self endon( "death" );
self endon( "kill_disconnect_paths_forever" );
if ( isdefined( self.script_disconnectpaths ) && !self.script_disconnectpaths )
{
self.dontDisconnectPaths = true;// lets other parts of the script know not to disconnect script
return;
}
wait( randomfloat( 1 ) );
while( isdefined( self ) )
{
if( self getspeed() < 1 )
{
if ( !isdefined( self.dontDisconnectPaths ) )
self disconnectpaths();
self notify( "speed_zero_path_disconnect" );
while( self getspeed() < 1 )
wait .05;
}
self connectpaths();
wait 1;
}
}
vehicle_setspeed( speed, rate, msg )
{
if( self getspeedmph() == 0 && speed == 0 )
return; // potential for disaster? keeps messages from overriding previous messages
/#
self thread debug_vehiclesetspeed( speed, rate, msg );
#/
self setspeed( speed, rate );
}
debug_vehiclesetspeed( speed, rate, msg )
{
/#
self notify( "new debug_vehiclesetspeed" );
self endon( "new debug_vehiclesetspeed" );
self endon( "resuming speed" );
self endon( "death" );
while( 1 )
{
while( getdvar( "debug_vehiclesetspeed" ) != "off" )
{
print3d( self.origin + ( 0, 0, 192 ), "vehicle setspeed: " + msg, ( 1, 1, 1 ), 1, 3 );
wait .05;
}
wait .5;
}
#/
}
script_resumespeed( msg, rate )
{
self endon( "death" );
fSetspeed = 0;
type = "resumespeed";
if( !isdefined( self.resumemsgs ) )
self.resumemsgs = [];
if( isdefined( self.waitingforgate ) && self.waitingforgate )
return; // ignore resumespeeds on waiting for gate.
if( isdefined( self.attacking ) )
{
if( self.attacking )
{
fSetspeed = self.attackspeed;
type = "setspeed";
}
}
self.zerospeed = false;
if( fSetspeed == 0 )
self.zerospeed = true;
if( type == "resumespeed" )
self resumespeed( rate );
else if( type == "setspeed" )
self vehicle_setspeed( fSetspeed, 15, "resume setspeed from attack" );
self notify( "resuming speed" );
/# self thread debug_vehicleresume( msg + " :" + type ); #/
}
/#
debug_vehicleresume( msg )
{
if( getdvar( "debug_vehicleresume" ) == "off" )
return;
self endon( "death" );
number = self.resumemsgs.size;
self.resumemsgs[ number ] = msg;
timer = 3;
self thread print_resumespeed( gettime() + ( timer * 1000 ) );
wait timer;
newarray = [];
for( i = 0; i < self.resumemsgs.size; i ++ )
{
if( i != number )
newarray[ newarray.size ] = self.resumemsgs[ i ];
}
self.resumemsgs = newarray;
}
#/
print_resumespeed( timer )
{
self notify( "newresumespeedmsag" );
self endon( "newresumespeedmsag" );
self endon( "death" );
while( gettime() < timer && isdefined( self.resumemsgs ) )
{
if( self.resumemsgs.size > 6 )
start = self.resumemsgs.size - 5;
else
start = 0;
for( i = start; i < self.resumemsgs.size; i ++ ) // only display last 5 messages
{
position = i * 32;
print3d( self.origin + ( 0, 0, position ), "resuming speed: " + self.resumemsgs[ i ], ( 0, 1, 0 ), 1, 3 );
}
wait .05;
}
}
/#
vclogin_vehicles()
{
if( getdvar( "vclogin_vehicles" ) == "off" )
return;
precachemodel( "vehicle_blackhawk" );
level.vclogin_vehicles = 1;
// paths = level.vehicle_startnodes;
vehicles = getentarray( "script_vehicle", "classname" );
for( i = 0; i < vehicles.size; i ++ )
vehicles[ i ] delete();
paths = getallvehiclenodes();
for( i = 0; i < paths.size; i ++ )
{
if( !( isdefined( paths[ i ].spawnflags ) && ( paths[ i ].spawnflags & 1 ) ) )
continue;
crashtype = paths[ i ].script_crashtype;
if( !isdefined( crashtype ) )
crashtype = "default";
if( crashtype == "plane" )
vehicle = spawnVehicle( "vehicle_blackhawk", "vclogger", "blackhawk", ( 0, 0, 0 ), ( 0, 0, 0 ) );
else
vehicle = spawnVehicle( "vehicle_blackhawk", "vclogger", "blackhawk", ( 0, 0, 0 ), ( 0, 0, 0 ) );
vehicle attachpath( paths[ i ] );
// SCRIPTER_MOD: JesseS (8/10/2007): removed level.player references
if( isdefined( vehicle.model ) && vehicle.model == "vehicle_blackhawk" )
{
tagorg = vehicle gettagorigin( "tag_bigbomb" );
get_players()[0] setorigin( tagorg );
get_players()[0] playerLinkTodelta( vehicle, "tag_bigbomb", 1.0 );
}
else
{
tagorg = vehicle gettagorigin( "tag_player" );
get_players()[0] setorigin( tagorg );
get_players()[0] playerLinkToDelta( vehicle, "tag_player", 0.1 );
}
vehicle startpath();
vehicle.zerospeed = false;
// vehicle setspeed( 100, 50 );
vehicle waittill( "reached_end_node" );
get_players()[0] unlink();
vehicle delete();
crashtype = undefined;
}
level waittill( "never" ); // map needs to be restarted at this point
}
#/
godon()
{
self.godmode = true;
}
godoff()
{
self.godmode = false;
}
setturretfireondrones( b )
{
if( isdefined( self.mgturret ) && self.mgturret.size )
for( i = 0; i < self.mgturret.size; i ++ )
self.mgturret[ i ].script_fireondrones = b;
}
getnormalanimtime( animation )
{
animtime = self getanimtime( animation );
animlength = getanimlength( animation );
if( animtime == 0 )
return 0;
return self getanimtime( animation ) / getanimlength( animation );
}
rotor_anim()
{
length = getanimlength( self getanim( "rotors" ) );
for ( ;; )
{
self setanim( self getanim( "rotors" ), 1, 0, 1 );
wait( length );
}
}
animate_drive_idle()
{
if( !isdefined( self.wheeldir ) )
self.wheeldir = 1;
model = self.model;
newanimtime = undefined;
self UseAnimTree( #animtree );
if( !isdefined( level.vehicle_DriveIdle[ model ] ) )
return;
if( !isdefined( level.vehicle_DriveIdle_r[ model ] ) )
level.vehicle_DriveIdle_r[ model ] = level.vehicle_DriveIdle[ model ]; // use forward animation if no backwards anim exists
self endon( "death" );
normalspeed = level.vehicle_DriveIdle_normal_speed[ model ];
thread animate_drive_idle_death();
animrate = 1.0;
if( ( isdefined( level.vehicle_DriveIdle_animrate ) ) && ( isdefined( level.vehicle_DriveIdle_animrate[ model ] ) ) )
animrate = level.vehicle_DriveIdle_animrate[ model ];
lastdir = self.wheeldir;
animatemodel = self;
animation = level.vehicle_DriveIdle[ model ];
while( 1 )
{
if( !normalspeed )
{
// vehicles like helicopters always play the same rate. will come up with better design if need arises.
animatemodel setanim( level.vehicle_DriveIdle[ model ], 1, .2, animrate );
thread animtimer( .5 );
self waittill( "animtimer" );
continue;
}
speed = self getspeedmph();
if( lastdir != self.wheeldir )
{
dif = 0;
if( self.wheeldir )
{
animation = level.vehicle_DriveIdle [ model ];
dif = 1 - animatemodel getnormalanimtime( level.vehicle_DriveIdle_r [ model ] );
animatemodel clearanim( level.vehicle_DriveIdle_r [ model ], 0 );
}
else
{
animation = level.vehicle_DriveIdle_r[ model ]; // reverse direction
dif = 1 - animatemodel getnormalanimtime( level.vehicle_DriveIdle [ model ] );
animatemodel clearanim( level.vehicle_DriveIdle[ model ], 0 );
}
newanimtime = 0.01;
if( newanimtime >= 1 || newanimtime == 0 )
newanimtime = 0.01; // think setting animtime to 0 or 1 messes things up
lastdir = self.wheeldir;
}
if( speed == 0 )
animatemodel setanim( animation, 1, .2, 0 );
else
animatemodel setanim( animation, 1, .2, speed / normalspeed );
if( isdefined( newanimtime ) )
{
animatemodel setanimtime( animation, newanimtime );
newanimtime = undefined;
}
thread animtimer( .2 );
self waittill( "animtimer" );
}
}
animtimer( time )
{
self endon( "animtimer" );
wait time;
self notify( "animtimer" );
}
animate_drive_idle_death()
{
model = self.model;
self UseAnimTree( #animtree );
self waittill( "death_finished" );
if( isdefined( self ) )
self clearanim( level.vehicle_DriveIdle[ model ], 0 );
}
setup_dynamic_detour( pathnode , get_func )
{
prevnode = [[ get_func ]]( pathnode.targetname );
assertex( isdefined( prevnode ), "detour can't be on start node" );
prevnode.detoured = 0;
}
/*
setup_origins()
{
triggers = [];
origins = getentarray( "script_origin", "classname" );
for( i = 0; i < origins.size; i ++ )
{
if( isdefined( origins[ i ].script_vehicledetour ) )
{
level.vehicle_detourpaths = array_2dadd( level.vehicle_detourpaths, origins[ i ].script_vehicledetour, origins[ i ] );
if( level.vehicle_detourpaths[ origins[ i ].script_vehicledetour ].size > 2 )
println( "more than two script_vehicledetour grouped in group number: ", origins[ i ].script_vehicledetour );
prevnode = getent( origins[ i ].targetname, "target" );
assertex( isdefined( prevnode ), "detour can't be on start node" );
triggers[ triggers.size ] = prevnode;
prevnode.detoured = 0;
prevnode = undefined;
}
}
return triggers;
}
*/
setup_ai()
{
ai = getaiarray();
for( i = 0; i < ai.size; i ++ )
{
if( isdefined( ai[ i ].script_vehicleride ) )
level.vehicle_RideAI = array_2dadd( level.vehicle_RideAI, ai[ i ].script_vehicleride, ai[ i ] );
else
if( isdefined( ai[ i ].script_vehiclewalk ) )
level.vehicle_WalkAI = array_2dadd( level.vehicle_WalkAI, ai[ i ].script_vehiclewalk, ai[ i ] );
}
ai = getspawnerarray();
for( i = 0; i < ai.size; i ++ )
{
if( isdefined( ai[ i ].script_vehicleride ) )
level.vehicle_RideSpawners = array_2dadd( level.vehicle_RideSpawners, ai[ i ].script_vehicleride, ai[ i ] );
if( isdefined( ai[ i ].script_vehiclewalk ) )
level.vehicle_walkspawners = array_2dadd( level.vehicle_walkspawners, ai[ i ].script_vehiclewalk, ai[ i ] );
}
}
array_2dadd( array, firstelem, newelem )
{
if( !isdefined( array[ firstelem ] ) )
array[ firstelem ] = [];
array[ firstelem ][ array[ firstelem ].size ] = newelem;
return array;
}
is_node_script_origin( pathnode )
{
return isdefined( pathnode.classname ) && pathnode.classname == "script_origin";
}
// this determines if the node will be sent through trigger_process. The uber trigger function that may get phased out.
node_trigger_process()
{
processtrigger = false;
// special treatment for start nodes
if( isdefined( self.spawnflags ) && ( self.spawnflags & 1 ) )
{
if( isdefined( self.script_crashtype ) )
level.vehicle_crashpaths[ level.vehicle_crashpaths.size ] = self;
level.vehicle_startnodes[ level.vehicle_startnodes.size ] = self;
}
if( isdefined( self.script_vehicledetour ) && isdefined( self.targetname ) )
{
get_func = undefined;
// get_func is differnt for struct types and script_origin types of paths
if( isdefined( get_from_entity( self.targetname ) ) )
get_func = ::get_from_entity_target;
if( isdefined( get_from_spawnstruct( self.targetname ) ) )
get_func = ::get_from_spawnstruct_target;
if( isdefined( get_func ) )
{
setup_dynamic_detour( self, get_func );
processtrigger = true; // the node with the script_vehicledetour waits for the trigger here unlike ground nodes which need to know 1 node in advanced that there's a detour, tricky tricky.
}
else
{
setup_groundnode_detour( self ); // other trickery. the node is set to process in there.
}
level.vehicle_detourpaths = array_2dadd( level.vehicle_detourpaths, self.script_vehicledetour, self );
if( level.vehicle_detourpaths[ self.script_vehicledetour ].size > 2 )
println( "more than two script_vehicledetour grouped in group number: ", self.script_vehicledetour );
}
// if a gate isn't open then the vehicle will stop there and wait for it to become open.
if( isdefined( self.script_gatetrigger ) )
{
level.vehicle_gatetrigger = array_2dadd( level.vehicle_gatetrigger, self.script_gatetrigger, self );
self.gateopen = false;
}
// init the flags!
if ( isdefined( self.script_flag_set ) )
{
if ( !isDefined( level.flag[ self.script_flag_set ] ) )
flag_init( self.script_flag_set );
}
// init the flags!
if ( isdefined( self.script_flag_clear ) )
{
if ( !isDefined( level.flag[ self.script_flag_clear ] ) )
flag_init( self.script_flag_clear );
}
if( isdefined( self.script_flag_wait ) )
{
if ( !isDefined( level.flag[ self.script_flag_wait ] ) )
flag_init( self.script_flag_wait );
}
// various nodes that will be sent through trigger_process
if (
isdefined( self.script_VehicleSpawngroup )
|| isdefined( self.script_VehicleStartMove )
|| isdefined( self.script_gatetrigger )
|| isdefined( self.script_Vehiclegroupdelete )
)
processtrigger = true;
if( processtrigger )
add_proccess_trigger( self );
}
setup_triggers()
{
// TODO: move this to _load under the triggers section. larger task than this simple cleanup.
// the processtriggers array is all the triggers and vehicle node triggers to be put through
// the trigger_process function. This is so that I only do a waittill trigger once
// in script to assure better sequencing on a multi - function trigger.
// some of the vehiclenodes don't need to waittill trigger on anything and are here only
// for being linked with other trigger
level.vehicle_processtriggers = [];
triggers = [];
triggers = array_combine( getallvehiclenodes(), getentarray( "script_origin", "classname" ) );
triggers = array_combine( triggers, level.struct );
triggers = array_combine( triggers, getentarray( "trigger_radius", "classname" ) );
triggers = array_combine( triggers, getentarray( "trigger_multiple", "classname" ) );
triggers = array_combine( triggers, getentarray( "trigger_lookat", "classname" ) );
array_thread( triggers, ::node_trigger_process );
}
is_node_script_struct( node )
{
if( ! isdefined( node.targetname ) )
return false;
return isdefined( getstruct( node.targetname, "targetname" ) );
}
setup_vehicles( allvehiclesprespawn )
{
// vehicles = getentarray( "script_vehicle", "classname" );
// old setup required targetname. blahrg
// vehicles = array_combine( vehicles, getentarray( "vehiclespawnmodel", "targetname" ) );
// this could be dangerous but lets give it a try and see what happesn.. yeehaw! - Nate
// vehicles = array_combine( vehicles, getentarray( "script_model", "classname" ) );
vehicles = allvehiclesprespawn;
spawnvehicles = [];
groups = [];
nonspawned = [];
for( i = 0; i < vehicles.size; i ++ )
{
if( isdefined( vehicles[ i ].script_vehiclespawngroup ) )
{
if( !isdefined( spawnvehicles[ vehicles[ i ].script_vehiclespawngroup ] ) )
spawnvehicles[ vehicles[ i ].script_vehiclespawngroup ] = [];
spawnvehicles[ vehicles[ i ].script_vehiclespawngroup ]
[ spawnvehicles[ vehicles[ i ].script_vehiclespawngroup ].size ] = vehicles[ i ];
addgroup[ 0 ] = vehicles[ i ].script_vehiclespawngroup;
groups = array_merge( groups, addgroup );
continue;
}
// else if( vehicles[ i ].classname == "script_vehicle" )
else
nonspawned[ nonspawned.size ] = vehicles[ i ];
}
for( i = 0; i < groups.size; i ++ )
thread spawner_setup( spawnvehicles[ groups[ i ] ], groups[ i ], "main" );
// init vehicles that aren't spawned
for( i = 0; i < nonspawned.size; i ++ )
thread vehicle_init( nonspawned[ i ] );
}
vehicle_life()
{
type = self.vehicletype;
if( !isdefined( level.vehicle_life ) || !isdefined( level.vehicle_life[ self.vehicletype ] ) )
{
wait 2;
}
assertEX( isdefined( level.vehicle_life[ type ] ), "need to specify build_life() in vehicle script for vehicletype: " + type );
if( isdefined( self.script_startinghealth ) )
self.health = self.script_startinghealth;
else
{
if( level.vehicle_life[ type ] == -1 )
return;
else if( isdefined( level.vehicle_life_range_low[ type ] ) && isdefined( level.vehicle_life_range_high[ type ] ) )
self.health = ( randomint( level.vehicle_life_range_high[ type ] - level.vehicle_life_range_low[ type ] ) + level.vehicle_life_range_low[ type ] );
else
self.health = level.vehicle_life[ type ];
}
// if( isdefined( level.destructible_model[ self.model ] ) )
// {
// self.health = 2000;
// self.destructible_type = level.destructible_model[ self.model ];
// self maps\_destructible::setup_destructibles( true );
// }
}
mginit()
{
type = self.vehicletype;
if( (( isdefined( self.script_nomg ) ) && ( self.script_nomg > 0 ) ) )
return;
// dpg 3/12/08 removed is_aircraft()
if( !isdefined( level.vehicle_mgturret[ type ] ) )
return;
mgangle = 0;
if( isdefined( self.script_mg_angle ) )
mgangle = self.script_mg_angle;
turret_template = level.vehicle_mgturret[ type ];
if( !isdefined( turret_template ) )
return;
for( i = 0; i < turret_template.size; i ++ )
{
self.mgturret[ i ] = spawnTurret( "misc_turret", ( 0, 0, 0 ), turret_template[ i ].info );
self.mgturret[ i ] linkto( self, turret_template[ i ].tag, ( 0, 0, 0 ), ( 0, -1 * mgangle, 0 ) );
self.mgturret[ i ] setmodel( turret_template[ i ].model );
self.mgturret[ i ].angles = self.angles;
self.mgturret[ i ].isvehicleattached = true; // lets mgturret know not to mess with this turret
self.mgturret[ i ] thread maps\_mgturret::burst_fire_unmanned();
self.mgturret[ i ] maketurretunusable();
level thread maps\_mgturret::mg42_setdifficulty( self.mgturret[ i ], getdifficulty() );
if( isdefined( self.script_fireondrones ) )
self.mgturret[ i ].script_fireondrones = self.script_fireondrones;
self.mgturret[ i ] setshadowhint( "never" );
if ( isdefined( turret_template[ i ].deletedelay ) )
self.mgturret[ i ].deletedelay = turret_template[ i ].deletedelay;
if( isdefined( turret_template[ i ].defaultOFFmode ) )
self.mgturret[ i ] setmode( turret_template[ i ].defaultOFFmode );
if( isdefined( turret_template[ i ].maxrange ) )
self.mgturret[ i ].maxrange = turret_template[ i ].maxrange;
if( isdefined( self.script_noteworthy ) && self.script_noteworthy == "onemg" )
break;
}
if( !isdefined( self.script_turretmg ) )
self.script_turretmg = true;;
if( isdefined( self.script_turretmg ) && self.script_turretmg == 0 )
self thread mgoff();
else
{
self.script_turretmg = 1;
self thread mgon();
}
self thread mgtoggle();
}
mgtoggle()
{
self endon( "death" );
if( self.script_turretmg )
lasttoggle = 1;
else
lasttoggle = 0;
while( 1 )
{
if( lasttoggle != self.script_turretmg )
{
lasttoggle = self.script_turretmg;
if( self.script_turretmg )
self thread mgon();
else
self thread mgoff();
}
wait .5;
}
}
mgoff()
{
type = self.vehicletype;
self.script_turretmg = 0;
if( ( self is_aircraft() ) && ( self hasHelicopterTurret() ) )
{
self thread chopper_Turret_Off();
return;
}
// dpg 7/3/07: if self isn't defined, return
if( !isdefined( self ) || !isdefined( self.mgturret ) )
return;
for( i = 0; i < self.mgturret.size; i ++ )
{
if( isdefined( self.mgturret[ i ].script_fireondrones ) )
self.mgturret[ i ].script_fireondrones = false;
if( isdefined( level.vehicle_mgturret[ type ][ i ].defaultOFFmode ) )
self.mgturret[ i ] setmode( level.vehicle_mgturret[ type ][ i ].defaultOFFmode );
else
self.mgturret[ i ] setmode( "manual" );
}
}
mgon()
{
type = self.vehicletype;
self.script_turretmg = 1; // fix me.. defense for scripts using mgon();
// MikeD 04/26/07: Helicopters not in CoD5.
// if( ( self is_aircraft() ) && ( self hasHelicopterTurret() ) )
// {
// self thread chopper_Turret_On();
// return;
// }
// dpg 7/3/07: if self isn't defined, return
if( !isdefined( self ) || !isdefined( self.mgturret ) )
return;
for( i = 0; i < self.mgturret.size; i ++ )
{
if( isDefined( self.mgturret[ i ].bAiControlled ) && !self.mgturret[i].bAiControlled )
{
continue;
}
if( isdefined( self.mgturret[ i ].script_fireondrones ) )
self.mgturret[ i ].script_fireondrones = true;
if( isdefined( level.vehicle_mgturret[ type ][ i ].defaultONmode ) )
self.mgturret[ i ] setmode( level.vehicle_mgturret[ type ][ i ].defaultONmode );
else
self.mgturret[ i ] setmode( "auto_nonai" );
if( ( self.script_team == "allies" ) || ( self.script_team == "friendly" ) )
self.mgturret[ i ] setTurretTeam( "allies" );
else if( ( self.script_team == "axis" ) || ( self.script_team == "enemy" ) )
self.mgturret[ i ] setTurretTeam( "axis" );
}
}
is_aircraft()
{
return isdefined( level.aircraft_list[self.vehicletype] );
}
is_boat()
{
return IsDefined( level.boat_list[self.vehicletype] );
}
isCheap()
{
if( !isdefined( self.script_cheap ) )
return false;
if( !self.script_cheap )
return false;
return true;
}
hasHelicopterDustKickup()
{
if( !is_aircraft() )
return false;
if( isCheap() )
return false;
return true;
}
hasHelicopterTurret()
{
if( !isdefined( self.vehicletype ) )
return false;
if( isCheap() )
return false;
if( self.vehicletype == "cobra" )
return true;
if( self.vehicletype == "cobra_player" )
return true;
return false;
}
// MikeD 04/26/07: Helicopters not in CoD5.
//Chopper_Turret_On()
//{
// self endon( "death" );
// self endon( "mg_off" );
//
// cosine55 = cos( 55 );
//
// while( self.health > 0 )
// {
// // target range, target fov, getAITargets, doTrace
// eTarget = self maps\_helicopter_globals::getEnemyTarget( 16000, cosine55, true, true );
// if( isdefined( eTarget ) )
// self thread maps\_helicopter_globals::shootEnemyTarget_Bullets( eTarget );
// wait 2;
// }
//}
chopper_Turret_Off()
{
self notify( "mg_off" );
}
playLoopedFxontag( effect, durration, tag )
{
eModel = get_dummy();
effectorigin = spawn( "script_origin", eModel.origin );
self endon( "fire_extinguish" );
thread playLoopedFxontag_originupdate( tag, effectorigin );
while( 1 )
{
playfx( effect, effectorigin.origin, effectorigin.upvec );
wait durration;
}
}
playLoopedFxontag_originupdate( tag, effectorigin )
{
effectorigin.angles = self gettagangles( tag );
effectorigin.origin = self gettagorigin( tag );
effectorigin.forwardvec = anglestoforward( effectorigin.angles );
effectorigin.upvec = anglestoup( effectorigin.angles );
while( isdefined( self ) && self.classname == "script_vehicle" && self getspeedmph() > 0 )
{
eModel = get_dummy();
effectorigin.angles = eModel gettagangles( tag );
effectorigin.origin = eModel gettagorigin( tag );
effectorigin.forwardvec = anglestoforward( effectorigin.angles );
effectorigin.upvec = anglestoup( effectorigin.angles );
wait .05;
}
}
build_turret( info, tag, model, bAicontrolled, maxrange, defaultONmode, defaultOFFmode, deletedelay, max_turrets )
{
if( !isdefined( level.vehicle_mgturret ) )
level.vehicle_mgturret = [];
if( !isdefined( level.vehicle_mgturret[ level.vttype ] ) )
level.vehicle_mgturret[ level.vttype ] = [];
// dpg 3/12/08 added max turrets incase you want two different models with the same vehicletype
if( IsDefined( max_turrets ) && level.vehicle_mgturret[ level.vttype ].size >= max_turrets )
{
return;
}
precachemodel( model );
precacheturret( info );
struct = spawnstruct();
struct.info = info;
struct.tag = tag;
struct.model = model;
struct.bAicontrolled = bAicontrolled;
struct.maxrange = maxrange;
struct.defaultONmode = defaultONmode;
struct.defaultOFFmode = defaultOFFmode;
struct.deletedelay = deletedelay;
level.vehicle_mgturret[ level.vttype ][ level.vehicle_mgturret[ level.vttype ].size ] = struct;
}
setup_dvars()
{
/#
if( getdvar( "debug_tankcrush" ) == "" )
setdvar( "debug_tankcrush", "0" );
if( getdvar( "vclogin_vehicles" ) == "" )
setdvar( "vclogin_vehicles", "off" );
if( getdvar( "debug_vehicleresume" ) == "" )
setdvar( "debug_vehicleresume", "off" );
if( getdvar( "debug_vehiclesetspeed" ) == "" )
setdvar( "debug_vehiclesetspeed", "off" );
#/
}
setup_levelvars()
{
level.vehicle_ResumeSpeed = 5;
level.vehicle_DeleteGroup = [];
level.vehicle_SpawnGroup = [];
level.vehicle_StartMoveGroup = [];
level.vehicle_RideAI = [];
level.vehicle_WalkAI = [];
level.vehicle_DeathSwitch = [];
level.vehicle_RideSpawners = [];
level.vehicle_walkspawners = [];
level.vehicle_gatetrigger = [];
level.vehicle_crashpaths = [];
level.vehicle_target = [];
level.vehicle_link = [];
level.vehicle_truckjunk = [];
level.vehicle_detourpaths = [];
// level.vehicle_linkedpaths = [];
level.vehicle_startnodes = [];
level.vehicle_spawners = [];
level.helicopter_crash_locations = getentarray( "helicopter_crash_location", "targetname" );
level.vclogin_vehicles = 0;
level.playervehicle = spawn( "script_origin", ( 0, 0, 0 ) ); // no isdefined for level.playervehicle
level.playervehiclenone = level.playervehicle; // no isdefined for level.playervehicle
// TODO in a thousand next games.. I don't like managing this variable. not so much that I don't like it, just that I haven't been = /
level.vehicles = []; // will contain all the vehicles that are spawned and alive
level.vehicles[ "allies" ] = [];
level.vehicles[ "axis" ] = [];
level.vehicles[ "neutral" ] = [];
if( !isdefined( level.vehicle_team ) )
{
level.vehicle_team = [];
}
if( !isdefined( level.vehicle_deathmodel ) )
{
level.vehicle_deathmodel = [];
}
if( !isdefined( level.vehicle_death_thread ) )
{
level.vehicle_death_thread = [];
}
if( !isdefined( level.vehicle_DriveIdle ) )
{
level.vehicle_DriveIdle = [];
}
if( !isdefined( level.vehicle_DriveIdle_r ) )
{
level.vehicle_DriveIdle_r = [];
}
if( !isdefined( level.attack_origin_condition_threadd ) )
{
level.attack_origin_condition_threadd = [];
}
if( !isdefined( level.vehiclefireanim ) )
{
level.vehiclefireanim = [];
}
if( !isdefined( level.vehiclefireanim_settle ) )
{
level.vehiclefireanim_settle = [];
}
if( !isdefined( level.vehicle_hasname ) )
{
level.vehicle_hasname = [];
}
if( !isdefined( level.vehicle_turret_requiresrider ) )
{
level.vehicle_turret_requiresrider = [];
}
if( !isdefined( level.vehicle_rumble ) )
{
level.vehicle_rumble = [];
}
if( !isdefined( level.vehicle_mgturret ) )
{
level.vehicle_mgturret = [];
}
if( !isdefined( level.vehicle_isStationary ) )
{
level.vehicle_isStationary = [];
}
if( !isdefined( level.vehicle_rumble ) )
{
level.vehicle_rumble = [];
}
if( !isdefined( level.vehicle_death_earthquake ) )
{
level.vehicle_death_earthquake = [];
}
if( !isdefined( level.vehicle_treads ) )
{
level.vehicle_treads = [];
}
if( !isdefined( level.vehicle_compassicon ) )
{
level.vehicle_compassicon = [];
}
if( !isdefined( level.vehicle_unloadgroups ) )
{
level.vehicle_unloadgroups = [];
}
if( !isdefined( level.vehicle_aianims ) )
{
level.vehicle_aianims = [];
}
if( !isdefined( level.vehicle_unloadwhenattacked ) )
{
level.vehicle_unloadwhenattacked = [];
}
if( !isdefined( level.vehicle_exhaust ) )
{
level.vehicle_exhaust = [];
}
if( !isdefined( level.vehicle_deckdust ) )
{
level.vehicle_deckdust = [];
}
// if( !isdefined( level.vehicle_shoot_shock ) )
// {
// level.vehicle_shoot_shock = [];
// }
if( !isdefined( level.vehicle_frontarmor ) )
{
level.vehicle_frontarmor = [];
}
// if( !isdefined( level.destructible_model ) )
// level.destructible_model = [];
if( !isdefined( level.vehicle_types ) )
{
level.vehicle_types = [];
}
if( !isdefined( level.vehicle_compass_types ) )
{
level.vehicle_compass_types = [];
}
if( !isdefined( level.vehicle_bulletshield ) )
{
level.vehicle_bulletshield = [];
}
if( !isdefined( level.vehicle_death_jolt ) )
{
level.vehicle_death_jolt = [];
}
if( !isdefined( level.vehicle_death_badplace ) )
{
level.vehicle_death_badplace = [];
}
// if( !isdefined( level.vehicle_shoot_rumble ) )
// {
// level.vehicle_shoot_rumble = [];
// }
maps\_vehicle_aianim::setup_aianimthreads();
}
attacker_isonmyteam( attacker )
{
if( ( isdefined( attacker ) ) && isdefined( attacker.script_team ) && ( isdefined( self.script_team ) ) && ( attacker.script_team == self.script_team ) )
{
return true;
}
else
return false;
}
is_godmode()
{
if( isdefined( self.godmode ) && self.godmode )
return true;
else
return false;
}
attacker_troop_isonmyteam( attacker )
{
if ( isdefined( self.script_team ) && self.script_team == "allies" && isdefined( attacker ) && isdefined(level.player) && attacker == level.player )
return true; // player is always on the allied team.. hahah! future CoD games that let the player be the enemy be damned!
else if( isai( attacker ) && attacker.team == self.script_team )
return true;
else
return false;
}
has_frontarmor()
{
return( isdefined( level.vehicle_frontarmor [ self.vehicletype ] ) );
}
bulletshielded( type )
{
if ( !isdefined( self.script_bulletshield ) )
return false;
type = tolower( type );
if ( ! isdefined( type ) || ! issubstr( type, "bullet" ) )
return false;
if ( self.script_bulletshield )
return true;
else
return false;
}
friendlyfire_shield()
{
self endon( "death" );
self endon( "stop_friendlyfire_shield" );
if ( isdefined( level.vehicle_bulletshield[ self.vehicletype ] ) && !isdefined( self.script_bulletshield ) )
self.script_bulletshield = level.vehicle_bulletshield[ self.vehicletype ];
self.healthbuffer = 20000;
self.health += self.healthbuffer;
self.currenthealth = self.health;
attacker = undefined;
while( self.health > 0 )
{
self waittill( "damage", amount, attacker, direction_vec, point, type, modelName, tagName );
if(
( ! isdefined( attacker ) && self.script_team != "neutral" )
|| is_godmode()
|| attacker_isonmyteam( attacker )
|| attacker_troop_isonmyteam( attacker )
|| isDestructible()
|| bulletshielded( type )
)
self.health = self.currenthealth; // give back health for these things
else if( self has_frontarmor() ) // regen health for tanks with armor in the front
{
self regen_front_armor( attacker, amount );
self.currenthealth = self.health;
}
else
self.currenthealth = self.health;
// if( maps\_destructible::getDamageType( type ) == "splash" )
// self.rocket_destroyed_for_achievement = true; // little bit of hackery, not perfect but contributes to achievement script for determining that this heli was destroyed by the players RPG.
// else
// self.rocket_destroyed_for_achievement = undefined;
if( self.health < self.healthbuffer )
break;
amount = undefined;
attacker = undefined;
direction_vec = undefined;
point = undefined;
modelName = undefined;
tagName = undefined;
type = undefined;
}
self notify( "death", attacker );
}
regen_front_armor( attacker, amount )
{
forwardvec = anglestoforward( self.angles );
othervec = vectorNormalize( attacker.origin - self.origin );
if( vectordot( forwardvec, othervec ) > .86 )
self.health += int( amount * level.vehicle_frontarmor [ self.vehicletype ] );
}
vehicle_kill_rumble_forever()
{
self notify( "kill_rumble_forever" );
}
vehicle_rumble()
{
// makes vehicle rumble
self endon( "kill_rumble_forever" );
type = self.vehicletype;
if( !isdefined( level.vehicle_rumble[ type ] ) )
return;
if(level.clientScripts)
return; // CODER_MOD : DSL 06/06/08 - moving vehicle rumble to the client, 4tw.
rumblestruct = level.vehicle_rumble[ type ];
height = rumblestruct.radius * 2;
zoffset = -1 * rumblestruct.radius;
areatrigger = spawn( "trigger_radius", self.origin + ( 0, 0, zoffset ), 0, rumblestruct.radius, height );
areatrigger enablelinkto();
areatrigger linkto( self );
self.rumbletrigger = areatrigger;
self endon( "death" );
// ( rumble, scale, duration, radius, basetime, randomaditionaltime )
if( !isdefined( self.rumbleon ) )
self.rumbleon = true;
if( isdefined( rumblestruct.scale ) )
self.rumble_scale = rumblestruct.scale;
else
self.rumble_scale = 0.15;
if( isdefined( rumblestruct.duration ) )
self.rumble_duration = rumblestruct.duration;
else
self.rumble_duration = 4.5;
if( isdefined( rumblestruct.radius ) )
self.rumble_radius = rumblestruct.radius;
else
self.rumble_radius = 600;
if( isdefined( rumblestruct.basetime ) )
self.rumble_basetime = rumblestruct.basetime;
else
self.rumble_basetime = 1;
if( isdefined( rumblestruct.randomaditionaltime ) )
self.rumble_randomaditionaltime = rumblestruct.randomaditionaltime;
else
self.rumble_randomaditionaltime = 1;
areatrigger.radius = self.rumble_radius;
while( 1 )
{
areatrigger waittill( "trigger" );
if( self getspeedmph() == 0 || !self.rumbleon )
{
wait .1;
continue;
}
self PlayRumbleLoopOnEntity( level.vehicle_rumble[ type ].rumble );
// SCRIPTER_MOD: JesseS (5/17/2007) - level.player istouching removed, changed to player array
players = get_players();
player_touching = false;
for (i = 0; i< players.size; i++)
{
if (players[i] istouching( areatrigger ))
{
player_touching = true;
}
}
while( player_touching && self.rumbleon && self getspeedmph() > 0 )
{
earthquake( self.rumble_scale, self.rumble_duration, self.origin, self.rumble_radius ); // scale duration source radius
wait( self.rumble_basetime + randomfloat( self.rumble_randomaditionaltime ) );
// SCRIPTER_MOD: JesseS (5/17/2007) - more co-op player checking
players = get_players();
player_touching = false;
for (i = 0; i < players.size; i++)
{
if (players[i] istouching( areatrigger ))
{
player_touching = true;
}
}
}
println("*** Server vehicle : Stop rumble.");
self StopRumble( level.vehicle_rumble[ type ].rumble );
}
}
vehicle_kill_badplace_forever()
{
self notify( "kill_badplace_forever" );
}
vehicle_badplace()
{
if( !isdefined( self.script_badplace ) )
return;
self endon( "kill_badplace_forever" );
self endon( "death" );
self endon( "delete" );
if( isdefined( level.custombadplacethread ) )
{
self thread [[ level.custombadplacethread ]]();
return;
}
hasturret = isdefined( level.vehicle_hasMainTurret[ self.model ] ) && level.vehicle_hasMainTurret[ self.model ];
bp_duration = .5;
bp_height = 300;
bp_angle_left = 17;
bp_angle_right = 17;
for( ;; )
{
if( !self.script_badplace )
{
// badplace_delete( "tankbadplace" );
while( !self.script_badplace )
wait .5;
}
speed = self getspeedmph();
if( speed <= 0 )
{
wait bp_duration;
continue;
}
if( speed < 5 )
bp_radius = 200;
else if( ( speed > 5 ) && ( speed < 8 ) )
bp_radius = 350;
else
bp_radius = 500;
if( isdefined( self.BadPlaceModifier ) )
bp_radius = ( bp_radius * self.BadPlaceModifier );
// bp_direction = anglestoforward( self.angles );
if( hasturret )
bp_direction = anglestoforward( self gettagangles( "tag_turret" ) );
else
bp_direction = anglestoforward( self.angles );
badplace_arc( "", bp_duration, self.origin, bp_radius * 1.9, bp_height, bp_direction, bp_angle_left, bp_angle_right, "allies", "axis" );
badplace_cylinder( "", bp_duration, self.origin, 200, bp_height, "allies", "axis" );
/// badplace_cylinder( "", bp_duration, self.colidecircle[ 1 ].origin, 200, bp_height, "allies", "axis" );
wait bp_duration + .05;
}
}
/*
// CODER_MOD: JamesS - 5/8/2008 - moved to client scripts
vehicle_treads()
{
if( !isdefined( level.vehicle_treads [ self.vehicletype ] ) )
return;
if( self is_aircraft() )
return;
if( isdefined( level.tread_override_thread ) )
{
self thread [[ level.tread_override_thread ]]( "tag_origin", "back_left", ( 160, 0, 0 ) );
}
// SCRIPTER_MOD: JesseS (10/2/2007): special cased buffalo since we're using wake FX
else if (self.vehicletype == "buffalo" || self.vehicletype == "amtank" || self.vehicletype == "rubber_raft" || self.vehicletype == "jap_ptboat" || self.vehicletype == "jap_shinyo")
{
self thread tread( "tag_wake", "back_left" );
}
else
{
self thread tread( "tag_wheel_back_left", "back_left" );
self thread tread( "tag_wheel_back_right", "back_right" );
}
}
vehicle_kill_treads_forever()
{
self notify( "kill_treads_forever" );
}
tread( tagname, side, relativeOffset )
{
self endon( "death" );
treadfx = treadget( self, side );
self endon( "kill_treads_forever" );
for( ;; )
{
speed = self getspeed();
if( speed == 0 )
{
wait 0.1;
continue;
}
waitTime = ( 1 / speed );
waitTime = ( waitTime * 35 );
if( waitTime < 0.1 )
waitTime = 0.1;
else if( waitTime > 0.3 )
waitTime = 0.3;
wait waitTime;
lastfx = treadfx;
treadfx = treadget( self, side );
if( treadfx != -1 )
{
ang = self getTagAngles( tagname );
forwardVec = anglestoforward( ang );
effectOrigin = self getTagOrigin( tagname );
forwardVec = maps\_utility::vector_multiply( forwardVec, waitTime );
playfx( treadfx, effectOrigin, ( 0, 0, 0 ) - forwardVec );
}
}
}
treadget( vehicle, side )
{
// MikeD (4/3/2008): If self is a boat, then return the water treadfx
if( vehicle is_boat() )
{
return level._vehicle_effect[vehicle.vehicletype]["water"];
}
surface = self getwheelsurface( side );
if( !isdefined( vehicle.vehicletype ) )
{
treadfx = -1;
return treadfx;
}
if( !isdefined( level._vehicle_effect[ vehicle.vehicletype ] ) )
{
println( "no treads setup for vehicle type: ", vehicle.vehicletype );
wait 1;
return - 1;
}
treadfx = level._vehicle_effect[ vehicle.vehicletype ][ surface ];
if( surface == "ice" )
self notify( "iminwater" );
if( !isdefined( treadfx ) )
treadfx = -1;
return treadfx;
}
*/
turret_attack_think()
{
// chad - disable this for now, will eventually handle shooting of missiles at targets
if( self is_aircraft() )
return;
// Nathan - Turrets don't think anymore. Sorry, and your welcome.
thread turret_shoot();
}
isStationary()
{
type = self.vehicletype;
if( isdefined( level.vehicle_isStationary[ type ] ) && level.vehicle_isStationary[ type ] )
return true;
else
return false;
}
turret_shoot()
{
type = self.vehicletype;
self endon( "death" );
self endon( "stop_turret_shoot" );
index = 0;
turrets = [];
// CODER_MOD - JamesS check if vehicle_mainTurrets is defined
if ( IsDefined(level.vehicle_mainTurrets) && level.vehicle_mainTurrets[ level.vtmodel ].size )
{
turrets = getarraykeys( level.vehicle_mainTurrets[ level.vtmodel ] );
}
while( self.health > 0 )
{
self waittill( "turret_fire" );// next game remove this. just a simple fireturret command should do
self notify( "groupedanimevent", "turret_fire" );
if ( ! turrets.size )
self fireWeapon();
else
{
self fireweapon( turrets[ index ] );
index++ ;
if ( index >= turrets.size )
index = 0;
}
}
}
vehicle_weapon_fired()
{
// if no shellshock is specified just get out of here.
// if( !isdefined( level.vehicle_shoot_shock[ self.model ] ) )
// {
// return;
// }
if( !(self vehicle_is_tank() ) )
{
return;
}
self endon( "death" );
while( true )
{
self waittill( "weapon_fired" ); // waits for Code notify when fireWeapon() is called.
// if( isdefined( self.shock_distance ) )
// {
// shock_distance = self.shock_distance;
// }
// else
// {
// shock_distance = 400;
// }
//
// if( isdefined( self.black_distance ) )
// {
// black_distance = self.black_distance;
// }
// else
// {
// black_distance = 800;
// }
owner = self GetVehicleOwner();
if( IsDefined( owner ) && IsPlayer( owner ) )
{
if( !IsDefined( owner.vehicle_shoot_shock_overlay ) )
{
owner.vehicle_shoot_shock_overlay = newClientHudElem( owner );
owner.vehicle_shoot_shock_overlay.x = 0;
owner.vehicle_shoot_shock_overlay.y = 0;
owner.vehicle_shoot_shock_overlay setshader( "black", 640, 480 );
owner.vehicle_shoot_shock_overlay.alignX = "left";
owner.vehicle_shoot_shock_overlay.alignY = "top";
owner.vehicle_shoot_shock_overlay.horzAlign = "fullscreen";
owner.vehicle_shoot_shock_overlay.vertAlign = "fullscreen";
owner.vehicle_shoot_shock_overlay.alpha = 0;
}
owner.vehicle_shoot_shock_overlay.alpha = .5;
owner.vehicle_shoot_shock_overlay fadeOverTime( 0.2 );
owner.vehicle_shoot_shock_overlay.alpha = 0;
}
// players = get_players();
// for( i = 0; i < players.size; i++ )
// {
// player_distance = distance( self.origin, players[i].origin );
//
// if( player_distance > black_distance )
// {
// continue;
// }
//
// if( player_distance > shock_distance )
// {
// continue;
// }
//
// fraction = player_distance / shock_distance;
// time = 4 - ( 3 * fraction );
//
// vehicle_occupants = self GetVehOccupants();
//
// if( IsDefined( players[i] ) )
// {
// if( ( IsDefined( owner ) && players[i] == owner ) || is_in_array( vehicle_occupants, players[i] ) )
// {
// // play different effect here
// }
// else
// {
// players[i] ShellShock( level.vehicle_shoot_shock[ self.model ], time );
//
//// if( IsDefined( level.vehicle_shoot_rumble[ self.model ] ) )
//// {
//// PlayRumbleOnPosition( level.vehicle_shoot_rumble[ self.model ], self.origin + ( 0, 0, 32 ) );
//// }
// }
//
// }
// }
}
}
vehicle_compasshandle()
{
wait_for_first_player();
self endon( "stop_vehicle_compasshandle" );
type = self.vehicletype;
if(!isdefined(level.vehicle_compassicon[ type ]))
return;
if (!level.vehicle_compassicon[ type ])
return;
self endon( "death" );
level.compassradius = int( getdvar( "compassMaxRange" ) );
self.onplayerscompass = false;
// TODO: complain to Code about this feature. I shouldn't have to poll the distance of the tank to remove it from the compass
while ( 1 )
{
player = get_closest_player( self.origin );
if ( distance( self.origin, player.origin ) < level.compassradius )
{
if ( !( self.onplayerscompass ) )
{
self AddVehicleToCompass( maps\_vehicletypes::get_compassTypeForVehicleType( self.vehicletype ) );
self.onplayerscompass = true;
}
}
else
{
if ( self.onplayerscompass )
{
self RemoveVehicleFromCompass();
self.onplayerscompass = false;
}
}
wait .5;
}
}
vehicle_setteam()
{
type = self.vehicletype;
if( !isdefined( self.script_team ) && isdefined( level.vehicle_team[ type ] ) )
self.script_team = level.vehicle_team[ type ];
// guzzo 4/1/08, reverting to cod2 method of assigning vehicle names
//if( isdefined( level.vehicle_hasname[ type ] ) )
// self thread maps\_vehiclenames::get_name();
// only allied vehicles get names
if( isdefined( self.script_team ) && self.script_team == "allies" )
{
// check if it's an aircraft or explicitly set to have no name
no_name = self is_aircraft() || ( IsDefined( self.script_friendname ) && ( self.script_friendname == "" || self.script_friendname == "none" || self.script_friendname == "0" ) );
// only tanks should have names. this replaces cod2's method of using "vehicle.simpletype"
correct_type = ( type == "sherman" || type == "sherman_flame" || type == "t34" || type == "ot34" || type == "see2_t34" || type == "see2_ot34" );
if( !no_name && correct_type )
{
self maps\_vehiclenames::get_name();
}
}
level.vehicles[ self.script_team ] = array_add( level.vehicles[ self.script_team ], self );
}
vehicle_handleunloadevent()
{
self endon( "death" );
type = self.vehicletype;
while( 1 )
{
self waittill( "unload", who );
// setting an unload group unloaded guys resets to "default"
// SCRIPTER_MOD: JesseS (5/14/2007) - took this out "who" for now, seems to be breaking unloadgroups.
//if( isdefined( who ) )
// self.unload_group = who;
// makes ai unload
self notify( "groupedanimevent", "unload" );
// if( isdefined( level.vehicle_hasMainTurret[ self.model ] ) && level.vehicle_hasMainTurret[ self.model ] && riders_check() )
// self clearTurretTarget();
}
}
get_vehiclenode_any_dynamic( target )
{
// the should return undefined
path_start = getvehiclenode( target, "targetname" );
if( !isdefined( path_start ) )
{
path_start = getent( target, "targetname" );
}
else if( is_aircraft() )
{
println( "helicopter node targetname: " + path_start.targetname );
println( "vehicletype: " + self.vehicletype );
assertmsg( "helicopter on vehicle path( see console for info )" );
}
if( !isdefined( path_start ) )
{
path_start = getstruct( target, "targetname" );
}
return path_start;
}
vehicle_resumepathvehicle()
{
if( !self is_aircraft() )
{
self resumespeed( 35 );
return;
}
node = undefined;
if( isdefined( self.currentnode.target ) )
node = get_vehiclenode_any_dynamic( self.currentnode.target );
if( !isdefined( node ) )
return;
vehicle_paths( node );
}
vehicle_landvehicle()
{
self setNearGoalNotifyDist( 2 );
self sethoverparams( 0, 0, 0 );
self cleargoalyaw();
self settargetyaw( flat_angle( self.angles )[ 1 ] );
self setvehgoalpos_wrap( groundpos( self.origin ), 1 );
self waittill( "goal" );
}
setvehgoalpos_wrap( origin, bStop )
{
if( self.health <= 0 )
return;
if( isdefined( self.originheightoffset ) )
origin += ( 0, 0, self.originheightoffset ); // TODO - FIXME: this is temporarily set in the vehicles init_local function working on getting it this requirement removed
self setvehgoalpos( origin, bStop );
}
vehicle_liftoffvehicle( height )
{
if( !isdefined( height ) )
height = 512;
dest = self.origin + ( 0, 0, height );
self setNearGoalNotifyDist( 10 );
self setvehgoalpos_wrap( dest, 1 );
self waittill( "goal" );
}
waittill_stable()
{
// wait for it to level out before unloading
offset = 12;
stabletime = 400;
timer = gettime() + stabletime;
while( isdefined( self ) )
{
if( self.angles[ 0 ] > offset || self.angles [ 0 ] < ( - 1 * offset ) )
timer = gettime() + stabletime;
if( self.angles[ 2 ] > offset || self.angles [ 2 ] < ( - 1 * offset ) )
timer = gettime() + stabletime;
if( gettime() > timer )
break;
wait .05;
}
}
unload_node( node )
{
if ( !isdefined( node.script_flag_wait ) )
{
// going to stop anyway so no need to kill the path
self notify( "newpath" );
}
assert( isdefined( self ) );
// self vehicle_detachfrompath();
pathnode = getnode( node.targetname, "target" );
if( isdefined( pathnode ) && self.riders.size )
for( i = 0; i < self.riders.size; i ++ )
if( isai( self.riders[ i ] ) )
self.riders[ i ] thread maps\_spawner::go_to_node( pathnode );
if( self is_aircraft() )
waittill_stable();
else
self setspeed( 0, 35 );
// self vehicle_to_dummy();
if( isdefined( node.script_noteworthy ) )
if( node.script_noteworthy == "wait_for_flag" )
flag_wait( node.script_flag );
self notify( "unload", node.script_unload );
if ( maps\_vehicle_aianim::riders_unloadable( node.script_unload ) )
self waittill( "unloaded" );
// self dummy_to_vehicle();
// if we want the helis to hang around for bog_b we can do some script_magic here.
// wait 1;
if ( isdefined( node.script_flag_wait ) )
{
return;
}
if( isdefined( self ) )
thread vehicle_resumepathvehicle();
}
move_turrets_here( model )
{
type = self.vehicletype;
if( !isdefined( self.mgturret ) )
return;
if( self.mgturret.size == 0 )
return;
for( i = 0; i < self.mgturret.size; i ++ )
{
self.mgturret[ i ] unlink();
self.mgturret[ i ] linkto( model, level.vehicle_mgturret[ type ][ i ].tag, ( 0, 0, 0 ), ( 0, 0, 0 ) );
}
}
vehicle_pathdetach()
{
self.attachedpath = undefined;
self notify( "newpath" );
self setGoalyaw( flat_angle( self.angles )[ 1 ] );
self setvehgoalpos( self.origin + ( 0, 0, 4 ), 1 );
}
vehicle_to_dummy()
{
// create a dummy model that takes the place of a vehicle, the vehicle gets hidden
assertEx( !isdefined( self.modeldummy ), "Vehicle_to_dummy was called on a vehicle that already had a dummy." );
self.modeldummy = spawn( "script_model", self.origin );
self.modeldummy setmodel( self.model );
self.modeldummy.origin = self.origin;
self.modeldummy.angles = self.angles;
self.modeldummy useanimtree( #animtree );
self hide();
self notify( "animtimer" );
// move rider characters to dummy model
self thread model_dummy_death();
move_riders_here( self.modelDummy );
move_turrets_here( self.modeldummy );
move_ghettotags_here( self.modeldummy );
move_lights_here( self.modeldummy );
move_effects_ent_here( self.modeldummy );
copy_destructable_attachments( self.modeldummy );// destructables are all attach()'d. Little bit different but not too tricky
// flag for various looping functions keeps them from doing isdefined a lot
self.modeldummyon = true;
return self.modeldummy;
}
move_effects_ent_here( model )
{
ent = deathfx_ent();
ent unlink();
ent linkto( model );
}
model_dummy_death()
{
// delete model dummy when the vehicle is deleted.
modeldummy = self.modeldummy;
modeldummy endon( "death" );
while ( isdefined( self ) )
{
self waittill( "death" );
waittillframeend;
}
modeldummy delete();
}
move_lights_here( model )
{
if ( !isdefined( self.lights ) )
return;
keys = getarraykeys( self.lights );
for ( i = 0 ; i < keys.size ; i++ )
{
// if ( ! isdefined( self.lights[ keys[ i ] ] ) )
// continue;//
self.lights[ keys[ i ] ] unlink();
self.lights[ keys[ i ] ] linkto( model, self.lights[ keys[ i ] ].lighttag, ( 0, 0, 0 ), ( 0, 0, 0 ) );
}
}
move_ghettotags_here( model )
{
if ( !isdefined( self.ghettotags ) )
return;
for ( i = 0 ; i < self.ghettotags.size ; i++ )
{
self.ghettotags[ i ] unlink();
self.ghettotags[ i ] linkto( model );
}
}
dummy_to_vehicle()
{
assertEx( isdefined( self.modeldummy ), "Tried to turn a vehicle from a dummy into a vehicle. Can only be called on vehicles that have been turned into dummies with vehicle_to_dummy." );
if( self is_aircraft() )
self.modeldummy.origin = self gettagorigin( "tag_ground" );
else
{
self.modeldummy.origin = self.origin;
self.modeldummy.angles = self.angles;
}
self show();
// move rider characters back to the vehicle
move_riders_here( self );
move_turrets_here( self );
move_lights_here( self );
move_effects_ent_here( self );
// flag for various looping functions keeps them from doing isdefined a lot
self.modeldummyon = false;
self.modeldummy delete();
self.modeldummy = undefined;
// helicopters do dust kickup fx
if ( self hasHelicopterDustKickup() )
{
if(!level.clientscripts)
{
self notify( "stop_kicking_up_dust" );
self thread aircraft_dust_kickup();
}
}
return self.modeldummy;
}
move_riders_here( base )
{
if ( !isdefined( self.riders ) )
return;
riders = self.riders;
// move rider characters to their new location
for( i = 0; i < riders.size; i ++ )
{
if( !isdefined( riders[ i ] ) )
continue;
guy = riders[ i ];
guy unlink();
animpos = maps\_vehicle_aianim::anim_pos( self, guy.pos );
guy linkto( base, animpos.sittag, ( 0, 0, 0 ), ( 0, 0, 0 ) );
if( isai( guy ) )
guy teleport( base gettagorigin( animpos.sittag ) );
else
guy.origin = base gettagorigin( animpos.sittag );
}
}
setup_targetname_spawners()
{
level.vehicle_targetname_array = [];
vehicles = array_combine( getentarray( "script_vehicle", "classname" ), get_script_modelvehicles() );
/*
// disabling this until we can get a spawner checkbox on vehicles
highestGroup = 0;
// get the highest script_vehicleSpawnGroup in use
for( i = 0; i < vehicles.size; i ++ )
{
if( !isdefined( vehicles[ i ].script_vehicleSpawnGroup ) )
continue;
if( vehicles[ i ].script_vehicleSpawnGroup > highestGroup )
{
highestGroup = vehicles[ i ].script_vehicleSpawnGroup;
}
}
*/
for( i = 0; i < vehicles.size; i ++ )
{
if( !isdefined( vehicles[ i ].targetname ) )
continue;
if( !isdefined( vehicles[ i ].script_vehicleSpawnGroup ) )
{
// vehicles that have no script_vehiclespawngroup get assigned one, if they have a targetname
// highestGroup ++ ;
// vehicles[ i ].script_vehicleSpawnGroup = highestGroup;
continue;
}
targetname = vehicles[ i ].targetname;
spawngroup = vehicles[ i ].script_vehicleSpawnGroup;
if( !isdefined( level.vehicle_targetname_array[ targetname ] ) )
level.vehicle_targetname_array[ targetname ] = [];
level.vehicle_targetname_array[ targetname ][ spawngroup ] = true;
}
}
spawn_vehicles_from_targetname( name )
{
// spawns an array of vehicles that all have the specified targetname in the editor,
// but are deleted at runtime
assertEx( isdefined( level.vehicle_targetname_array[ name ] ), "No vehicle spawners had targetname " + name );
array = level.vehicle_targetname_array[ name ];
keys = getArrayKeys( array );
vehicles = [];
for( i = 0; i < keys.size; i ++ )
{
vehicleArray = scripted_spawn( keys[ i ] );
vehicles = array_combine( vehicles, vehicleArray );
}
return vehicles;
}
spawn_vehicle_from_targetname( name )
{
// spawns 1 vehicle and makes sure it gets 1
vehicleArray = spawn_vehicles_from_targetname( name );
assertEx( vehicleArray.size == 1, "Tried to spawn a vehicle from targetname " + name + " but it returned " + vehicleArray.size + " vehicles, instead of 1" );
return vehicleArray[ 0 ];
}
spawn_vehicle_from_targetname_and_drive( name )
{
// spawns 1 vehicle and makes sure it gets 1
vehicleArray = spawn_vehicles_from_targetname( name );
assertEx( vehicleArray.size == 1, "Tried to spawn a vehicle from targetname " + name + " but it returned " + vehicleArray.size + " vehicles, instead of 1" );
thread gopath( vehicleArray[ 0 ] );
return vehicleArray[ 0 ];
}
spawn_vehicles_from_targetname_and_drive( name )
{
// spawns 1 vehicle and makes sure it gets 1
vehicleArray = spawn_vehicles_from_targetname( name );
for( i = 0; i < vehicleArray.size; i ++ )
thread goPath( vehicleArray[ i ] );
return vehicleArray;
}
aircraft_dust_kickup( model )
{
self endon( "death_finished" );
self endon( "stop_kicking_up_dust" );
assert( isdefined( self.vehicletype ) );
maxHeight = 1200;
minHeight = 350;
slowestRepeatWait = 0.15;
fastestRepeatWait = 0.05;
numFramesPerTrace = 3;
doTraceThisFrame = numFramesPerTrace;
defaultRepeatRate = 1.0;
repeatRate = defaultRepeatRate;
trace = undefined;
d = undefined;
trace_ent = self;
if ( isdefined( model ) )
trace_ent = model;
while( isdefined( self ) )
{
if( repeatRate <= 0 )
repeatRate = defaultRepeatRate;
wait repeatRate;
if( !isdefined( self ) )
return;
doTraceThisFrame -- ;
// prof_begin( "helicopter_dust_kickup" );
if( doTraceThisFrame <= 0 )
{
doTraceThisFrame = numFramesPerTrace;
trace = bullettrace( trace_ent.origin, trace_ent.origin - ( 0, 0, 100000 ), false, trace_ent );
/*
trace[ "entity" ]
trace[ "fraction" ]
trace[ "normal" ]
trace[ "position" ]
trace[ "surfacetype" ]
*/
d = distance( trace_ent.origin, trace[ "position" ] );
repeatRate = ( ( d - minHeight ) / ( maxHeight - minHeight ) ) * ( slowestRepeatWait - fastestRepeatWait ) + fastestRepeatWait;
}
if( !isdefined( trace ) )
continue;
assert( isdefined( d ) );
if( d > maxHeight )
{
repeatRate = defaultRepeatRate;
continue;
}
if( isdefined( trace[ "entity" ] ) )
{
repeatRate = defaultRepeatRate;
continue;
}
if( !isdefined( trace[ "position" ] ) )
{
repeatRate = defaultRepeatRate;
continue;
}
if( !isdefined( trace[ "surfacetype" ] ) )
trace[ "surfacetype" ] = "dirt";
assertEx( isdefined( level._vehicle_effect[ self.vehicletype ] ), self.vehicletype + " vehicle script hasn't run _tradfx properly" );
assertEx( isdefined( level._vehicle_effect[ self.vehicletype ][ trace[ "surfacetype" ] ] ), "UNKNOWN SURFACE TYPE: " + trace[ "surfacetype" ] );
// prof_end( "helicopter_dust_kickup" );
if( level._vehicle_effect[ self.vehicletype ][ trace[ "surfacetype" ] ] != -1 )
playfx( level._vehicle_effect[ self.vehicletype ][ trace[ "surfacetype" ] ], trace[ "position" ] );
}
}
tank_crush( crushedVehicle, endNode, tankAnim, truckAnim, animTree, soundAlias )
{
// Chad G's tank crushing vehicle script. Self corrects for node positioning errors.
assert( isdefined( crushedVehicle ) );
assert( isdefined( endNode ) );
assert( isdefined( tankAnim ) );
assert( isdefined( truckAnim ) );
assert( isdefined( animTree ) );
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
// Create an animatable tank and move the real tank to the next path and store required info
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
animatedTank = vehicle_to_dummy();
self setspeed( 7, 5, 5 );
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
// Total time for animation, and correction and uncorrection times
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
animLength = getanimlength( tankAnim );
move_to_time = ( animLength / 3 );
move_from_time = ( animLength / 3 );
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
// Node information used for calculating both starting and ending points for the animation
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
// get node vecs
node_origin = crushedVehicle.origin;
node_angles = crushedVehicle.angles;
node_forward = anglesToForward( node_angles );
node_up = anglesToUp( node_angles );
node_right = anglesToRight( node_angles );
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
// Calculate Starting Point for the animation from crushedVehicle and create the dummy
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
// get anim starting point origin and angle
anim_start_org = getStartOrigin( node_origin, node_angles, tankAnim );
anim_start_ang = getStartAngles( node_origin, node_angles, tankAnim );
// get anim starting point vecs
animStartingVec_Forward = anglesToForward( anim_start_ang );
animStartingVec_Up = anglesToUp( anim_start_ang );
animStartingVec_Right = anglesToRight( anim_start_ang );
// get tank vecs
tank_Forward = anglesToForward( animatedTank.angles );
tank_Up = anglesToUp( animatedTank.angles );
tank_Right = anglesToRight( animatedTank.angles );
// spawn dummy with appropriate offset
offset_Vec = ( node_origin - anim_start_org );
offset_Forward = vectorDot( offset_Vec, animStartingVec_Forward );
offset_Up = vectorDot( offset_Vec, animStartingVec_Up );
offset_Right = vectorDot( offset_Vec, animStartingVec_Right );
dummy = spawn( "script_origin", animatedTank.origin );
dummy.origin += vector_multiply( tank_Forward, offset_Forward );
dummy.origin += vector_multiply( tank_Up, offset_Up );
dummy.origin += vector_multiply( tank_Right, offset_Right );
// set dummy angles to reflect the different in animation starting angles and the tanks actual angles
offset_Vec = anglesToForward( node_angles );
offset_Forward = vectorDot( offset_Vec, animStartingVec_Forward );
offset_Up = vectorDot( offset_Vec, animStartingVec_Up );
offset_Right = vectorDot( offset_Vec, animStartingVec_Right );
dummyVec = vector_multiply( tank_Forward, offset_Forward );
dummyVec += vector_multiply( tank_Up, offset_Up );
dummyVec += vector_multiply( tank_Right, offset_Right );
dummy.angles = vectorToAngles( dummyVec );
// -- -- -- -- -- -- -- -- -- -- -
// Debug Lines
// -- -- -- -- -- -- -- -- -- -- -
/#
if( getdvar( "debug_tankcrush" ) == "1" )
{
// line to where tank1 is
thread draw_line_from_ent_for_time( get_players()[0], animatedTank.origin, 1, 0, 0, animLength / 2 );
// line to where tank1 SHOULD be
thread draw_line_from_ent_for_time( get_players()[0], anim_start_org, 0, 1, 0, animLength / 2 );
// line to the dummy
thread draw_line_from_ent_to_ent_for_time( get_players()[0], dummy, 0, 0, 1, animLength / 2 );
}
#/
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
// Animate the animatable tank and self correct into the crushed vehicle
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
if( isdefined( soundAlias ) )
level thread play_sound_in_space( soundAlias, node_origin );
animatedTank linkto( dummy );
crushedVehicle useAnimTree( animTree );
animatedTank useAnimTree( animTree );
assert( isdefined( level._vehicle_effect[ "tankcrush" ][ "window_med" ] ) );
assert( isdefined( level._vehicle_effect[ "tankcrush" ][ "window_large" ] ) );
crushedVehicle thread tank_crush_fx_on_tag( "tag_window_left_glass_fx", level._vehicle_effect[ "tankcrush" ][ "window_med" ], "veh_glass_break_small", 0.2 );
crushedVehicle thread tank_crush_fx_on_tag( "tag_window_right_glass_fx", level._vehicle_effect[ "tankcrush" ][ "window_med" ], "veh_glass_break_small", 0.4 );
crushedVehicle thread tank_crush_fx_on_tag( "tag_windshield_back_glass_fx", level._vehicle_effect[ "tankcrush" ][ "window_large" ], "veh_glass_break_large", 0.7 );
crushedVehicle thread tank_crush_fx_on_tag( "tag_windshield_front_glass_fx", level._vehicle_effect[ "tankcrush" ][ "window_large" ], "veh_glass_break_large", 1.5 );
crushedVehicle animscripted( "tank_crush_anim", node_origin, node_angles, truckAnim );
animatedTank animscripted( "tank_crush_anim", dummy.origin, dummy.angles, tankAnim );
// animatedTank VehForceMaterialSpeed( true, 20 );
dummy moveTo( node_origin, move_to_time, ( move_to_time / 2 ), ( move_to_time / 2 ) );
dummy rotateTo( node_angles, move_to_time, ( move_to_time / 2 ), ( move_to_time / 2 ) );
wait move_to_time;
animLength -= move_to_time;
animLength -= move_from_time;
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -
// Tank plays animation in the exact correct location for a while
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -
wait animLength;
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -
// Calculate Ending Point for the animation from crushedVehicle
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -
// get anim ending point origin and angle
// anim_end_org = anim_start_org + getMoveDelta( tankAnim, 0, 1 );
temp = spawn( "script_model", ( anim_start_org ) );
temp.angles = anim_start_ang;
anim_end_org = temp localToWorldCoords( getMoveDelta( tankAnim, 0, 1 ) );
anim_end_ang = anim_start_ang + ( 0, getAngleDelta( tankAnim, 0, 1 ), 0 );
temp delete();
// get anim ending point vecs
animEndingVec_Forward = anglesToForward( anim_end_ang );
animEndingVec_Up = anglesToUp( anim_end_ang );
animEndingVec_Right = anglesToRight( anim_end_ang );
// get ending tank pos vecs
attachPos = self getAttachPos( endNode );
tank_Forward = anglesToForward( attachPos[ 1 ] );
tank_Up = anglesToUp( attachPos[ 1 ] );
tank_Right = anglesToRight( attachPos[ 1 ] );
// see what the dummy's final origin will be
offset_Vec = ( node_origin - anim_end_org );
offset_Forward = vectorDot( offset_Vec, animEndingVec_Forward );
offset_Up = vectorDot( offset_Vec, animEndingVec_Up );
offset_Right = vectorDot( offset_Vec, animEndingVec_Right );
dummy.final_origin = attachPos[ 0 ];
dummy.final_origin += vector_multiply( tank_Forward, offset_Forward );
dummy.final_origin += vector_multiply( tank_Up, offset_Up );
dummy.final_origin += vector_multiply( tank_Right, offset_Right );
// set dummy angles to reflect the different in animation starting angles and the tanks actual angles
offset_Vec = anglesToForward( node_angles );
offset_Forward = vectorDot( offset_Vec, animEndingVec_Forward );
offset_Up = vectorDot( offset_Vec, animEndingVec_Up );
offset_Right = vectorDot( offset_Vec, animEndingVec_Right );
dummyVec = vector_multiply( tank_Forward, offset_Forward );
dummyVec += vector_multiply( tank_Up, offset_Up );
dummyVec += vector_multiply( tank_Right, offset_Right );
dummy.final_angles = vectorToAngles( dummyVec );
// -- -- -- -- -- -- -- -- -- -- -
// Debug Lines
// -- -- -- -- -- -- -- -- -- -- -
// SCRIPTER_MOD: JesseS (8/10/2007): level.player referneces removed
if( getdvar( "debug_tankcrush" ) == "1" )
{
// line to where tank2 is
thread draw_line_from_ent_for_time( get_players()[0], self.origin, 1, 0, 0, animLength / 2 );
// line to where tank2 SHOULD be
thread draw_line_from_ent_for_time( get_players()[0], anim_end_org, 0, 1, 0, animLength / 2 );
// line to the dummy
thread draw_line_from_ent_to_ent_for_time( get_players()[0], dummy, 0, 0, 1, animLength / 2 );
}
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -
// Tank uncorrects to the real location of the tank on the spline
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -
dummy moveTo( dummy.final_origin, move_from_time, ( move_from_time / 2 ), ( move_from_time / 2 ) );
dummy rotateTo( dummy.final_angles, move_from_time, ( move_from_time / 2 ), ( move_from_time / 2 ) );
wait move_from_time;
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
// Tank is done animating now, remove the animatable tank and show the real one( they should be perfectly aligned now )
// -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
self dontInterpolate();
self attachPath( endNode );
dummy_to_vehicle();
}
tank_crush_fx_on_tag( tagName, fxName, soundAlias, startDelay )
{
if( isdefined( startDelay ) )
wait startDelay;
playfxontag( fxName, self, tagName );
if( isdefined( soundAlias ) )
self thread play_sound_on_tag( soundAlias, tagName );
}
loadplayer( player, position, animfudgetime )
{
/#
if( getdvar( "fastrope_arms" ) == "" )
setdvar( "fastrope_arms", "0" );
#/
if( !isdefined( animfudgetime ) )
animfudgetime = 0;
assert( isdefined( self.riders ) );
assert( self.riders.size );
guy = undefined;
for( i = 0; i < self.riders.size; i ++ )
{
if( self.riders[ i ].pos == position )
{
guy = self.riders[ i ];
guy.drone_delete_on_unload = true;
guy.playerpiggyback = true;
break;
}
}
assertex( !isai( guy ), "guy in position of player needs to have script_drone set, use script_startingposition ans script drone in your map" );
assert( isdefined( guy ) );
thread show_rigs( position );
animpos = maps\_vehicle_aianim::anim_pos( self, position );
// guy stopanimscripted();
// guy stopuseanimtree();
guy notify( "newanim" );
guy detachall();
// guy setmodel( "" );
guy setmodel( "fastrope_arms" );
guy useanimtree( animpos.player_animtree );
thread maps\_vehicle_aianim::guy_idle( guy, position );
// playerlinktodelta( <linkto entity> , <tag> , <viewpercentag fraction> , <right arc> , <left arc> , <top arc> , <bottom arc> )
// SCRIPTER_MOD: JesseS (8/10/2007): passing in player now, use player instead of level.player
player playerlinktodelta( guy, "tag_player", 1.0, 70, 70, 90, 90 );
// level.player setplayerangles( guy gettagangles( "tag_player" ) );
// level.player allowcrouch( false );
// level.player allowprone( false );
// level.player allowstand( true );
guy hide();
animtime = getanimlength( animpos.getout );
animtime -= animfudgetime;
self waittill( "unload" );
/#
if( getdvar( "fastrope_arms" ) != "0" )
guy show();
#/
player disableweapons();
// guy waittill( "jumpedout" );
guy notsolid();
wait animtime;
player unlink();
player enableweapons();
// level.player allowcrouch( true );
// level.player allowprone( true );
}
show_rigs( position )
{
wait .01;
self thread maps\_vehicle_aianim::getout_rigspawn( self, position ); // spawn the getoutrig for this position
if( !self.riders.size )
return;
for( i = 0; i < self.riders.size; i ++ )
self thread maps\_vehicle_aianim::getout_rigspawn( self, self.riders[ i ].pos );
}
vehicle_deleteme()
{
self delete();
}
turret_deleteme( turret )
{
if ( isdefined( self ) )
if ( isdefined( turret.deletedelay ) )
wait turret.deletedelay;
turret delete();
}
wheeldirectionchange( direction )
{
if( direction <= 0 )
self.wheeldir = 0;
else
self.wheeldir = 1;
}
maingun_FX()
{
if( !isdefined( level.vehicle_deckdust[ self.model ] ) )
return;
self endon( "death" );
while( true )
{
self waittill( "weapon_fired" ); // waits for Code notify when fireWeapon() is called.
playfxontag( level.vehicle_deckdust[ self.model ], self, "tag_engine_exhaust" );
barrel_origin = self gettagorigin( "tag_flash" );
ground = physicstrace( barrel_origin, barrel_origin + ( 0, 0, -128 ) );
physicsExplosionSphere( ground, 192, 100, 1 );
}
}
playTankExhaust()
{
if( !isdefined( level.vehicle_exhaust[ self.model ] ) )
return;
exhaustDelay = 0.1;
for( ;; )
{
if( !isdefined( self ) )
return;
if( !isalive( self ) )
return;
playfxontag( level.vehicle_exhaust[ self.model ], self, "tag_engine_left" );
playfxontag( level.vehicle_exhaust[ self.model ], self, "tag_engine_right" );
wait exhaustDelay;
}
}
/*
=============
///ScriptDocBegin
"Name: build_light( <model> , <name> , <tag> , <effect> , <group> , <delay> )"
"Summary: "
"Module: Entity"
"CallOn: An entity"
"MandatoryArg: <param1> : "
"MandatoryArg: <name> : "
"MandatoryArg: <tag> : "
"MandatoryArg: <effect> : "
"MandatoryArg: <group> : "
"MandatoryArg: <delay> : "
"Example: "
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_light( model, name, tag, effect, group, delay )
{
if( !isdefined( level.vehicle_lights ) )
level.vehicle_lights = [];
struct = spawnstruct();
struct.name = name;
struct.tag = tag;
struct.delay = delay;
struct.effect = loadfx( effect );
level.vehicle_lights[ model ][ name ] = struct;
group_light( model, name, "all" );
if( isdefined( group ) )
group_light( model, name, group );
}
group_light( model, name, group )
{
if( !isdefined( level.vehicle_lights_group ) )
level.vehicle_lights_group = [];
if( !isdefined( level.vehicle_lights_group[ model ] ) )
level.vehicle_lights_group[ model ] = [];
if( !isdefined( level.vehicle_lights_group[ model ][ group ] ) )
level.vehicle_lights_group[ model ][ group ] = [];
level.vehicle_lights_group[ model ][ group ][ level.vehicle_lights_group[ model ][ group ].size ] = name;
}
lights_on( group )
{
groups = strtok( group, " " );
array_levelthread( groups, ::lights_on_internal );
}
lights_delayfxforframe()
{
level notify ( "new_lights_delayfxforframe");
level endon ("new_lights_delayfxforframe");
if( !isdefined( level.fxdelay ) )
level.fxdelay = 0;
level.fxdelay += randomfloatrange(0.2, 0.4);
if( level.fxdelay > 2 )
level.fxdelay = 0;
wait 0.05;
level.fxdelay = undefined;
}
lights_on_internal( group )
{
level.lastlighttime = gettime();
if( !isdefined( group ) )
group = "all";
if ( ! isdefined( level.vehicle_lights_group[ self.model ] )
|| !isdefined( level.vehicle_lights_group[ self.model ][ group ] )
)
return;
// self endon( "death" );
thread lights_delayfxforframe();
if( !isdefined( self.lights ) )
self.lights = [];
lights = level.vehicle_lights_group[ self.model ][ group ];
count = 0;
delayoffsetter = [];
for( i = 0; i < lights.size; i ++ )
{
if( isdefined( self.lights[ lights[ i ] ] ) )
continue; // light is already on
template = level.vehicle_lights[ self.model ][ lights[ i ] ];
if( isdefined( template.delay ) )
delay = template.delay;
else
delay = 0;
while( isdefined(delayoffsetter[""+delay]) )
delay+= .05; //don't start these on the same frame.
delay+=level.fxdelay;
delayoffsetter[""+delay] = true;
// Nate - effects can only be stopped on an object when it is deleted. therefor we fake it by spawning a model,
// this is expensive so it's required that people turn it on.
if( isdefined( self.script_light_toggle ) && self.script_light_toggle )
{
light = spawn( "script_model", ( 0, 0, 0 ) );
light setmodel( "fx" );
light hide();
light linkto( self, template.tag, ( 0, 0, 0 ), ( 0, 0, 0 ) );
thread playfxontag_delay( template.effect, light, "Trim_Char_F_1_1", delay );
}
else
{
light = self;
thread playfxontag_delay( template.effect, light, template.tag, delay );
}
light.lighttag = template.tag;// just for dummy transfering lights
self.lights[ lights[ i ] ] = light;
if ( !isdefined( self ) )
break;
}
level.fxdelay = false;
}
playfxontag_delay( effect, entity, tag, delay )
{
entity endon( "death" );
wait delay;
playfxontag( effect, entity, tag );
}
deathfx_ent()
{
if ( !isdefined( self.deathfx_ent ) )
{
ent = spawn( "script_model", ( 0, 0, 0 ) );
emodel = get_dummy();
ent setmodel( self.model );
ent.origin = emodel.origin;
ent.angles = emodel.angles;
ent notsolid();
ent hide();
ent linkto( emodel );
self.deathfx_ent = ent;
}
else
self.deathfx_ent setmodel( self.model );
return self.deathfx_ent;
}
lights_off( group )
{
groups = strtok( group, " " );
array_levelthread( groups, ::lights_off_internal );
}
lights_kill()
{
if ( !isdefined( self.lights ) )
return;
keys = getarraykeys( self.lights );
for ( i = 0; i < keys.size; i++ )
{
if ( self.lights[ keys[ i ] ] == self )
return;
self.lights[ keys[ i ] ] delete();
if( isdefined( self ) )
self.lights[ keys[ i ] ] = undefined;
}
}
lights_off_internal( group )
{
if( !isdefined( group ) )
group = "all";
assertEX( isdefined( self.script_light_toggle ) && self.script_light_toggle, "can't turn off lights on a vehicle without script_light_toggle" );
if( !isdefined( self.lights ) )
return;
if( !isdefined( level.vehicle_lights_group[ self.model ][ group ] ) )
{
println( "vehicletype: " + self.vehicletype );
println( "light group: " + group );
assertmsg( "lights not defined for this vehicle( see console" );
}
lights = level.vehicle_lights_group[ self.model ][ group ];
for( i = 0; i < lights.size; i ++ )
{
self.lights[ lights[ i ] ] delete();
self.lights[ lights[ i ] ] = undefined;
}
}
/*
=============
///ScriptDocBegin
"Name: build_deathmodel( <model> , <deathmodel> )"
"Summary: called in individual vehicle file - assigns death model to vehicles with this model. "
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"MandatoryArg: <model> : name of model to associate death model"
"OptionalArg: <deathmodel> : name of death model to be associated with model"
"OptionalArg: <swapDelay> : number of seconds to wait before setting the death model after the vehicle dies. Defaults to 0"
"Example: build_deathmodel( "bmp", "bmp_destroyed" );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_deathmodel( model, deathmodel, swapDelay, do_build_death )
{
// If we've explicitly set up a vehicle to not build the deathmodel, set it to the regular model
if (isdefined(do_build_death) && !do_build_death)
{
deathmodel = model;
}
if( model != level.vtmodel )
return;
if( !isdefined( deathmodel ) )
deathmodel = model;
precachemodel( model );
precachemodel( deathmodel );
level.vehicle_deathmodel[ model ] = deathmodel;
if( !isdefined( swapDelay ) )
swapDelay = 0;
level.vehicle_deathmodel_delay[ model ] = swapDelay;
}
/*
=============
///ScriptDocBegin
"Name: build_shoot_shock( <shock> )"
"Summary: called in individual vehicle file - assigns shock file to be played when main cannon on a tank fires "
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"MandatoryArg: <shock> : the shock asset"
"Example: build_shoot_shock( "tankblast" );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_shoot_shock( shock )
{
// shock script uses "black" hudelem or something. I don't know . Just had to move it out of _m1a1.gsc
precacheShader( "black" );
precacheshellshock( shock );
// level.vehicle_shoot_shock[ level.vtmodel ] = shock; // Moved to CSC
}
/*
=============
///ScriptDocBegin
"Name: build_shoot_rumble( <rumble> )"
"Summary: called in individual vehicle file - assigns a rumble file to be played when main cannon on a tank fires "
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"MandatoryArg: <rumble> : the rumble gdt entry"
"Example: build_shoot_rumble( "tank_fire" );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_shoot_rumble( rumble )
{
PrecacheRumble( rumble );
// level.vehicle_shoot_rumble[ level.vtmodel ] = rumble; // Moved to CSC
}
/*
=============
///ScriptDocBegin
"Name: build_drive( <forward> , <reverse> , <normalspeed> , <rate> )"
"Summary: called in individual vehicle file - assigns animations to be used on vehicles"
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"MandatoryArg: <forward> : forward animation"
"OptionalArg: <reverse> : reverse animation"
"OptionalArg: <normalspeed> : speed at which animation will be played at 1x defaults to 10mph"
"OptionalArg: <rate> : scales speed of animation( please only use this for testing )"
"Example: build_drive( %abrams_movement, %abrams_movement_backwards, 10 );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_drive( forward, reverse, normalspeed, rate )
{
if( !isdefined( normalspeed ) )
normalspeed = 10;
level.vehicle_DriveIdle[ level.vtmodel ] = forward;
if( isdefined( reverse ) )
level.vehicle_DriveIdle_r[ level.vtmodel ] = reverse;
level.vehicle_DriveIdle_normal_speed[ level.vtmodel ] = normalspeed;
if( isdefined( rate ) )
level.vehicle_DriveIdle_animrate[ level.vtmodel ] = rate;
}
/*
=============
///ScriptDocBegin
"Name: build_template( <type> , <model> , <typeoverride> )"
"Summary: called in individual vehicle file - mandatory to call this in all vehicle files at the top!"
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"MandatoryArg: <type> : vehicle type to set"
"MandatoryArg: <model> : model to set( this is usually generated by the level script )"
"OptionalArg: <typeoverride> : this overrides the type, used for copying a vehicle script"
"Example: build_template( "bmp", model, type );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_template( type, model, typeoverride )
{
if( isdefined( typeoverride ) )
type = typeoverride;
precachevehicle( type );
if( !isdefined( level.vehicle_death_fx ) )
level.vehicle_death_fx = [];
if( !isdefined( level.vehicle_death_fx[ type ] ) )
level.vehicle_death_fx[ type ] = []; // can have overrides
level.vehicle_compassicon[ type ] = false;
level.vehicle_team[ type ] = "axis";
level.vehicle_life[ type ] = 999;
level.vehicle_hasMainTurret[ model ] = false;
level.vehicle_mainTurrets[ model ] = [];
level.vtmodel = model;
level.vttype = type;
}
/*
=============
///ScriptDocBegin
"Name: build_exhaust( <exhaust_effect_str> )"
"Summary: called in individual vehicle file - assign an exhaust effect to this vehicle!"
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"MandatoryArg: <exhaust_effect_str> : exhaust effect in string format"
"Example: build_exhaust( "distortion / abrams_exhaust" );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_exhaust( effect )
{
level.vehicle_exhaust[ level.vtmodel ] = loadfx( effect );
}
/*
=============
///ScriptDocBegin
"Name: build_treadfx()"
"Summary: called in individual vehicle file - enables treadfx"
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"Example: build_treadfx();"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_treadfx( type )
{
if( ! isdefined( type ) )
type = level.vttype;
maps\_treadfx::main( type );
}
/*
=============
///ScriptDocBegin
"Name: build_team( <team> )"
"Summary: called in individual vehicle file - sets team"
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"MandatoryArg: <team> : team"
"Example: build_team( "allies" );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_team( team )
{
level.vehicle_team[ level.vttype ] = team;
}
/*
=============
///ScriptDocBegin
"Name: build_mainturret( <firetime> , <tag1> , <tag2> , <tag3> , <tag4> )"
"Summary: called in individual vehicle file - enables main( cannon ) turret"
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"OptionalArg: <tag1> : additional tags to fire from"
"OptionalArg: <tag2> : additional tags to fire from"
"OptionalArg: <tag3> : additional tags to fire from"
"OptionalArg: <tag4> : additional tags to fire from"
"Example: build_mainturret();"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_mainturret( tag1, tag2, tag3, tag4 )
{
level.vehicle_hasMainTurret[ level.vtmodel ] = true;
if ( isdefined( tag1 ) )
level.vehicle_mainTurrets[ level.vtmodel ][ tag1 ] = true;
if ( isdefined( tag2 ) )
level.vehicle_mainTurrets[ level.vtmodel ][ tag2 ] = true;
if ( isdefined( tag3 ) )
level.vehicle_mainTurrets[ level.vtmodel ][ tag3 ] = true;
if ( isdefined( tag4 ) )
level.vehicle_mainTurrets[ level.vtmodel ][ tag4 ] = true;
}
build_bulletshield( bShield )
{
assert( isdefined( bShield ) );
level.vehicle_bulletshield[ level.vttype ] = bShield;
}
/*
=============
///ScriptDocBegin
"Name: build_aianims( <aithread> , <vehiclethread> )"
"Summary: called in individual vehicle file - set threads for ai animation and vehicle animation assignments"
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"MandatoryArg: <aithread> : ai thread"
"OptionalArg: <vehiclethread> : vehicle thread"
"Example: build_aianims( ::setanims, ::set_vehicle_anims );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_aianims( aithread, vehiclethread )
{
level.vehicle_aianims[ level.vttype ] = [[ aithread ]]();
if( isdefined( vehiclethread ) )
level.vehicle_aianims[ level.vttype ] = [[ vehiclethread ]]( level.vehicle_aianims[ level.vttype ] );
}
/*
=============
///ScriptDocBegin
"Name: build_frontarmor( <armor> )"
"Summary: called in individual vehicle file - sets percentage of health to regen on attacks from the front"
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"MandatoryArg: <armor> : ercentage of health to regen on attacks from the front"
"Example: build_frontarmor( .33 );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_frontarmor( armor )
{
level.vehicle_frontarmor [ level.vttype ] = armor;
}
/*
=============
///ScriptDocBegin
"Name: build_crashanim( <modelsthread> )"
"Summary: called in individual vehicle file - thread for building attached models( ropes ) with animation"
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"MandatoryArg: <modelsthread> : thread"
"Example: build_attach_models( ::set_attached_models );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_attach_models( modelsthread )
{
level.vehicle_attachedmodels[ level.vttype ] = [[ modelsthread ]]();;
}
/*
=============
///ScriptDocBegin
"Name: build_unload_groups( <unloadgroupsthread> )"
"Summary: called in individual vehicle file - thread for building unload groups"
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"MandatoryArg: <modelsthread> : thread"
"Example: build_unload_groups( ::Unload_Groups );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_unload_groups( unloadgroupsthread )
{
level.vehicle_unloadgroups[ level.vttype ] = [[ unloadgroupsthread ]]();
}
/*
=============
///ScriptDocBegin
"Name: build_life( <health> , <minhealth> , <maxhealth> , )"
"Summary: called in individual vehicle file - sets health for vehicles"
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"MandatoryArg: <health> : health"
"OptionalArg: <minhealth> : randomly chooses between the minhealth, maxhealth"
"OptionalArg: <maxhealth> : randomly chooses between the minhealth, maxhealth"
"Example: build_life( 999, 500, 1500 );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_life( health, minhealth, maxhealth )
{
level.vehicle_life[ level.vttype ] = health;
level.vehicle_life_range_low[ level.vttype ] = minhealth;
level.vehicle_life_range_high[ level.vttype ] = maxhealth;
}
/*
=============
///ScriptDocBegin
"Name: build_compassicon()"
"Summary: called in individual vehicle file - enables vehicle on the compass( they use tank icons )"
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"Example: build_compassicon();"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_compassicon()
{
level.vehicle_compassicon[ level.vttype ] = true;
}
/*
=============
///ScriptDocBegin
"Name: build_deckdust( <effect> )"
"Summary: called in individual vehicle file - sets a deckdust effect on a vehicle?"
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"MandatoryArg: <effect> : effect to be assigned as deckdust"
"Example: build_deckdust( "dust / abrams_desk_dust" );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_deckdust( effect )
{
level.vehicle_deckdust[ level.vtmodel ] = loadfx( effect );
}
/*
=============
///ScriptDocBegin
"Name: build_vehiclewalk( <walk num> )"
"Summary: called in individual vehicle file - this sets the max numer of walkers the vehicle has
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"MandatoryArg: <walk num> : max number of walkers this vehicle supports"
"Example: build_localinit( 6 );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_vehiclewalk( num_walkers )
{
level.vehicle_walkercount[ level.vttype ] = num_walkers;
}
/*
=============
///ScriptDocBegin
"Name: build_localinit( <init_thread> )"
"Summary: called in individual vehicle file - mandatory for all vehicle files, this sets the individual init thread for those special sequences, it is also used to determine that a vehicle is being precached or not"
"Module: vehicle_build( vehicle.gsc )"
"CallOn: "
"MandatoryArg: <init_thread> : local thread to the vehicle to be called when it spawns"
"Example: build_localinit( ::init_local );"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
build_localinit( init_thread )
{
level.vehicleInitThread[ level.vttype ][ level.vtmodel ] = init_thread;
}
get_from_spawnstruct( target )
{
return getstruct( target, "targetname" );
}
get_from_entity( target )
{
return getent( target, "targetname" );
}
get_from_spawnstruct_target( target )
{
return getstruct( target, "target" );
}
get_from_entity_target( target )
{
return getent( target, "target" );
}
get_from_vehicle_node( target )
{
return getvehiclenode( target, "targetname" );
}
set_lookat_from_dest( dest )
{
viewTarget = getent( dest.script_linkto, "script_linkname" );
// temp fix for an issue with Hunted
// I use script_linktos as well but for some other purpose.
// I don't have the time to fix this propper now.
if ( !isdefined( viewTarget ) || level.script == "hunted" )
return;
self setLookAtEnt( viewTarget );
self.set_lookat_point = true;
}
getspawner_byid( id )
{
return level.vehicle_spawners[ id ];
}
vehicle_getspawner()
{
assert( isdefined( self.spawner_id ) );
return getspawner_byid( self.spawner_id );
}
isDestructible()
{
return isdefined( self.destructible_type );
}
// SCRIPTER_MOD: JesseS (5/12/200) - readded script_vehicleattackgroup scripting
// This allows vehicles to attack other vehicles automatically. Just set the script_vehicleattackgroup of the attacker
// to the script_vehiclespawngroup you want to attack.
attackgroup_think()
{
self endon ("death");
self endon ("switch group");
self endon ("killed all targets");
if (isdefined (self.script_vehicleattackgroupwait))
{
wait (self.script_vehicleattackgroupwait);
}
for(;;)
{
//get all the vehicles
group = getentarray("script_vehicle", "classname");
// get our target array
valid_targets = [];
for (i = 0; i< group.size; i++)
{
// dpg: not all script vehicles have a script_vehiclespawngroup
if( !IsDefined( group[i].script_vehiclespawngroup ) )
{
continue;
}
// get all vehicles with the same spawngroup as the agroup we want to attack
if (group[i].script_vehiclespawngroup == self.script_vehicleattackgroup)
{
// Make sure we only attack different teams vehicles
if (group[i].script_team != self.script_team)
{
valid_targets = array_add(valid_targets, group[i]);
}
}
}
// Try again every .5 seconds if there are no valid targets.
if (valid_targets.size == 0)
{
wait (0.5);
continue;
}
// Main loop which makes the vehicle fire on the nearest group it's been set to attack
for (;;)
{
current_target = undefined;
if (valid_targets.size != 0)
{
current_target = self get_nearest_target(valid_targets);
}
else
{
// We killed them all, end this thread
self notify ("killed all targets");
}
// if it's a death vehicle, remove it from the list of valid targets
if (current_target.health <= 0)
{
valid_targets = array_remove(valid_targets, current_target);
continue;
}
else
{
// set the target ent for the vehicle. offset a bit so it doesnt shoot into the ground
self setturrettargetent( current_target, (0,0,50) );
// DPG 10/22/07 - in case we want specific wait values on the vehicle
if( IsDefined( self.fire_delay_min ) && IsDefined( self.fire_delay_max ) )
{
// in case max is less than min
if( self.fire_delay_max < self.fire_delay_min )
{
self.fire_delay_max = self.fire_delay_min;
}
wait ( randomintrange(self.fire_delay_min, self.fire_delay_max) );
}
else{
wait (randomintrange(4, 6));
}
self fireweapon();
}
}
}
// if this is called again, just add new group to queue
// queue should be sorted by distance
// delete from queue once vehicle is dead, check to see if vehicles health > 0
}
// Gets the nearest ent to self
get_nearest_target(valid_targets)
{
nearest_dist = 999999;
nearest = undefined;
for (i = 0; i < valid_targets.size; i++)
{
if( !isDefined( valid_targets[i] ) )
{
continue;
}
current_dist = distance( self.origin, valid_targets[i].origin );
if (current_dist < nearest_dist)
{
nearest_dist = current_dist;
nearest = valid_targets[i];
}
}
return nearest;
}
//---------------//
// Debug section //
//---------------//
/#
debug_vehicle()
{
self endon( "death" );
if( GetDvar( "debug_vehicle_health" ) == "" )
{
SetDvar( "debug_vehicle_health", "0" );
}
while( 1 )
{
if( GetDvarInt( "debug_vehicle_health" ) > 0 )
{
print3d( self.origin, "Health: " + self.health, ( 1, 1, 1 ), 1, 3 );
}
wait( 0.05 );
}
}
#/
generate_colmaps_vehicles()
{
/#
if ( !isdefined( level.vehicleInitThread ) || ! level.vehicleInitThread.size )
return;
// detect missing script_vehicles
script_vehicles = getentarray( "script_vehicle", "classname" );
hascol = [];
for ( i = 0 ; i < script_vehicles.size ; i++ )
{
if ( !isdefined( hascol[ script_vehicles[ i ].vehicletype ] ) )
hascol[ script_vehicles[ i ].vehicletype ] = [];
hascol[ script_vehicles[ i ].vehicletype ][ script_vehicles[ i ].model ] = true;
}
dump = false;
keys1 = getarraykeys( level.vehicleInitThread );
needscol = [];
for ( i = 0 ; i < keys1.size ; i++ )
{
keys2 = getarraykeys( level.vehicleInitThread [ keys1[ i ] ] );
for ( j = 0 ; j < keys2.size ; j++ )
{
if ( !isdefined( hascol[ keys1[ i ] ] )
|| !isdefined( hascol[ keys1[ i ] ][ keys2[ j ] ] ) )
{
dump = true;
needscol[ keys1[ i ] ][ keys2[ j ] ] = true;
}
}
if ( dump )
break;
}
// finished detecting
if ( dump )
dump_vehicles();
#/
// delete the models that go in the prefab
array_levelthread( getentarray( "colmap_vehicle", "targetname" ), maps\_utility::deleteEnt );
}
dump_vehicles()
{
/#
level.script = tolower( getdvar( "mapname" ) );
fileprint_map_start( level.script + "_vehicledump" );
stackupinc = 64;// probably unnessesary but may help make it unique looking in the editor and easily identified as the cache pile
keys1 = getarraykeys( level.vehicleInitThread );
for ( i = 0 ; i < keys1.size ; i++ )
{
keys2 = getarraykeys( level.vehicleInitThread [ keys1[ i ] ] );
for ( j = 0 ; j < keys2.size ; j++ )
{
origin = fileprint_radiant_vec( ( 0, 0, ( stackupinc * i ) ) );// convert these vectors to mapfile keypair format
fileprint_map_entity_start();
fileprint_map_keypairprint( "classname", "script_vehicle" );
fileprint_map_keypairprint( "model", keys2[ j ] );
fileprint_map_keypairprint( "origin", origin );
fileprint_map_keypairprint( "vehicletype", keys1[ i ] );
fileprint_map_keypairprint( "targetname", "colmap_vehicle" );// these get deleted by _vehicle
fileprint_map_entity_end();
}
}
fileprint_end();
println( " " );
println( " -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- - " );
println( " " );
println( "add this prefab to your map, it contains the vehicles needed to make the system happy" );
fabname = "xenon_export\\" + level.script + "_vehicledump.map";
println( "prefab is : " + fabname );
println( " " );
println( "after you've done this you'll need to compile" );
println( "I appologize for this very roundabout workaround to some stuff. " );
println( "It's on my next - game list to be fixed - nate" );
println( "the workaround is for getting colmaps on script_vehicles that exists as script_models in the map source and are purely spawned by scripts" );
println( " " );
println( " -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- - " );
println( " " );
// assertmsg( "Colmap .map file generated follow instructions in console.log" );
#/
}
get_deletegroups( script_vehiclegroupdelete )
{
deletegroups = [];
vehicles = getentarray( "script_vehicle", "classname" );
for ( i = 0 ; i < vehicles.size ; i++ )
{
if ( !isdefined( vehicles[ i ].script_vehiclegroupdelete )
|| vehicles[ i ].script_vehiclegroupdelete != script_vehiclegroupdelete
)
continue;
deletegroups[ deletegroups.size ] = vehicles[ i ];
}
return deletegroups;
}
damage_hints()
{
level.armorDamageHints = false;
self.displayingDamageHints = false;
self thread damage_hints_cleanup();
while ( isdefined( self ) )
{
self waittill( "damage", amount, attacker, direction_vec, point, type );
if ( !IsPlayer( attacker ) )
continue;
switch( tolower( type ) )
{
case "mod_grenade":
case "mod_grenade_splash":
case "mod_pistol_bullet":
case "mod_rifle_bullet":
case "bullet":
if ( !level.armorDamageHints )
{
level.armorDamageHints = true;
self.displayingDamageHints = true;
display_hint( "armor_damage" );
wait( 8 );
level.armorDamageHints = false;
self.displayingDamageHints = false;
break;
}
}
}
}
damage_hints_cleanup()
{
self waittill( "death" );
if ( self.displayingDamageHints )
level.armorDamageHints = false;
}
copy_destructable_attachments( modeldummy )
{
// does all attachments
attachedModelCount = self getattachsize();
attachedModels = [];
for ( i = 0 ; i < attachedModelCount ; i++ )
attachedModels[ i ] = tolower( self getAttachModelName( i ) );
for ( i = 0 ; i < attachedModels.size ; i++ )
modeldummy attach( attachedModels[ i ], tolower( self getattachtagname( i ) ) );
// original model still has it's own attachments because that's too much script to rewrite
// to move them completely to the dummy, I just want to translate the effects of an
// explosion to the dummy so that's all I'm doing. So don't expect a dummy destructable
// vehicle to react to damage and all of that jazz it's the original model that will do this
}
get_dummy()
{
if ( self.modeldummyon )
eModel = self.modeldummy;
else
eModel = self;
return eModel;
}
apply_truckjunk( eVehicle, truckjunk )
{
if ( !isdefined( self.spawner_id ) )
return;
if ( !isdefined( level.vehicle_truckjunk[ self.spawner_id ] ) )
return;
truckjunk = level.vehicle_truckjunk[ self.spawner_id ];
self.truckjunk = [];
for ( i = 0 ; i < truckjunk.size ; i++ )
{
model = spawn( "script_model", self.origin );
model setmodel( truckjunk[ i ].model );
model linkto( self, "tag_body", truckjunk[ i ].origin, truckjunk[ i ].angles );
self.truckjunk[ i ] = model;
}
}
truckjunk()
{
assert( isdefined( self.target ) );
linked_vehicle = getent( self.target, "targetname" );
assert( isdefined( linked_vehicle ) );
spawner_id = vehicle_spawnidgenerate( linked_vehicle.origin );
target = getent( self.target, "targetname" );
ghettotag = ghetto_tag_create( target );
if ( isdefined( self.script_noteworthy ) )
ghettotag.script_noteworthy = self.script_noteworthy;
if ( !isdefined( level.vehicle_truckjunk[ spawner_id ] ) )
level.vehicle_truckjunk[ spawner_id ] = [];
if ( isdefined( self.script_startingposition ) )
ghettotag.script_startingposition = self.script_startingposition;
level.vehicle_truckjunk[ spawner_id ][ level.vehicle_truckjunk[ spawner_id ].size ] = ghettotag;
self delete();
}
ghetto_tag_create( target )
{
struct = spawnstruct();
struct.origin = self.origin - target gettagorigin( "tag_body" );
struct.angles = self.angles - target gettagangles( "tag_body" );
struct.model = self.model;
if ( isdefined( struct.targetname ) )
level.struct_class_names[ "targetname" ][ struct.targetname ] = undefined;// done with this forever. don't stick around
if ( isdefined( struct.target ) )
level.struct_class_names[ "target" ][ struct.target ] = undefined;// done with this forever. don't stick around
return struct;
}
/*
=============
///ScriptDocBegin
"Name: vehicle_dump()"
"Summary: exports current vehicle positions to a map. Usefull for scripting complex interactions with vehicles"
"Module: Vehicle"
"CallOn: level"
"Example: vehicle_dump();"
"SPMP: singleplayer"
///ScriptDocEnd
=============
*/
vehicle_dump()
{
/#
// starts a map with the necessary blank layer info and a blank worldspawn.
// Throught he magic of junction this file ends up in "map_source\xenon_export\jeepride_veh_ref.map"
// I keep the directory structure somewhat flat because permissions in winnt is painful when dealing with the xenonremote share stuff.
// junction.bat keeps it in check
// this simple script exports all of the vehicles as script models that have a delete_on_load targetname
predumpvehicles = getentarray( "script_vehicle", "classname" );
vehicles = [];
// dumping can jump a frame in which the information could be altered, this stores the necessary info real quick
for ( i = 0 ; i < predumpvehicles.size ; i++ )
{
struct = spawnstruct();
struct.classname = predumpvehicles[ i ].classname;
struct.origin = predumpvehicles[ i ].origin;
struct.angles = predumpvehicles[ i ].angles;
struct.spawner_id = predumpvehicles[ i ].spawner_id;
struct.speedbeforepause = predumpvehicles[ i ] getspeedmph();
struct.script_vehiclespawngroup = predumpvehicles[ i ].script_vehiclespawngroup;
struct.script_vehiclestartmove = predumpvehicles[ i ].script_vehiclestartmove;
struct.model = predumpvehicles[ i ].model;
struct.angles = predumpvehicles[ i ].angles;
if ( isdefined( level.playersride ) && predumpvehicles[ i ] == level.playersride )
struct.playersride = true;
vehicles[ i ] = struct;
}
fileprint_map_start( level.script + "_veh_ref" );
for ( i = 0;i < vehicles.size;i++ )
{
origin = fileprint_radiant_vec( vehicles[ i ].origin );// convert these vectors to mapfile keypair format
angles = fileprint_radiant_vec( vehicles[ i ].angles );
fileprint_map_entity_start();
fileprint_map_keypairprint( "classname", "script_struct" );
fileprint_map_keypairprint( "spawnflags", "4" );
fileprint_map_keypairprint( "model", vehicles[ i ].model );
fileprint_map_keypairprint( "origin", origin );
fileprint_map_keypairprint( "angles", angles );
if ( isdefined( vehicles[ i ].playersride ) )
fileprint_map_keypairprint( "target", "delete_on_load" );// _load deletes these.
else
{
fileprint_map_keypairprint( "target", "structtarg" + i );// _load deletes these.
fileprint_map_keypairprint( "targetname", "delete_on_load" );// _load deletes these.
}
if ( isdefined( vehicles[ i ].speedbeforepause ) )
fileprint_map_keypairprint( "current_speed", vehicles[ i ].speedbeforepause );
if ( isdefined( vehicles[ i ].script_vehiclespawngroup ) )
fileprint_map_keypairprint( "script_vehiclespawngroup", vehicles[ i ].script_vehiclespawngroup );
if ( isdefined( vehicles[ i ].script_vehiclestartmove ) )
fileprint_map_keypairprint( "script_vehiclestartmove", vehicles[ i ].script_vehiclestartmove );
fileprint_map_entity_end();
// struct shows where the spawner is
if (
!isdefined( vehicles[ i ].spawner_id )
|| !isdefined( level.vehicle_spawners )
|| !isdefined( level.vehicle_spawners[ vehicles[ i ].spawner_id ] )
)
continue;
fileprint_map_entity_start();
fileprint_map_keypairprint( "classname", "script_struct" );
fileprint_map_keypairprint( "origin", fileprint_radiant_vec( level.vehicle_spawners[ vehicles[ i ].spawner_id ].origin ) );
fileprint_map_keypairprint( "_color", "0.300000 0.300000 0.300000" );
fileprint_map_keypairprint( "angles", angles );
fileprint_map_keypairprint( "model", vehicles[ i ].model );
fileprint_map_keypairprint( "targetname", "structtarg" + i );
if ( isdefined( vehicles[ i ].speedbeforepause ) )
fileprint_map_keypairprint( "current_speed", vehicles[ i ].speedbeforepause );
if ( isdefined( vehicles[ i ].script_vehiclespawngroup ) )
fileprint_map_keypairprint( "script_vehiclespawngroup", vehicles[ i ].script_vehiclespawngroup );
if ( isdefined( vehicles[ i ].script_vehiclestartmove ) )
fileprint_map_keypairprint( "script_vehiclestartmove", vehicles[ i ].script_vehiclestartmove );
fileprint_map_entity_end();
}
fileprint_end();
// iprintlnbold( "dumped vehicles" ); <- localize or die
#/
}
dump_handle()
{
/#
button1 = "r";
button2 = "CTRL";
wait_for_first_player();
while ( 1 )
{
while ( !twobuttonspressed( button1, button2 ) )
wait .05;
vehicle_dump();
while ( twobuttonspressed( button1, button2 ) )
wait .05;
}
#/
}
twobuttonspressed( button1, button2 )
{
/#
players = get_players();
if(!isdefined(players[0]))
return false;
return players[0] buttonpressed( button1 ) && players[0] buttonpressed( button2 );
#/
}
vehicle_load_ai( ai )
{
maps\_vehicle_aianim::load_ai( ai );
}
volume_up( timer )
{
self notify( "new_volume_command" );
self endon( "new_volume_command" );
assertex( isdefined( timer ), "No timer defined! ");
self endon( "death" );
timer = timer * 20;
for ( i = 0; i <= timer; i++ )
{
self setenginevolume( i / timer );
wait( 0.05 );
}
}
volume_down( timer )
{
self notify( "new_volume_command" );
self endon( "new_volume_command" );
assertex( isdefined( timer ), "No timer defined! ");
self endon( "death" );
timer = timer * 20;
for ( i = 0; i <= timer; i++ )
{
self setenginevolume( ( timer - i ) / timer );
wait( 0.05 );
}
}
lerp_enginesound( time, base_vol, dest_vol )
{
self endon( "death" );
assert( isdefined( base_vol ) );
assert( isdefined( dest_vol ) );
if ( time == 0 )
{
self SetEngineVolume( dest_vol );
return;
}
incs = int( time / .05 );
inc_vol = ( dest_vol - base_vol ) / incs ;
current_vol = base_vol;
for ( i = 0; i < incs; i++ )
{
current_vol += inc_vol;
self SetEngineVolume( current_vol );
wait .05;
}
}
kill_badplace( type )
{
if ( !isdefined( level.vehicle_death_badplace[ type ] ) )
return;
struct = level.vehicle_death_badplace[ type ];
if ( isdefined( struct.delay ) )
wait struct.delay;
if(!isdefined(self))
return;
BadPlace_Cylinder( "vehicle_kill_badplace", struct.duration, self.origin, struct.radius, struct.height, struct.team1, struct.team2 );
}
build_death_badplace( delay, duration, height, radius, team1, team2 )
{
if ( !isdefined( level.vehicle_death_badplace ) )
level.vehicle_death_badplace = [];
struct = spawnstruct();
struct.delay = delay;
struct.duration = duration;
struct.height = height;
struct.radius = radius;
struct.team1 = team1;
struct.team2 = team2;
level.vehicle_death_badplace[ level.vttype ] = struct;
}
build_death_jolt( delay )
{
if ( !isdefined( level.vehicle_death_jolt ) )
level.vehicle_death_jolt = [];
struct = spawnstruct();
struct.delay = delay;
level.vehicle_death_jolt[ level.vttype ] = struct;
}
kill_jolt( type )
{
if ( isdefined( level.vehicle_death_jolt[ type ] ) )
{
self.dontfreeme = true;
wait level.vehicle_death_jolt[ type ].delay;// this is all that exists currently, not to elaborate untill needed.
}
if( !isdefined( self ) )
return;
self joltbody( ( self.origin + ( 23, 33, 64 ) ), 3 );
wait 2;
if( !isdefined( self ) )
return;
self.dontfreeme = undefined;
}
build_tankmantle()
{
}