1
0
Fork 0

Air-to-air refueling enhancements

- Support selectable tankers
- allow configuration of refueling radius, speed
- add optional reporting of contact
This commit is contained in:
Stuart Buchanan 2013-02-24 22:34:11 +00:00
parent e9b433c184
commit 0fc2bf56fb
4 changed files with 346 additions and 67 deletions

View file

@ -19,6 +19,7 @@ var ai_enabled = nil;
var engines = nil; var engines = nil;
var tanks = []; var tanks = [];
var refuelingN = nil; var refuelingN = nil;
var contactN = nil;
var aimodelsN = nil; var aimodelsN = nil;
var types = {}; var types = {};
@ -47,7 +48,18 @@ var update_loop = func {
} }
var refueling = serviceable and size(tankers) > 0; var refueling = serviceable and size(tankers) > 0;
refuelingN.setBoolValue(refueling);
if (refuelingN.getNode("report-contact", 1).getValue()) {
if (refueling and !contactN.getValue()) {
setprop("/sim/messages/copilot", "Engage");
}
if (!refueling and contactN.getValue()) {
setprop("/sim/messages/copilot", "Disengage");
}
}
contactN.setBoolValue(refueling);
if (fuel_freeze) if (fuel_freeze)
return settimer(update_loop, UPDATE_PERIOD); return settimer(update_loop, UPDATE_PERIOD);
@ -65,8 +77,12 @@ var update_loop = func {
# calculate fuel received # calculate fuel received
if (refueling) { if (refueling) {
# assume max flow rate is 6000 lbs/min (for KC135) # Flow rate is the minimum of the tanker maxium rate
var received = 100 * UPDATE_PERIOD; # and the aircraft maximum rate. Both are expressed
# in lbs/min
var fuel_rate = math.min(tankers[0].getNode("refuel/max-fuel-transfer-lbs-min", 1).getValue() or 6000,
refuelingN.getNode("max-fuel-transfer-lbs-min", 1).getValue());
var received = UPDATE_PERIOD * fuel_rate / 60;
consumed -= received; consumed -= received;
} }
@ -169,7 +185,8 @@ setlistener("/sim/signals/fdm-initialized", func {
if (contains(globals, "fuel") and typeof(fuel) == "hash") if (contains(globals, "fuel") and typeof(fuel) == "hash")
fuel.loop = func nil; # kill $FG_ROOT/Nasal/fuel.nas' loop fuel.loop = func nil; # kill $FG_ROOT/Nasal/fuel.nas' loop
refuelingN = props.globals.initNode("/systems/refuel/contact", 0, "BOOL"); contactN = props.globals.initNode("/systems/refuel/contact", 0, "BOOL");
refuelingN = props.globals.getNode("/systems/refuel", 1);
aimodelsN = props.globals.getNode("ai/models", 1); aimodelsN = props.globals.getNode("ai/models", 1);
engines = props.globals.getNode("engines", 1).getChildren("engine"); engines = props.globals.getNode("engines", 1).getChildren("engine");

View file

@ -6,11 +6,6 @@ if (globals["tanker"] != nil) {
} }
#-------------------------------------------------------------------------------------------------- #--------------------------------------------------------------------------------------------------
var boom_tanker = "Models/Geometry/KC135/KC135.xml";
var probe_tanker = "Models/Geometry/KA6-D/KA6-D.xml";
var oclock = func(bearing) int(0.5 + geo.normdeg(bearing) / 30) or 12; var oclock = func(bearing) int(0.5 + geo.normdeg(bearing) / 30) or 12;
@ -75,7 +70,7 @@ var identity = {
var Tanker = { var Tanker = {
new: func(aiid, callsign, tacan, type, kias, heading, coord) { new: func(aiid, callsign, tacan, type, model, kias, maxfuel, pattern, heading, coord) {
var m = { parents: [Tanker] }; var m = { parents: [Tanker] };
m.callsign = callsign; m.callsign = callsign;
m.tacan = tacan; m.tacan = tacan;
@ -83,7 +78,7 @@ var Tanker = {
m.heading = m.course = m.track_course = heading; m.heading = m.course = m.track_course = heading;
m.out_of_range_time = 0; m.out_of_range_time = 0;
m.interval = 10; m.interval = 10;
m.length = (getprop("tanker/pattern-length-nm") or 50) * NM2M; m.length = pattern;
m.roll = 0; m.roll = 0;
m.coord = geo.Coord.new(coord); m.coord = geo.Coord.new(coord);
m.anchor = geo.Coord.new(coord).apply_course_distance(m.track_course, m.length); # ARCP m.anchor = geo.Coord.new(coord).apply_course_distance(m.track_course, m.length); # ARCP
@ -111,6 +106,7 @@ var Tanker = {
m.ai.getNode("valid", 1).setBoolValue(1); m.ai.getNode("valid", 1).setBoolValue(1);
m.ai.getNode("navaids/tacan/channel-ID", 1).setValue(m.tacan); m.ai.getNode("navaids/tacan/channel-ID", 1).setValue(m.tacan);
m.ai.getNode("refuel/type", 1).setValue(type); m.ai.getNode("refuel/type", 1).setValue(type);
m.ai.getNode("refuel/max-fuel-transfer-lbs-min", 1).setValue(maxfuel);
m.ai.getNode("refuel/contact", 1).setBoolValue(0); m.ai.getNode("refuel/contact", 1).setBoolValue(0);
m.ai.getNode("radar/in-range", 1).setBoolValue(1); m.ai.getNode("radar/in-range", 1).setBoolValue(1);
@ -130,7 +126,8 @@ var Tanker = {
m.vOffsetN = m.ai.getNode("radar/v-offset", 1); m.vOffsetN = m.ai.getNode("radar/v-offset", 1);
m.update(); m.update();
m.model.getNode("path", 1).setValue(type == "boom" ? boom_tanker : probe_tanker);
m.model.getNode("path", 1).setValue(model);
m.model.getNode("latitude-deg-prop", 1).setValue(m.latN.getPath()); m.model.getNode("latitude-deg-prop", 1).setValue(m.latN.getPath());
m.model.getNode("longitude-deg-prop", 1).setValue(m.lonN.getPath()); m.model.getNode("longitude-deg-prop", 1).setValue(m.lonN.getPath());
m.model.getNode("elevation-ft-prop", 1).setValue(m.altN.getPath()); m.model.getNode("elevation-ft-prop", 1).setValue(m.altN.getPath());
@ -206,6 +203,7 @@ var Tanker = {
var dalt = alt - me.ac.alt(); var dalt = alt - me.ac.alt();
var ac_hdg = getprop("/orientation/heading-deg"); var ac_hdg = getprop("/orientation/heading-deg");
var ac_pitch = getprop("/orientation/pitch-deg"); var ac_pitch = getprop("/orientation/pitch-deg");
var ac_contact_dist = getprop("/systems/refuel/contact-radius-m");
var elev = math.atan2(dalt, me.distance) * R2D; var elev = math.atan2(dalt, me.distance) * R2D;
me.latN.setDoubleValue(me.coord.lat()); me.latN.setDoubleValue(me.coord.lat());
@ -219,11 +217,13 @@ var Tanker = {
me.rangeN.setDoubleValue(me.distance * M2NM); me.rangeN.setDoubleValue(me.distance * M2NM);
me.brgN.setDoubleValue(me.bearing); me.brgN.setDoubleValue(me.bearing);
me.elevN.setDoubleValue(elev); me.elevN.setDoubleValue(elev);
me.contactN.setBoolValue(me.distance < 76 and dalt > 0 # 250 ft
and abs(view.normdeg(me.bearing - ac_hdg)) < 20);
me.hOffsetN.setDoubleValue(me.bearing - ac_hdg); me.contactN.setBoolValue(me.distance < ac_contact_dist and
me.vOffsetN.setDoubleValue(elev - ac_pitch); dalt > 0 and
abs(view.normdeg(me.bearing - ac_hdg)) < 20);
me.hOffsetN.setDoubleValue(me.bearing - ac_hdg);
me.vOffsetN.setDoubleValue(elev - ac_pitch);
var droll = me.roll_target - me.roll; var droll = me.roll_target - me.roll;
if (droll > 0) { if (droll > 0) {
@ -268,47 +268,80 @@ var Tanker = {
active: {}, active: {},
}; };
# Factory methods
# Create a tanker based on a given /sim/ai/tankers/tanker property node
var create_tanker = func(tanker_node, course) {
var (aiid, callsign, tacanid) =_= identity.get();
var model = tanker_node.getNode("model", 1).getValue();
var type = tanker_node.getNode("type", 1).getValue();
var spd = tanker_node.getNode("speed-kts", 1).getValue() or 250;
var pattern = (tanker_node.getNode("pattern-length-nm", 1).getValue() or 50) * NM2M;
var maxfuel = tanker_node.getNode("max-fuel-transfer-lbs-min", 1).getValue() or 6000;
var request = func { var alt = int(10 + rand() * 15) * 1000; # FL100--FL250
alt = skip_cloud_layer(alt * FT2M);
var dist = 6000 + rand() * 4000;
var coord = geo.aircraft_position().apply_course_distance(course, dist).set_alt(alt);
Tanker.new(aiid, callsign, tacanid, type, model, spd, maxfuel, pattern, course, coord);
}
# Request a new tanker
var request_new = func(tanker_node=nil) {
var tanker = values(Tanker.active);
if (size(tanker)) tanker[0].del();
request(tanker_node);
}
var request = func(tanker_node=nil) {
var tanker = values(Tanker.active); var tanker = values(Tanker.active);
if (size(tanker)) if (size(tanker))
return tanker[0].identify(); return tanker[0].identify();
var type = props.globals.getNode("systems/refuel", 1).getChildren("type"); if (tanker_node == nil) {
if (!size(type)) var type = props.globals.getNode("systems/refuel", 1).getChildren("type");
return; if (!size(type))
type = type[rand() * size(type)].getValue(); return;
type = type[rand() * size(type)].getValue();
var tankers = props.globals.getNode("/sim/ai/tankers", 1).getChildren("tanker");
foreach (var tanker; tankers) {
if (tanker.getNode("type", 1).getValue() == type) {
tanker_node = tanker;
break;
}
}
}
var (aiid, callsign, tacanid) =_= identity.get();
var hdg = getprop("orientation/heading-deg"); var hdg = getprop("orientation/heading-deg");
var course = hdg + (rand() - 0.5) * 60; var course = hdg + (rand() - 0.5) * 60;
var dist = 6000 + rand() * 4000;
var alt = int(10 + rand() * 15) * 1000; # FL100--FL250 create_tanker(tanker_node, course);
alt = skip_cloud_layer(alt * FT2M);
var coord = geo.aircraft_position().apply_course_distance(course, dist).set_alt(alt);
Tanker.new(aiid, callsign, tacanid, type, 250, hdg, coord);
} }
var request_random = func(tanker_node=nil) {
var tanker = values(Tanker.active);
if (size(tanker))
return tanker[0].identify();
var request_random = func { if (tanker_node == nil) {
var tanker = values(Tanker.active); var type = props.globals.getNode("systems/refuel", 1).getChildren("type");
if (size(tanker)) if (!size(type))
return tanker[0].identify(); return;
type = type[rand() * size(type)].getValue();
var type = props.globals.getNode("systems/refuel", 1).getChildren("type"); var tankers = props.globals.getNode("/sim/ai/tankers", 1).getChildren("tanker");
if (!size(type)) foreach (var tanker; tankers) {
return; if (tanker.getNode("type", 1).getValue() == type) {
type = type[rand() * size(type)].getValue(); tanker_node = tanker;
break;
}
}
}
var (aiid, callsign, tacanid) =_= identity.get(); var course = rand() * 360;
var hdg = rand() * 360; create_tanker(tanker_node, course);
var course = rand() * 360;
var dist = 6000 + rand() * 4000;
var alt = int(10 + rand() * 15) * 1000; # FL100--FL250
alt = skip_cloud_layer(alt * FT2M);
var coord = geo.aircraft_position().apply_course_distance(course, dist).set_alt(alt);
Tanker.new(aiid, callsign, tacanid, type, 250, hdg, coord);
} }

View file

@ -5,12 +5,63 @@
<x>-10</x> <x>-10</x>
<layout>vbox</layout> <layout>vbox</layout>
<nasal>
<open>
var dlgRoot = cmdarg();
var tankers = props.globals.getNode("/sim/ai/tankers/", 1).getChildren("tanker");
var types = props.globals.getNode("/systems/refuel/", 1).getChildren("type");
var tanker_node = props.globals.getNode("/sim/gui/dialogs/tanker/tanker", 1);
# Force default speed of 250kts
setprop("/sim/gui/dialogs/tanker/tanker/speed-kts", 250.0);
if (size(types) == 0) {
# This really shouldn't happen, as Nasal/tanker.nas disables this menu item
# if no refueling type is available.
gui.popupTip("Air to air refueling unavailable in this aircraft", 5);
fgcommand("dialog-close", props.Node.new({ "dialog-name" : "tanker"}));
}
if (size(tankers) > 0) {
var combo = gui.findElementByName(dlgRoot, "tanker-combo");
var idx = 0;
foreach (var t; tankers) {
foreach(var type; types) {
if (type.getValue() == t.getNode("type", 1).getValue()) {
combo.getChild("value", idx, 1).setValue(t.getNode("name", 1).getValue());
idx += 1;
}
}
}
}
var select_tanker = func() {
var name = getprop("/sim/gui/dialogs/tanker/selected-tanker");
foreach (var t; tankers) {
if (name == t.getNode("name", 1).getValue()) {
props.copy(t, tanker_node);
}
}
}
var generate_tanker = func() {
if (tanker_node.getNode("name", 1).getValue()) {
tanker.request_new(tanker_node);
}
}
</open>
</nasal>
<group> <group>
<layout>hbox</layout> <layout>hbox</layout>
<empty><stretch>1</stretch></empty> <empty><stretch>1</stretch></empty>
<text> <text>
<label>Tanker</label> <label>Air-to-Air Refueling Tanker</label>
</text> </text>
<empty><stretch>1</stretch></empty> <empty><stretch>1</stretch></empty>
@ -30,21 +81,195 @@
<hrule/> <hrule/>
<button> <group>
<legend>Request</legend> <layout>table</layout>
<equal>true</equal>
<binding> <text>
<command>nasal</command> <row>0</row>
<script>tanker.request()</script> <col>0</col>
</binding> <halign>right</halign>
</button> <label>Tanker:</label>
</text>
<combo>
<name>tanker-combo</name>
<row>0</row>
<col>1</col>
<colspan>2</colspan>
<halign>left</halign>
<property>/sim/gui/dialogs/tanker/selected-tanker</property>
<editable>false</editable>
<pref-width>200</pref-width>
<halign>fill</halign>
<binding>
<command>dialog-apply</command>
<object-name>tanker-combo</object-name>
</binding>
<binding>
<command>nasal</command>
<script>select_tanker();</script>
</binding>
</combo>
<text>
<row>1</row>
<col>0</col>
<halign>right</halign>
<label>Type:</label>
</text>
<text>
<row>1</row>
<col>1</col>
<visible>
<equals>
<property>/sim/gui/dialogs/tanker/tanker/type</property>
<value>probe</value>
</equals>
</visible>
<colspan>3</colspan>
<halign>left</halign>
<label>Drogue and Probe</label>
</text>
<text>
<row>1</row>
<col>1</col>
<visible>
<equals>
<property>/sim/gui/dialogs/tanker/tanker/type</property>
<value>boom</value>
</equals>
</visible>
<colspan>3</colspan>
<halign>left</halign>
<label>Boom</label>
</text>
<text>
<row>2</row>
<col>0</col>
<halign>right</halign>
<label>Speed:</label>
</text>
<slider>
<name>tanker-speed</name>
<row>2</row>
<col>1</col>
<halign>fill</halign>
<min>100</min>
<max>350</max>
<live>true</live>
<property>/sim/gui/dialogs/tanker/tanker/speed-kts</property>
<binding>
<command>dialog-apply</command>
<object-name>tanker-speed</object-name>
</binding>
</slider>
<text>
<row>2</row>
<col>2</col>
<halign>left</halign>
<format>%2.0fkts</format>
<label>250</label>
<property>/sim/gui/dialogs/tanker/tanker/speed-kts</property>
<live>true</live>
</text>
<text>
<row>3</row>
<col>0</col>
<halign>right</halign>
<label>Contact radius:</label>
</text>
<slider>
<name>contact-radius</name>
<row>3</row>
<col>1</col>
<halign>fill</halign>
<min>1</min>
<max>100</max>
<property>/systems/refuel/contact-radius-m</property>
<binding>
<command>dialog-apply</command>
<object-name>contact-radius</object-name>
</binding>
</slider>
<text>
<row>3</row>
<col>2</col>
<halign>left</halign>
<format>%2.0fm</format>
<property>/systems/refuel/contact-radius-m</property>
<live>true</live>
</text>
<text>
<row>4</row>
<col>0</col>
<halign>right</halign>
<label>Report refueling:</label>
</text>
<checkbox>
<name>report-contact</name>
<row>4</row>
<col>1</col>
<halign>left</halign>
<property>/systems/refuel/report-contact</property>
<binding>
<command>dialog-apply</command>
<object-name>report-contact</object-name>
</binding>
</checkbox>
</group>
<hrule/>
<group>
<layout>hbox</layout>
<default-padding>5</default-padding>
<empty><stretch>true</stretch></empty>
<button>
<legend>Request</legend>
<equal>true</equal>
<binding>
<command>nasal</command>
<script>generate_tanker();</script>
</binding>
</button>
<empty><stretch>true</stretch></empty>
<button>
<legend>Get Position</legend>
<equal>true</equal>
<binding>
<command>nasal</command>
<script>tanker.report()</script>
</binding>
</button>
<empty><stretch>true</stretch></empty>
<button>
<legend>Close</legend>
<equal>true</equal>
<key>Esc</key>
<binding>
<command>dialog-close</command>
</binding>
</button>
<empty><stretch>true</stretch></empty>
</group>
<button>
<legend>Get Position</legend>
<equal>true</equal>
<binding>
<command>nasal</command>
<script>tanker.report()</script>
</binding>
</button>
</PropertyList> </PropertyList>

View file

@ -757,6 +757,7 @@ Started September 2000 by David Megginson, david@megginson.com
<scenarios-enabled type="bool" userarchive="y">true</scenarios-enabled> <scenarios-enabled type="bool" userarchive="y">true</scenarios-enabled>
<scenario>nimitz_demo</scenario> <scenario>nimitz_demo</scenario>
<groundnet-cache type="bool">true</groundnet-cache> <groundnet-cache type="bool">true</groundnet-cache>
<tankers include="AI/tankers.xml"/>
</ai> </ai>
<multiplay preserve="y"> <multiplay preserve="y">
@ -1202,6 +1203,9 @@ Started September 2000 by David Megginson, david@megginson.com
</vacuum> </vacuum>
<refuel> <refuel>
<serviceable type="bool" archive="y">true</serviceable> <serviceable type="bool" archive="y">true</serviceable>
<contact-radius-m type="double">76</contact-radius-m>
<report-contact type="bool">false</report-contact>
<max-fuel-transfer-lbs-min type="double">6000</max-fuel-transfer-lbs-min>
</refuel> </refuel>
</systems> </systems>