2006-03-22 20:25:58 +00:00
|
|
|
|
2006-03-16 16:24:52 +00:00
|
|
|
# maximum speed -----------------------------------------------------------------------------------
|
2006-03-15 14:27:47 +00:00
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
|
2006-03-15 00:29:53 +00:00
|
|
|
var maxspeed = props.globals.getNode("engines/engine/speed-max-mps");
|
|
|
|
var speed = [10, 20, 50, 100, 200, 500, 1000, 2000, 5000, 10000, 20000, 50000];
|
|
|
|
var current = 7;
|
|
|
|
|
|
|
|
|
|
|
|
controls.flapsDown = func(x) {
|
|
|
|
if (!x) {
|
|
|
|
return;
|
|
|
|
} elsif (x < 0 and current > 0) {
|
|
|
|
current -= 1;
|
|
|
|
} elsif (x > 0 and current < size(speed) - 1) {
|
|
|
|
current += 1;
|
|
|
|
}
|
|
|
|
var s = speed[current];
|
|
|
|
maxspeed.setDoubleValue(s);
|
2006-03-22 20:25:58 +00:00
|
|
|
gui.popupTip("Max. Speed " ~ s ~ " m/s");
|
2006-03-15 00:29:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2006-03-16 16:24:52 +00:00
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
|
|
|
|
# library stuff -----------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
var ERAD = 6378138.12; # Earth radius
|
|
|
|
var D2R = math.pi / 180;
|
|
|
|
var R2D = 180 / math.pi;
|
2006-03-16 16:24:52 +00:00
|
|
|
|
2006-03-15 00:29:53 +00:00
|
|
|
|
|
|
|
ft2m = func { arg[0] * 0.3048 }
|
|
|
|
m2ft = func { arg[0] / 0.3048 }
|
2006-03-22 20:25:58 +00:00
|
|
|
floor = func(v) { v < 0.0 ? -int(-v) - 1 : int(v) }
|
|
|
|
printf = func(_...) { print(call(sprintf, _)) }
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# convert [lon, lat] to [x, y, z]
|
|
|
|
#
|
|
|
|
lonlat2xyz = func(lonlat) {
|
|
|
|
var lonr = lonlat[0] * D2R;
|
|
|
|
var latr = lonlat[1] * D2R;
|
|
|
|
var cosphi = math.cos(latr);
|
|
|
|
var x = cosphi * math.cos(lonr);
|
|
|
|
var y = cosphi * math.sin(lonr);
|
|
|
|
var z = math.sin(latr);
|
|
|
|
return [x, y, z];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
# convert [x, y, z] to [lon, lat]
|
|
|
|
#
|
|
|
|
xyz2lonlat = func(xyz) {
|
|
|
|
var x = xyz[0];
|
|
|
|
var y = xyz[1];
|
|
|
|
var z = xyz[2];
|
|
|
|
var aux = x * x + y * y;
|
|
|
|
var lat = math.atan2(z, math.sqrt(aux)) * R2D;
|
|
|
|
var lon = math.atan2(y, x) * R2D;
|
|
|
|
return [lon, lat];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
# return squared distance between two [x, y, z]
|
|
|
|
#
|
|
|
|
coord_dist_sq = func(xyz0, xyz1) {
|
|
|
|
var x = xyz0[0] - xyz1[0];
|
|
|
|
var y = xyz0[1] - xyz1[1];
|
|
|
|
var z = xyz0[2] - xyz1[2];
|
|
|
|
return x * x + y * y + z * z;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
# sort vector of strings (bubblesort)
|
|
|
|
#
|
|
|
|
sort = func(list) {
|
|
|
|
var l = list;
|
|
|
|
while (1) {
|
|
|
|
var n = 0;
|
|
|
|
for (var i = 0; i < size(l) - 1; i += 1) {
|
|
|
|
if (cmp(l[i], l[i + 1]) > 0) {
|
|
|
|
var t = l[i + 1];
|
|
|
|
l[i + 1] = l[i];
|
|
|
|
l[i] = t;
|
|
|
|
n += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!n) {
|
|
|
|
return l;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-03-24 00:20:22 +00:00
|
|
|
# binary search of string in sorted vector; returns index or -1 if not found
|
|
|
|
#
|
|
|
|
search = func(list, which) {
|
|
|
|
var left = 0;
|
|
|
|
var right = size(list);
|
|
|
|
var middle = nil;
|
|
|
|
while (1) {
|
|
|
|
middle = int((left + right) / 2);
|
|
|
|
var c = cmp(list[middle], which);
|
|
|
|
if (!c) {
|
|
|
|
return middle;
|
|
|
|
} elsif (left == middle) {
|
|
|
|
return -1;
|
|
|
|
} elsif (c > 0) {
|
|
|
|
right = middle;
|
|
|
|
} elsif (c < 0) {
|
|
|
|
left = middle;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
# scan all objects in subdir of $FG_ROOT. (Prefer *.xml files to *.ac files.)
|
|
|
|
#
|
|
|
|
scan_models = func(base) {
|
|
|
|
var result = [];
|
|
|
|
var list = directory(getprop("/sim/fg-root") ~ "/" ~ base);
|
|
|
|
if (list == nil) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
var xml = {};
|
|
|
|
var ac = {};
|
|
|
|
foreach (var d; list) {
|
|
|
|
if (d[0] != `.` and d != "CVS") {
|
|
|
|
if (substr(d, size(d) - 4) == ".xml") {
|
|
|
|
xml[base ~ "/" ~ d] = 1;
|
|
|
|
} elsif (substr(d, size(d) - 3) == ".ac") {
|
|
|
|
ac[base ~ "/" ~ d] = 1;
|
|
|
|
} else {
|
|
|
|
foreach (var s; scan_models(base ~ "/" ~ d)) {
|
|
|
|
append(result, s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
foreach (var m; keys(xml)) {
|
|
|
|
append(result, m);
|
|
|
|
delete(ac, var x = substr(m, 0, size(m) - 3) ~ "ac");
|
|
|
|
}
|
|
|
|
foreach (var m; keys(ac)) {
|
|
|
|
append(result, m);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
2006-03-15 00:29:53 +00:00
|
|
|
|
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
# normalize degree to 0 <= angle < 360
|
|
|
|
#
|
2006-03-15 00:29:53 +00:00
|
|
|
normdeg = func(angle) {
|
|
|
|
while (angle < 0) {
|
|
|
|
angle += 360;
|
|
|
|
}
|
|
|
|
while (angle >= 360) {
|
|
|
|
angle -= 360;
|
|
|
|
}
|
|
|
|
angle;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-03-15 14:27:47 +00:00
|
|
|
bucket_span = func(lat) {
|
|
|
|
if (lat >= 89.0 ) {
|
|
|
|
360.0;
|
|
|
|
} elsif (lat >= 88.0 ) {
|
|
|
|
8.0;
|
|
|
|
} elsif (lat >= 86.0 ) {
|
|
|
|
4.0;
|
|
|
|
} elsif (lat >= 83.0 ) {
|
|
|
|
2.0;
|
|
|
|
} elsif (lat >= 76.0 ) {
|
|
|
|
1.0;
|
|
|
|
} elsif (lat >= 62.0 ) {
|
|
|
|
0.5;
|
|
|
|
} elsif (lat >= 22.0 ) {
|
|
|
|
0.25;
|
|
|
|
} elsif (lat >= -22.0 ) {
|
|
|
|
0.125;
|
|
|
|
} elsif (lat >= -62.0 ) {
|
|
|
|
0.25;
|
|
|
|
} elsif (lat >= -76.0 ) {
|
|
|
|
0.5;
|
|
|
|
} elsif (lat >= -83.0 ) {
|
|
|
|
1.0;
|
|
|
|
} elsif (lat >= -86.0 ) {
|
|
|
|
2.0;
|
|
|
|
} elsif (lat >= -88.0 ) {
|
|
|
|
4.0;
|
|
|
|
} elsif (lat >= -89.0 ) {
|
|
|
|
8.0;
|
|
|
|
} else {
|
|
|
|
360.0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
tile_index = func(lon, lat) {
|
2006-03-23 16:16:44 +00:00
|
|
|
var lon_floor = floor(lon);
|
|
|
|
var lat_floor = floor(lat);
|
|
|
|
var span = bucket_span(lat);
|
|
|
|
var x = 0;
|
2006-03-15 14:27:47 +00:00
|
|
|
|
|
|
|
if (span < 0.0000001) {
|
|
|
|
lon = 0;
|
|
|
|
} elsif (span <= 1.0) {
|
|
|
|
x = int((lon - lon_floor) / span);
|
|
|
|
} else {
|
|
|
|
if (lon >= 0) {
|
|
|
|
lon = int(int(lon / span) * span);
|
|
|
|
} else {
|
|
|
|
lon = int(int((lon + 1) / span) * span - span);
|
|
|
|
if (lon < -180) {
|
|
|
|
lon = -180;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-03-23 16:16:44 +00:00
|
|
|
var y = int((lat - lat_floor) * 8);
|
2006-03-15 14:27:47 +00:00
|
|
|
(lon_floor + 180) * 16384 + (lat_floor + 90) * 64 + y * 8 + x;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-03-23 16:16:44 +00:00
|
|
|
format = func(lon, lat) {
|
|
|
|
sprintf("%s%03d%s%02d", lon < 0 ? "w" : "e", abs(lon), lat < 0 ? "s" : "n", abs(lat));
|
2006-03-15 14:27:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-03-23 16:16:44 +00:00
|
|
|
tile_path = func(lon, lat) {
|
|
|
|
var p = format(floor(lon / 10.0) * 10, floor(lat / 10.0) * 10);
|
|
|
|
p ~= "/" ~ format(floor(lon), floor(lat));
|
|
|
|
p ~= "/" ~ tile_index(lon, lat) ~ ".stg";
|
|
|
|
}
|
2006-03-22 20:25:58 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2006-03-23 16:16:44 +00:00
|
|
|
# cursor ------------------------------------------------------------------------------------------
|
2006-03-22 20:25:58 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2006-03-15 14:27:47 +00:00
|
|
|
Value = {
|
|
|
|
new : func(baseN, name, init) {
|
2006-03-22 20:25:58 +00:00
|
|
|
var m = { parents: [Value] };
|
2006-03-15 14:27:47 +00:00
|
|
|
m.lastOffs = 0;
|
|
|
|
m.init = init;
|
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
# offset node; fed by the dialog slider
|
2006-03-16 16:24:52 +00:00
|
|
|
m.inOffsN = baseN.getNode("offsets/" ~ name, 1);
|
2006-03-15 14:27:47 +00:00
|
|
|
m.inOffsN.setValue(m.lastOffs);
|
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
# live number property
|
|
|
|
m.outN = baseN.getNode("adjust/" ~ name, 1);
|
|
|
|
m.outN.setDoubleValue(init);
|
2006-03-15 14:27:47 +00:00
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
m.listener = setlistener(m.inOffsN, func { m.update() });
|
2006-03-15 14:27:47 +00:00
|
|
|
return m;
|
|
|
|
},
|
|
|
|
del : func {
|
2006-03-23 16:16:44 +00:00
|
|
|
removelistener(me.listener);
|
2006-03-15 14:27:47 +00:00
|
|
|
},
|
|
|
|
reset : func {
|
|
|
|
me.center();
|
|
|
|
me.outN.setValue(me.init);
|
|
|
|
},
|
|
|
|
center : func {
|
|
|
|
me.update();
|
|
|
|
me.inOffsN.setValue(me.lastOffs = 0);
|
|
|
|
},
|
|
|
|
update : func {
|
|
|
|
var offs = me.inOffsN.getValue();
|
|
|
|
me.outN.setValue(me.outN.getValue() + offs - me.lastOffs);
|
|
|
|
me.lastOffs = offs;
|
|
|
|
},
|
|
|
|
set : func(v) {
|
|
|
|
me.center();
|
|
|
|
me.outN.setValue(v);
|
|
|
|
},
|
|
|
|
get : func {
|
|
|
|
me.outN.getValue();
|
|
|
|
},
|
|
|
|
add : func(v) {
|
|
|
|
me.center();
|
|
|
|
me.outN.setValue(me.outN.getValue() + v);
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
Adjust = {
|
|
|
|
new : func(prop) {
|
|
|
|
var m = { parents: [Adjust] };
|
|
|
|
m.node = props.globals.getNode(prop, 1);
|
|
|
|
m.val = {
|
|
|
|
lon: Value.new(m.node, "longitude-deg", 0),
|
|
|
|
lat: Value.new(m.node, "latitude-deg", 0),
|
|
|
|
elev: Value.new(m.node, "elevation-ft", -10000),
|
2006-03-15 14:27:47 +00:00
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
hdg: Value.new(m.node, "heading-deg", 0),
|
|
|
|
pitch: Value.new(m.node, "pitch-deg", 0),
|
|
|
|
roll: Value.new(m.node, "roll-deg", 0),
|
2006-03-15 14:27:47 +00:00
|
|
|
};
|
2006-03-24 15:33:06 +00:00
|
|
|
m.stk_hdgN = m.node.getNode("sticky-heading", 1);
|
|
|
|
m.stk_orientN = m.node.getNode("sticky-orientation", 1);
|
|
|
|
m.stk_hdgN.setBoolValue(0);
|
|
|
|
m.stk_orientN.setBoolValue(0);
|
2006-03-15 14:27:47 +00:00
|
|
|
return m;
|
|
|
|
},
|
2006-03-22 20:25:58 +00:00
|
|
|
del : func {
|
|
|
|
foreach (var v; keys(me.val)) {
|
|
|
|
me.val[v].del();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
offsetNode : func(which) {
|
|
|
|
me.val[which].inOffsN;
|
|
|
|
},
|
|
|
|
outNode : func(which) {
|
|
|
|
me.val[which].outN;
|
|
|
|
},
|
|
|
|
get : func(which) {
|
|
|
|
me.val[which].get();
|
|
|
|
},
|
|
|
|
set : func(which, value) {
|
|
|
|
me.val[which].set(value);
|
|
|
|
},
|
2006-03-24 15:33:06 +00:00
|
|
|
setall : func(lon, lat, elev, hdg = nil, pitch = nil, roll = nil) {
|
2006-03-22 20:25:58 +00:00
|
|
|
me.val["lon"].set(lon);
|
|
|
|
me.val["lat"].set(lat);
|
|
|
|
me.val["elev"].set(elev);
|
2006-03-24 15:33:06 +00:00
|
|
|
if (hdg != nil) {
|
|
|
|
me.val["hdg"].set(hdg);
|
|
|
|
} elsif (!me.stk_hdgN.getBoolValue()) {
|
|
|
|
me.val["hdg"].reset();
|
|
|
|
}
|
|
|
|
if (pitch != nil) {
|
|
|
|
me.val["pitch"].set(pitch);
|
|
|
|
} elsif (!me.stk_orientN.getBoolValue()) {
|
|
|
|
me.val["pitch"].reset();
|
|
|
|
}
|
|
|
|
if (roll != nil) {
|
|
|
|
me.val["roll"].set(roll);
|
|
|
|
} elsif (!me.stk_orientN.getBoolValue()) {
|
|
|
|
me.val["roll"].reset();
|
|
|
|
}
|
2006-03-22 20:25:58 +00:00
|
|
|
},
|
2006-03-15 14:27:47 +00:00
|
|
|
reset : func {
|
2006-03-22 20:25:58 +00:00
|
|
|
foreach (var v; keys(me.val)) {
|
|
|
|
me.val[v].reset();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
step : func(which, step) {
|
|
|
|
me.val[which].add(step);
|
|
|
|
},
|
|
|
|
upright : func {
|
|
|
|
me.val["pitch"].set(0);
|
|
|
|
me.val["roll"].set(0);
|
2006-03-15 14:27:47 +00:00
|
|
|
},
|
2006-03-24 15:33:06 +00:00
|
|
|
orient : func {
|
|
|
|
me.val["hdg"].set(0);
|
|
|
|
},
|
|
|
|
center_sliders : func {
|
2006-03-22 20:25:58 +00:00
|
|
|
foreach (var v; keys(me.val)) {
|
|
|
|
me.val[v].center();
|
|
|
|
}
|
2006-03-15 14:27:47 +00:00
|
|
|
},
|
2006-03-22 20:25:58 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
Model = {
|
|
|
|
# searches first free slot and sets path
|
|
|
|
new : func(path) {
|
|
|
|
var m = { parents: [Model] };
|
|
|
|
var models = props.globals.getNode("/models", 1);
|
|
|
|
|
|
|
|
for (var i = 0; 42; i += 1) {
|
|
|
|
if (models.getChild("model", i, 0) == nil) {
|
|
|
|
m.node = models.getChild("model", i, 1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
m.path = path;
|
|
|
|
m.node.getNode("path", 1).setValue(m.path);
|
|
|
|
return m;
|
2006-03-15 14:27:47 +00:00
|
|
|
},
|
2006-03-22 20:25:58 +00:00
|
|
|
# signal modelmgr.cxx to load model
|
|
|
|
load : func {
|
|
|
|
me.node.getNode("load", 1).setValue(1);
|
|
|
|
me.node.removeChildren("load");
|
|
|
|
},
|
|
|
|
add_derived_props : func(node) {
|
|
|
|
var path = node.getNode("path").getValue();
|
|
|
|
var lon = node.getNode("longitude-deg").getValue();
|
|
|
|
var lat = node.getNode("latitude-deg").getValue();
|
|
|
|
var elev = node.getNode("elevation-ft").getValue();
|
|
|
|
var hdg = node.getNode("heading-deg").getValue();
|
|
|
|
|
|
|
|
var type = "OBJECT_" ~ ((path != nil and split("/", path)[0] == "Models")
|
|
|
|
? "SHARED" : "STATIC");
|
|
|
|
var stg_hdg = normdeg(360 - hdg);
|
|
|
|
var stg_path = tile_path(lon, lat);
|
|
|
|
var abs_path = getprop("/sim/fg-root") ~ "/" ~ path;
|
|
|
|
var obj_line = sprintf("%s %s %.6f %.6f %.4f %.1f", type, path, lon, lat,
|
|
|
|
ft2m(elev), stg_hdg);
|
|
|
|
|
|
|
|
node.getNode("absolute-path", 1).setValue(abs_path);
|
|
|
|
node.getNode("stg-path", 1).setValue(stg_path);
|
|
|
|
node.getNode("stg-heading-deg", 1).setDoubleValue(stg_hdg);
|
|
|
|
node.getNode("object-line", 1).setValue(obj_line)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
Static = {
|
|
|
|
new : func(path, lon, lat, elev, hdg, pitch, roll) {
|
|
|
|
var m = Model.new(path);
|
|
|
|
m.parents = [Static, Model];
|
|
|
|
|
|
|
|
m.node.getNode("longitude-deg", 1).setDoubleValue(m.lon = lon);
|
|
|
|
m.node.getNode("latitude-deg", 1).setDoubleValue(m.lat = lat);
|
|
|
|
m.node.getNode("elevation-ft", 1).setDoubleValue(m.elev = elev);
|
|
|
|
m.node.getNode("heading-deg", 1).setDoubleValue(m.hdg = hdg);
|
|
|
|
m.node.getNode("pitch-deg", 1).setDoubleValue(m.pitch = pitch);
|
|
|
|
m.node.getNode("roll-deg", 1).setDoubleValue(m.roll = roll);
|
|
|
|
m.load();
|
|
|
|
return m;
|
|
|
|
},
|
|
|
|
del : func {
|
|
|
|
var parent = me.node.getParent();
|
|
|
|
if (parent != nil) {
|
|
|
|
parent.removeChild(me.node.getName(), me.node.getIndex());
|
|
|
|
}
|
|
|
|
},
|
|
|
|
distance_from : func(xyz) {
|
2006-03-23 16:16:44 +00:00
|
|
|
return coord_dist_sq(xyz, lonlat2xyz([me.lon, me.lat]));
|
2006-03-22 20:25:58 +00:00
|
|
|
},
|
|
|
|
get_data : func {
|
|
|
|
var n = props.Node.new();
|
|
|
|
props.copy(me.node, n);
|
|
|
|
me.add_derived_props(n);
|
|
|
|
return n;
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
Dynamic = {
|
2006-03-24 15:33:06 +00:00
|
|
|
new : func(path, lon, lat, elev, hdg = nil, pitch = nil, roll = nil) {
|
2006-03-22 20:25:58 +00:00
|
|
|
var m = Model.new(path);
|
|
|
|
m.parents = [Dynamic, Model];
|
|
|
|
|
|
|
|
adjust.setall(lon, lat, elev, hdg, pitch, roll);
|
|
|
|
m.node.getNode("longitude-deg-prop", 1).setValue(adjust.outNode("lon").getPath());
|
|
|
|
m.node.getNode("latitude-deg-prop", 1).setValue(adjust.outNode("lat").getPath());
|
|
|
|
m.node.getNode("elevation-ft-prop", 1).setValue(adjust.outNode("elev").getPath());
|
|
|
|
m.node.getNode("heading-deg-prop", 1).setValue(adjust.outNode("hdg").getPath());
|
|
|
|
m.node.getNode("pitch-deg-prop", 1).setValue(adjust.outNode("pitch").getPath());
|
|
|
|
m.node.getNode("roll-deg-prop", 1).setValue(adjust.outNode("roll").getPath());
|
|
|
|
m.load();
|
|
|
|
return m;
|
|
|
|
},
|
|
|
|
del : func {
|
|
|
|
var parent = me.node.getParent();
|
|
|
|
if (parent != nil) {
|
|
|
|
parent.removeChild(me.node.getName(), me.node.getIndex());
|
|
|
|
}
|
|
|
|
},
|
|
|
|
make_static : func {
|
|
|
|
var static = Static.new(me.path,
|
|
|
|
adjust.get("lon"), adjust.get("lat"), adjust.get("elev"),
|
|
|
|
adjust.get("hdg"), adjust.get("pitch"), adjust.get("roll"));
|
|
|
|
me.del();
|
|
|
|
return static;
|
|
|
|
},
|
|
|
|
distance_from : func(xyz) {
|
|
|
|
var lon = adjust.get("lon");
|
|
|
|
var lat = adjust.get("lat");
|
2006-03-23 16:16:44 +00:00
|
|
|
return coord_dist_sq(xyz, lonlat2xyz([lon, lat]));
|
2006-03-22 20:25:58 +00:00
|
|
|
},
|
|
|
|
get_data : func {
|
|
|
|
var n = props.Node.new();
|
|
|
|
n.getNode("path", 1).setValue(me.path);
|
|
|
|
props.copy(props.globals.getNode("/data/adjust"), n);
|
|
|
|
me.add_derived_props(n);
|
|
|
|
return n;
|
2006-03-15 16:16:47 +00:00
|
|
|
},
|
2006-03-15 14:27:47 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
Static.make_dynamic = func {
|
|
|
|
me.del();
|
|
|
|
return Dynamic.new(me.path, me.lon, me.lat, me.elev, me.hdg, me.pitch, me.roll);
|
|
|
|
};
|
|
|
|
|
2006-03-15 14:27:47 +00:00
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
ModelMgr = {
|
|
|
|
new : func(path) {
|
|
|
|
var m = { parents: [ModelMgr] };
|
2006-03-15 14:27:47 +00:00
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
var click = props.globals.getNode("/sim/input/click", 1);
|
|
|
|
m.lonN = click.getNode("longitude-deg", 1);
|
|
|
|
m.latN = click.getNode("latitude-deg", 1);
|
|
|
|
m.elevN = click.getNode("elevation-ft", 1);
|
2006-03-15 14:27:47 +00:00
|
|
|
|
2006-03-25 18:30:32 +00:00
|
|
|
m.lonN.setValue(0);
|
|
|
|
m.latN.setValue(0);
|
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
m.spacebarN = props.globals.getNode("/controls/engines/engine/starter", 1);
|
|
|
|
m.modelpath = path;
|
2006-03-15 14:27:47 +00:00
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
m.dynamic = nil;
|
|
|
|
m.static = [];
|
|
|
|
m.block = 0;
|
2006-03-25 18:30:32 +00:00
|
|
|
m.count = 0;
|
2006-03-22 20:25:58 +00:00
|
|
|
return m;
|
|
|
|
},
|
|
|
|
click : func {
|
|
|
|
if (me.block) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (me.spacebarN.getBoolValue()) {
|
|
|
|
me.select();
|
|
|
|
} else {
|
|
|
|
me.add_instance();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
add_instance : func {
|
|
|
|
if (me.dynamic != nil) {
|
|
|
|
append(me.static, me.dynamic.make_static());
|
|
|
|
}
|
|
|
|
me.dynamic = Dynamic.new(me.modelpath, me.lonN.getValue(), me.latN.getValue(),
|
2006-03-24 15:33:06 +00:00
|
|
|
me.elevN.getValue());
|
2006-03-23 16:56:55 +00:00
|
|
|
# refresh status line to reset display timer
|
2006-03-25 18:30:32 +00:00
|
|
|
me.count += 1;
|
|
|
|
me.display_status(me.modelpath);
|
2006-03-22 20:25:58 +00:00
|
|
|
},
|
|
|
|
select : func {
|
|
|
|
var click_xyz = lonlat2xyz([me.lonN.getValue(), me.latN.getValue()]);
|
|
|
|
var min_dist = me.dynamic != nil ? me.dynamic.distance_from(click_xyz) : 1000000;
|
|
|
|
var nearest = nil;
|
|
|
|
|
|
|
|
# find nearest static object
|
|
|
|
forindex (var i; me.static) {
|
|
|
|
var dist = me.static[i].distance_from(click_xyz);
|
|
|
|
if (dist < min_dist) {
|
|
|
|
min_dist = dist;
|
|
|
|
nearest = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (nearest != nil) {
|
|
|
|
# swap dynamic with nearest static
|
|
|
|
if (me.dynamic != nil) {
|
|
|
|
var st = me.dynamic.make_static();
|
|
|
|
me.dynamic = me.static[nearest].make_dynamic();
|
|
|
|
me.static[nearest] = st;
|
2006-03-23 16:16:44 +00:00
|
|
|
# actively selected: use this model type
|
|
|
|
me.modelpath = me.dynamic.path;
|
2006-03-22 20:25:58 +00:00
|
|
|
} else {
|
|
|
|
me.dynamic = me.static[nearest].make_dynamic();
|
|
|
|
|
|
|
|
var left = subvec(me.static, 0, nearest);
|
|
|
|
if (nearest + 1 < size(me.static)) {
|
|
|
|
foreach (var v; subvec(me.static, nearest + 1)) {
|
|
|
|
append(left, v);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
me.static = left;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (me.dynamic == nil) { # last one removed
|
|
|
|
return;
|
|
|
|
}
|
2006-03-23 16:16:44 +00:00
|
|
|
me.flash();
|
|
|
|
},
|
|
|
|
flash : func {
|
2006-03-22 20:25:58 +00:00
|
|
|
me.block = 1;
|
2006-03-23 16:16:44 +00:00
|
|
|
var t = 0.33;
|
2006-03-25 18:30:32 +00:00
|
|
|
me.display_status(me.dynamic.path, 1);
|
2006-03-23 16:16:44 +00:00
|
|
|
settimer(func { adjust.set("elev", adjust.get("elev") - 10000) }, t * 1);
|
|
|
|
settimer(func { adjust.set("elev", adjust.get("elev") + 10000) }, t * 2);
|
|
|
|
settimer(func { adjust.set("elev", adjust.get("elev") - 10000) }, t * 3);
|
|
|
|
settimer(func { adjust.set("elev", adjust.get("elev") + 10000) }, t * 4);
|
|
|
|
settimer(func { me.block = 0 }, t * 4.5);
|
2006-03-25 18:30:32 +00:00
|
|
|
settimer(func { me.display_status(me.modelpath) }, 5);
|
2006-03-22 20:25:58 +00:00
|
|
|
},
|
|
|
|
remove_selected : func {
|
|
|
|
if (me.block) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (me.dynamic != nil) {
|
|
|
|
me.dynamic.del();
|
|
|
|
me.dynamic = nil;
|
2006-03-25 18:30:32 +00:00
|
|
|
me.count -= 1;
|
2006-03-22 20:25:58 +00:00
|
|
|
}
|
|
|
|
me.select();
|
|
|
|
},
|
|
|
|
setmodelpath : func(path) {
|
|
|
|
me.modelpath = path;
|
2006-03-25 18:30:32 +00:00
|
|
|
me.display_status(path);
|
|
|
|
},
|
|
|
|
display_status : func(p, m = 0) {
|
|
|
|
var c = [
|
|
|
|
[0.6, 1, 0.6, 1],
|
|
|
|
[1.0, 0.6, 0.0, 1.0],
|
|
|
|
];
|
|
|
|
display.write("(" ~ me.count ~ ") " ~ p, c[m][0], c[m][1], c[m][2], c[m][3]);
|
2006-03-22 20:25:58 +00:00
|
|
|
},
|
|
|
|
get_data : func {
|
|
|
|
var n = props.Node.new();
|
|
|
|
if (me.dynamic != nil) {
|
|
|
|
props.copy(me.dynamic.get_data(), n.getChild("model", 0, 1));
|
|
|
|
}
|
|
|
|
forindex (var i; me.static) {
|
|
|
|
props.copy(me.static[i].get_data(), n.getChild("model", i + 1, 1));
|
|
|
|
}
|
|
|
|
return n;
|
|
|
|
},
|
2006-03-24 00:20:22 +00:00
|
|
|
cycle : func(up) {
|
|
|
|
var i = search(modellist, me.modelpath) + up;
|
|
|
|
if (i < 0) {
|
|
|
|
i = size(modellist) - 1;
|
|
|
|
} elsif (i >= size(modellist)) {
|
|
|
|
i = 0;
|
|
|
|
}
|
|
|
|
me.setmodelpath(modellist[i]);
|
|
|
|
if (me.dynamic != nil) {
|
|
|
|
var st = me.dynamic.make_static();
|
|
|
|
st.path = me.modelpath;
|
|
|
|
me.dynamic.del();
|
|
|
|
me.dynamic = st.make_dynamic();
|
|
|
|
}
|
|
|
|
},
|
2006-03-22 20:25:58 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
2006-03-24 00:20:22 +00:00
|
|
|
incElevator = controls.incElevator;
|
|
|
|
controls.incElevator = func(step, apstep) {
|
|
|
|
if (getprop("/controls/engines/engine/starter")) {
|
|
|
|
modelmgr.cycle(step > 0 ? 1 : -1);
|
|
|
|
} else {
|
|
|
|
incElevator(step, apstep);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
|
|
|
|
|
|
|
|
# what to do on terrain clicks
|
|
|
|
|
|
|
|
#var lastXYZ = lonlat2xyz([getprop("/position/longitude-deg"), getprop("/position/latitude-deg")]);
|
|
|
|
#var lastElev = 0;
|
|
|
|
|
|
|
|
# # print distance to last cursor coordinates (horizontal distance
|
|
|
|
# # doesn't consider elevation and is rather imprecise)
|
|
|
|
# newXYZ = lonlat2xyz([lon, lat]);
|
|
|
|
# var hdist = math.sqrt(coord_dist_sq(lastXYZ, newXYZ) * ERAD);
|
|
|
|
# var vdist = ft2m(elev - lastElev);
|
|
|
|
# var s = hdist < 4 ? sprintf("%.1f m HOR, %.1f m VERT", hdist * 1000, vdist)
|
|
|
|
# : sprintf("%.1f km HOR, %.1f m VERT", hdist, vdist);
|
|
|
|
# screen.log.write(s);
|
|
|
|
|
|
|
|
# lastXYZ = newXYZ;
|
|
|
|
# lastElev = elev;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
scanDirs = func(csv) {
|
|
|
|
var list = [];
|
|
|
|
foreach(var dir; split(",", csv)) {
|
|
|
|
foreach(var m; scan_models(dir)) {
|
|
|
|
append(list, m);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return sort(list);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-03-23 16:16:44 +00:00
|
|
|
|
|
|
|
|
|
|
|
# interface functions -----------------------------------------------------------------------------
|
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
printData = func {
|
|
|
|
var rule = "------------------------------------------------------------------";
|
|
|
|
var data = modelmgr.get_data();
|
|
|
|
var bucket = {};
|
|
|
|
# group all objects of a bucket
|
|
|
|
foreach (var m; data.getChildren("model")) {
|
|
|
|
var stg = m.getNode("stg-path").getValue();
|
|
|
|
var obj = m.getNode("object-line").getValue();
|
|
|
|
if (contains(bucket, stg)) {
|
|
|
|
append(bucket[stg], obj);
|
|
|
|
} else {
|
|
|
|
bucket[stg] = [obj];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
print(rule);
|
|
|
|
foreach (var key; keys(bucket)) {
|
|
|
|
print("\n# ", key);
|
|
|
|
foreach (var obj; bucket[key]) {
|
|
|
|
print(obj);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
print();
|
|
|
|
print(rule);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
exportData = func {
|
|
|
|
savexml = func(name, node) {
|
|
|
|
fgcommand("savexml", props.Node.new({"filename": name, "sourcenode": node}));
|
|
|
|
}
|
|
|
|
var tmp = "save-ufo-data";
|
|
|
|
save = props.globals.getNode(tmp, 1);
|
|
|
|
props.copy(modelmgr.get_data(), save);
|
|
|
|
savexml(getprop("/sim/fg-home") ~ "/ufo-model-export.xml", save.getPath());
|
|
|
|
props.globals.removeChild(tmp);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
removeSelectedModel = func { modelmgr.remove_selected() }
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2006-03-23 16:16:44 +00:00
|
|
|
# init --------------------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
var display = nil;
|
|
|
|
var modellist = nil;
|
|
|
|
var adjust = nil;
|
|
|
|
var modelmgr = nil;
|
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
|
2006-03-23 16:16:44 +00:00
|
|
|
settimer(func {
|
2006-03-23 16:56:55 +00:00
|
|
|
display = screen.window.new(8, 8, 1, 180);
|
2006-03-23 16:16:44 +00:00
|
|
|
display.font = "HELVETICA_12";
|
2006-03-24 15:33:06 +00:00
|
|
|
display.halign = "left";
|
2006-03-22 20:25:58 +00:00
|
|
|
|
2006-03-23 16:16:44 +00:00
|
|
|
modellist = scanDirs(getprop("/source"));
|
|
|
|
adjust = Adjust.new("/data");
|
|
|
|
modelmgr = ModelMgr.new(getprop("/model"));
|
|
|
|
setlistener("/sim/signals/click", func { modelmgr.click() });
|
|
|
|
}, 1);
|
2006-03-22 20:25:58 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# dialogs -----------------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
|
|
var dialog = {};
|
|
|
|
|
|
|
|
showModelSelectDialog = func {
|
|
|
|
name = "ufo-model-select-dialog";
|
|
|
|
|
|
|
|
if (contains(dialog, name)) {
|
|
|
|
closeModelSelectDialog();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
var title = 'Select Model';
|
|
|
|
|
|
|
|
dialog[name] = gui.Widget.new();
|
|
|
|
dialog[name].set("layout", "vbox");
|
|
|
|
dialog[name].set("name", name);
|
2006-03-24 15:33:06 +00:00
|
|
|
dialog[name].set("x", -20);
|
2006-03-22 20:25:58 +00:00
|
|
|
dialog[name].set("pref-width", 600);
|
|
|
|
|
|
|
|
# "window" titlebar
|
|
|
|
titlebar = dialog[name].addChild("group");
|
|
|
|
titlebar.set("layout", "hbox");
|
|
|
|
titlebar.addChild("empty").set("stretch", 1);
|
|
|
|
titlebar.addChild("text").set("label", title);
|
|
|
|
titlebar.addChild("empty").set("stretch", 1);
|
|
|
|
|
|
|
|
dialog[name].addChild("hrule").addChild("dummy");
|
|
|
|
|
|
|
|
w = titlebar.addChild("button");
|
|
|
|
w.set("pref-width", 16);
|
|
|
|
w.set("pref-height", 16);
|
|
|
|
w.set("legend", "");
|
|
|
|
w.set("default", 1);
|
|
|
|
w.set("keynum", 27);
|
|
|
|
w.set("border", 1);
|
|
|
|
w.prop().getNode("binding[0]/command", 1).setValue("nasal");
|
|
|
|
w.prop().getNode("binding[0]/script", 1).setValue("ufo.closeModelSelectDialog()");
|
|
|
|
|
|
|
|
w = dialog[name].addChild("list");
|
|
|
|
w.set("halign", "fill");
|
|
|
|
w.set("pref-height", 300);
|
|
|
|
w.set("property", "/model");
|
|
|
|
forindex (var i; modellist) {
|
|
|
|
w.prop().getChild("value", i, 1).setValue(modellist[i]);
|
|
|
|
}
|
2006-03-23 19:06:18 +00:00
|
|
|
w.prop().getNode("binding[0]/command", 1).setValue("dialog-apply");
|
|
|
|
w.prop().getNode("binding[1]/command", 1).setValue("nasal");
|
|
|
|
w.prop().getNode("binding[1]/script", 1).setValue("ufo.modelmgr.setmodelpath(getprop('/model'))");
|
2006-03-22 20:25:58 +00:00
|
|
|
|
|
|
|
fgcommand("dialog-new", dialog[name].prop());
|
|
|
|
gui.showDialog(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
closeModelSelectDialog = func {
|
|
|
|
var name = "ufo-model-select-dialog";
|
|
|
|
var dlg = props.Node.new({"dialog-name": name});
|
|
|
|
fgcommand("dialog-apply", dlg);
|
|
|
|
fgcommand("dialog-close", dlg);
|
|
|
|
delete(dialog, name);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-03-24 15:33:06 +00:00
|
|
|
showModelAdjustDialog = func {
|
2006-03-15 14:27:47 +00:00
|
|
|
name = "ufo-cursor-dialog";
|
2006-03-22 20:25:58 +00:00
|
|
|
|
|
|
|
if (contains(dialog, name)) {
|
2006-03-15 14:27:47 +00:00
|
|
|
fgcommand("dialog-close", props.Node.new({ "dialog-name" : name }));
|
2006-03-22 20:25:58 +00:00
|
|
|
delete(dialog, name);
|
2006-03-15 14:27:47 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2006-03-24 15:33:06 +00:00
|
|
|
adjust.center_sliders();
|
2006-03-15 14:27:47 +00:00
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
dialog[name] = gui.Widget.new();
|
|
|
|
dialog[name].set("layout", "vbox");
|
|
|
|
dialog[name].set("name", name);
|
2006-03-24 15:33:06 +00:00
|
|
|
dialog[name].set("x", -20);
|
|
|
|
dialog[name].set("y", -20);
|
2006-03-15 14:27:47 +00:00
|
|
|
|
|
|
|
# "window" titlebar
|
2006-03-22 20:25:58 +00:00
|
|
|
titlebar = dialog[name].addChild("group");
|
2006-03-15 14:27:47 +00:00
|
|
|
titlebar.set("layout", "hbox");
|
|
|
|
titlebar.addChild("empty").set("stretch", 1);
|
2006-03-24 15:33:06 +00:00
|
|
|
titlebar.addChild("text").set("label", "Adjust Model");
|
2006-03-15 14:27:47 +00:00
|
|
|
titlebar.addChild("empty").set("stretch", 1);
|
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
dialog[name].addChild("hrule").addChild("dummy");
|
2006-03-15 14:27:47 +00:00
|
|
|
|
|
|
|
w = titlebar.addChild("button");
|
|
|
|
w.set("pref-width", 16);
|
|
|
|
w.set("pref-height", 16);
|
|
|
|
w.set("legend", "");
|
|
|
|
w.set("default", 1);
|
|
|
|
w.set("keynum", 27);
|
|
|
|
w.set("border", 1);
|
|
|
|
w.prop().getNode("binding[0]/command", 1).setValue("nasal");
|
2006-03-22 20:25:58 +00:00
|
|
|
w.prop().getNode("binding[0]/script", 1).setValue("delete(ufo.dialog, \"" ~ name ~ "\")");
|
2006-03-15 14:27:47 +00:00
|
|
|
w.prop().getNode("binding[1]/command", 1).setValue("dialog-close");
|
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
slider = func(legend, col, coarse, fine) {
|
|
|
|
group = dialog[name].addChild("group");
|
2006-03-15 14:27:47 +00:00
|
|
|
group.set("layout", "hbox");
|
|
|
|
group.set("default-padding", 0);
|
2006-03-22 20:25:58 +00:00
|
|
|
|
2006-03-15 14:27:47 +00:00
|
|
|
button = func(leg, step) {
|
|
|
|
b = group.addChild("button");
|
|
|
|
b.set("legend", leg);
|
|
|
|
b.set("pref-width", 22);
|
|
|
|
b.set("pref-height", 22);
|
|
|
|
b.set("live", 1);
|
|
|
|
b.prop().getNode("binding[0]/command", 1).setValue("nasal");
|
2006-03-22 20:25:58 +00:00
|
|
|
b.prop().getNode("binding[0]/script", 1).setValue('ufo.adjust.step("'~legend~'", '~step~')');
|
2006-03-15 14:27:47 +00:00
|
|
|
return b;
|
|
|
|
}
|
|
|
|
|
|
|
|
cl = button("<<", -coarse);
|
|
|
|
fl = button("<", -fine);
|
|
|
|
|
|
|
|
s = group.addChild("slider");
|
2006-03-22 20:25:58 +00:00
|
|
|
s.set("property", adjust.offsetNode(legend).getPath());
|
2006-03-15 14:27:47 +00:00
|
|
|
s.set("legend", legend);
|
|
|
|
s.set("pref-width", 250);
|
|
|
|
s.set("live", 1);
|
|
|
|
s.set("min", -2 * fine);
|
|
|
|
s.set("max", 2 * fine);
|
|
|
|
s.setColor(col[0], col[1], col[2]);
|
|
|
|
s.prop().getNode("binding[0]/command", 1).setValue("dialog-apply");
|
|
|
|
|
|
|
|
fr = button(">", fine);
|
|
|
|
cr = button(">>", coarse);
|
|
|
|
}
|
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
slider("lon", [1.0, 0.6, 0.6], 0.0002, 0.00002);
|
|
|
|
slider("lat", [0.6, 1.0, 0.6], 0.0002, 0.00002);
|
|
|
|
slider("elev", [0.6, 0.6, 1.0], 10, 2);
|
2006-03-15 14:27:47 +00:00
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
slider("hdg", [1.0, 1.0, 0.6], 36, 6);
|
|
|
|
slider("pitch", [1.0, 0.6, 1.0], 36, 6);
|
|
|
|
slider("roll", [0.6, 1.0, 1.0], 36, 6);
|
2006-03-15 14:27:47 +00:00
|
|
|
|
2006-03-24 15:33:06 +00:00
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
g = dialog[name].addChild("group");
|
2006-03-15 16:16:47 +00:00
|
|
|
g.set("layout", "hbox");
|
|
|
|
|
2006-03-24 15:33:06 +00:00
|
|
|
w = g.addChild("text");
|
2006-03-15 16:16:47 +00:00
|
|
|
w.set("halign", "left");
|
2006-03-24 15:33:06 +00:00
|
|
|
w.set("label", "Heading ");
|
|
|
|
|
|
|
|
w = g.addChild("text");
|
|
|
|
w.set("halign", "center");
|
|
|
|
w.set("label", "Sliders");
|
|
|
|
|
|
|
|
w = g.addChild("text");
|
|
|
|
w.set("halign", "right");
|
|
|
|
w.set("label", "Orientation");
|
|
|
|
|
|
|
|
|
|
|
|
g = dialog[name].addChild("group");
|
|
|
|
g.set("layout", "hbox");
|
|
|
|
g.set("default-padding", 2);
|
|
|
|
var wide = 60;
|
|
|
|
var narrow = 55;
|
|
|
|
|
|
|
|
w = g.addChild("button");
|
|
|
|
w.set("halign", "right");
|
|
|
|
w.set("legend", "Reset");
|
2006-03-15 16:16:47 +00:00
|
|
|
w.set("pref-height", 22);
|
2006-03-24 15:33:06 +00:00
|
|
|
w.set("pref-width", wide);
|
2006-03-15 16:16:47 +00:00
|
|
|
w.prop().getNode("binding[0]/command", 1).setValue("nasal");
|
2006-03-24 15:33:06 +00:00
|
|
|
w.prop().getNode("binding[0]/script", 1).setValue("ufo.adjust.orient()");
|
|
|
|
|
|
|
|
w = g.addChild("button");
|
|
|
|
w.set("legend", "Sticky");
|
|
|
|
w.set("one-shot", 0);
|
|
|
|
w.set("pref-height", 22);
|
|
|
|
w.set("pref-width", narrow);
|
|
|
|
w.set("live", 1);
|
|
|
|
w.set("property", adjust.stk_hdgN.getPath());
|
|
|
|
w.prop().getNode("binding[0]/command", 1).setValue("dialog-apply");
|
|
|
|
|
|
|
|
g.addChild("empty").set("stretch", 1);
|
2006-03-15 16:16:47 +00:00
|
|
|
|
|
|
|
w = g.addChild("button");
|
|
|
|
w.set("halign", "center");
|
|
|
|
w.set("legend", "Center");
|
2006-03-15 14:27:47 +00:00
|
|
|
w.set("pref-height", 22);
|
2006-03-24 15:33:06 +00:00
|
|
|
w.set("pref-width", wide);
|
2006-03-15 14:27:47 +00:00
|
|
|
w.prop().getNode("binding[0]/command", 1).setValue("nasal");
|
2006-03-24 15:33:06 +00:00
|
|
|
w.prop().getNode("binding[0]/script", 1).setValue("ufo.adjust.center_sliders()");
|
|
|
|
|
|
|
|
g.addChild("empty").set("stretch", 1);
|
2006-03-15 14:27:47 +00:00
|
|
|
|
2006-03-15 16:16:47 +00:00
|
|
|
w = g.addChild("button");
|
2006-03-24 15:33:06 +00:00
|
|
|
w.set("legend", "Sticky");
|
|
|
|
w.set("one-shot", 0);
|
2006-03-15 16:16:47 +00:00
|
|
|
w.set("pref-height", 22);
|
2006-03-24 15:33:06 +00:00
|
|
|
w.set("pref-width", narrow);
|
|
|
|
w.set("live", 1);
|
|
|
|
w.set("property", adjust.stk_orientN.getPath());
|
|
|
|
w.prop().getNode("binding[0]/command", 1).setValue("dialog-apply");
|
|
|
|
|
|
|
|
w = g.addChild("button");
|
|
|
|
w.set("halign", "left");
|
|
|
|
w.set("legend", "Reset");
|
|
|
|
w.set("pref-height", 22);
|
|
|
|
w.set("pref-width", wide);
|
2006-03-15 16:16:47 +00:00
|
|
|
w.prop().getNode("binding[0]/command", 1).setValue("nasal");
|
2006-03-24 15:33:06 +00:00
|
|
|
w.prop().getNode("binding[0]/script", 1).setValue("ufo.adjust.upright()");
|
2006-03-15 16:16:47 +00:00
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
fgcommand("dialog-new", dialog[name].prop());
|
2006-03-15 14:27:47 +00:00
|
|
|
gui.showDialog(name);
|
2006-03-15 00:29:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2006-03-23 16:16:44 +00:00
|
|
|
# attic -------------------------------------------------------------------------------------------
|
2006-03-22 20:25:58 +00:00
|
|
|
|
|
|
|
|
2006-03-15 14:27:47 +00:00
|
|
|
dumpCoords = func {
|
|
|
|
print("\n---------------------------- UFO -----------------------------");
|
2006-03-15 00:29:53 +00:00
|
|
|
|
|
|
|
var lon = getprop("/position/longitude-deg");
|
|
|
|
var lat = getprop("/position/latitude-deg");
|
|
|
|
var alt_ft = getprop("/position/altitude-ft");
|
|
|
|
var elev_m = getprop("/position/ground-elev-m");
|
|
|
|
var heading = getprop("/orientation/heading-deg");
|
|
|
|
var agl_ft = alt_ft - m2ft(elev_m);
|
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
printf("Longitude: %.6f deg", lon);
|
|
|
|
printf("Latitude: %.6f deg", lat);
|
|
|
|
printf("Altitude ASL: %.4f m (%.4f ft)", ft2m(alt_ft), alt_ft);
|
|
|
|
printf("Altitude AGL: %.4f m (%.4f ft)", ft2m(agl_ft), agl_ft);
|
|
|
|
printf("Heading: %.1f deg", normdeg(heading));
|
|
|
|
printf("Ground Elev: %.4f m (%.4f ft)", elev_m, m2ft(elev_m));
|
2006-03-15 00:29:53 +00:00
|
|
|
print("");
|
2006-03-15 14:27:47 +00:00
|
|
|
print(tile_path(lon, lat));
|
2006-03-22 20:25:58 +00:00
|
|
|
printf("OBJECT_STATIC %.6f %.6f %.4f %.1f", lon, lat, elev_m, normdeg(360 - heading));
|
2006-03-16 16:24:52 +00:00
|
|
|
print("");
|
2006-03-15 15:50:35 +00:00
|
|
|
|
2006-03-16 16:24:52 +00:00
|
|
|
var hdg = normdeg(heading + getprop("/sim/current-view/goal-pitch-offset-deg"));
|
|
|
|
var fgfs = sprintf("$ fgfs --aircraft=ufo --lon=%.6f --lat=%.6f --altitude=%.2f --heading=%.1f",
|
|
|
|
lon, lat, agl_ft, hdg);
|
|
|
|
print(fgfs);
|
2006-03-15 15:50:35 +00:00
|
|
|
|
|
|
|
|
2006-03-16 16:24:52 +00:00
|
|
|
print("\n\n--------------------------- Cursor ---------------------------");
|
2006-03-15 20:37:38 +00:00
|
|
|
|
2006-03-22 20:25:58 +00:00
|
|
|
var alt = cursor.val["alt"].get();
|
|
|
|
printf("Longitude: %.6f deg", var clon = cursor.val["lon"].get());
|
|
|
|
printf("Latitude: %.6f deg", var clat = cursor.val["lat"].get());
|
|
|
|
printf("Altitude ASL: %.4f m (%.4f ft)", var celev = ft2m(alt), alt);
|
|
|
|
printf("Heading: %.1f deg", var chdg = normdeg(cursor.val["hdg"].get()));
|
|
|
|
printf("Pitch: %.1f deg", normdeg(cursor.val["pitch"].get()));
|
|
|
|
printf("Roll: %.1f deg", normdeg(cursor.val["roll"].get()));
|
2006-03-15 15:50:35 +00:00
|
|
|
print("");
|
2006-03-15 00:29:53 +00:00
|
|
|
print("--------------------------------------------------------------");
|
2006-03-16 16:24:52 +00:00
|
|
|
saveData();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2006-03-23 16:16:44 +00:00
|
|
|
|