oprf_assets/Frigate/Nasal/fire-control.nas
2016-12-22 00:31:49 -08:00

209 lines
7.0 KiB
Plaintext

################### GLOBALS
var false = 0;
var true = 1;
var radar_update_time = 2;
var launch_update_time = 0.3;
var missile_delay_time = 0;
var ciws_delay_time = 0;
var AIR = 0;
var MARINE = 1;
var SURFACE = 2;
var ORDNANCE = 3;
var ACTIVE_MISSILE = 0;
var NUM_MISSILES = 7;
var ROUNDS = 30;
var RELOAD_TIME = 600;
################### MISSILE INFO
var missile_name = "KN-06";
var missile_brevity = "Grumble";
var missile_max_distance = 85; #max distance in nm
var missile_min_distance = 2; #minimum distance in nm
var lockon_time = 6; #time in seconds it takes to lock on to a target
################### IFF
var target = {
new: func (callsign) {
var m = { parents: [target] };
m.callsign = callsign;
m.fired = false;
return m;
},
};
var targets = {
"pinto": target.new("pinto"),
"Leto": target.new("Leto"),
"YV-187": target.new("YV-187"),
"swamp": target.new("swamp"),
"Raider1": target.new("Raider1"),
"AF-MA13": target.new("AF-MA13"),
"KOL24M": target.new("KOL24M"),
"SNOWY1": target.new("SNOWY1"),
};
################### MAIN LOOP
var scan = func() {
if ( getprop("/carrier/sunk") == 1 ) {
return;
}
#### ITERATE THROUGH MP LIST ####
var my_pos = geo.aircraft_position();
foreach(var mp; props.globals.getNode("/ai/models").getChildren("multiplayer")){
#### DO WE HAVE FIRING SOLUTION... ####
# is plane in range, do we still have missiles, and is a missile already inbound, and has it been 4 seconds since the last missile launch?
var trigger = fire_control(mp, my_pos);
#print("dist to target = " ~ dist_to_target);
#### ... FOR THE MISSILE ####
if ( trigger == true and targets[mp.getNode("callsign").getValue()].fired == false and ACTIVE_MISSILE <= NUM_MISSILES and ( systime() - missile_delay_time > 7 ) ) { #
#print("callsign " ~ cs ~ " found at " ~ dist_to_target);
missile_delay_time = systime();
targets[mp.getNode("callsign").getValue()].fired = true;
mp.getNode("unique",1).setValue(rand());
armament.contact = radar_logic.Contact.new(mp, AIR);
missile_launch(mp, systime());
#### ... FOR THE CIWS ####
} elsif ( trigger == 2 and ROUNDS > 0 and ( systime() - ciws_delay_time > 1.0 ) ) {
hit_msg = defeatSpamFilter("Gun Splash On : " ~ mp.getNode("callsign").getValue());
print("CIWS fired | rounds remaining: " ~ ROUNDS ~ " | hit on: " ~ mp.getNode("callsign").getValue());
ciws_delay_time = systime();
ROUNDS = ROUNDS - 1;
setprop("/sim/multiplay/chat",hit_msg);
}
}
settimer(scan,radar_update_time);
}
################## FIRE CONTROL
################## ALL AI LOGIC RELATED TO FIGURING OUT IF A TARGET SHOULD BE SHOT AT SHOULD GO HERE.
var fire_control = func(mp, my_pos) {
#gather some data about the target
var ufo_pos = geo.Coord.new().set_latlon(mp.getNode("position/latitude-deg").getValue(),mp.getNode("position/longitude-deg").getValue(),(mp.getNode("position/altitude-ft").getValue() * 0.3048));
var target_distance = my_pos.direct_distance_to(ufo_pos) * .000539957; #in nautical miles
var target_heading = mp.getNode("orientation/true-heading-deg").getValue();
var target_bearing = my_pos.course_to(ufo_pos);
var relative_bearing = math.abs(ufo_pos.course_to(my_pos) - target_heading);
var target_altitude = mp.getNode("position/altitude-ft").getValue();
var target_airspeed = mp.getNode("velocities/true-airspeed-kt").getValue();
# can the radar see it?
if ( mp.getNode("valid").getValue() == false ) { return false; }
if ( radar_logic.isNotBehindTerrain(mp) == false ) { return false; }
if ( target_airspeed < 40 ) { return false; }
if ( target_altitude - props.globals.getNode("/position/altitude-ft").getValue() < 150 ) { return false; }
if ( target_altitude > 70000 ) { return false; }
if ( target_distance > missile_max_distance ) { return false; }
# is this plane a friend or foe?
if ( targets[mp.getNode("callsign").getValue()] == nil ) { return false; }
#special CIWS handling
if ( target_distance < 0.5 ) { return 2; }
#should we shoot? using linear interpolation, with minimum probability of 0.01 to 0.10 and maximum probability of 1
var distance_probability = 1 + (0.01 - 1) * (( target_distance - missile_min_distance ) / ( missile_max_distance - missile_min_distance ));
var bearing_probability = 1 + (0.05 - 1) * (( relative_bearing ) / ( 180 ));
var altitude_probability = 1 + (0.05 - 1) * (( target_altitude - 150 ) / ( 70000 - 150 ));
var speed_probability = distance_probability / 2.5 + (1 - distance_probability / 2.5) * (( target_airspeed - 1000 ) / ( 40 - 1000 ));
var distance_weight = 0.4;
var bearing_weight = 0.3;
var altitude_weight = 0.2;
var speed_weight = 0.1;
var fire_probability = (distance_probability * distance_weight) + (bearing_probability * bearing_weight) + (altitude_probability * altitude_weight) + (speed_probability * speed_weight);
var the_dice = rand();
print("probability for " ~ mp.getNode("callsign").getValue() ~ ": "~fire_probability);
if ( fire_probability > 0.60 and fire_probability > the_dice ) {
return true;
} else {
return false;
}
}
################### MISSILE CONTROL
### missile reload
var reload = func() {
#figure out how many to add
for ( var i = 0; i <= NUM_MISSILES; i = i + 1 ) {
armament.AIM.new(i,missile_name,missile_brevity);
armament.AIM.active[i].status = 0;
armament.AIM.active[i].search();
}
ACTIVE_MISSILE = 0;
}
### missile launch
var missile_launch = func(mp, launchtime) {
if ( armament.AIM.active[ACTIVE_MISSILE].status == 1 and systime() - launchtime > lockon_time and radar_logic.isNotBehindTerrain(mp) == true ) {
armament.AIM.active[ACTIVE_MISSILE].release();
setprop("/sim/multiplay/chat", defeatSpamFilter("KN-06 fired at: " ~ mp.getNode("callsign").getValue()));
ACTIVE_MISSILE = ACTIVE_MISSILE + 1;
print("Fired KN-06 #" ~ ACTIVE_MISSILE ~ " at: " ~ mp.getNode("callsign").getValue());
return;
}
settimer( func { missile_launch(mp, launchtime); },launch_update_time);
}
### missile set launched to false for target
var incoming_listener = func {
var history = getprop("/sim/multiplay/chat-history");
var hist_vector = split("\n", history);
if (size(hist_vector) > 0) {
var last = hist_vector[size(hist_vector)-1];
var last_vector = split(":", last);
var author = last_vector[0];
var callsign = getprop("sim/multiplay/callsign");
if (size(last_vector) > 1 and author == callsign) {
var last1 = split(" ", last_vector[1]);
if(size(last1) > 2 and last1[size(last1)-1] == "exploded" ) {
#print("missile hitting someone");
if (size(last_vector) > 3) {
#print("that someone is me!");
var type = last1[1];
#callsign = target, type = missile
last_vector[3] = right(last_vector[3],size(last_vector[3]) - 1);
if ( targets[last_vector[3]] != nil and type == "KN-06" ) {
targets[last_vector[3]].fired = false;
}
}
}
}
}
}
################### MISC
var spams = 1;
var defeatSpamFilter = func (str) {
spams += 1;
if (spams == 15) {
spams = 1;
}
str = str~":";
for (var i = 1; i <= spams; i+=1) {
str = str~".";
}
return str;
}
reload();
scan();
setlistener("/sim/multiplay/chat-history", incoming_listener, 0, 0);