323 lines
12 KiB
Text
323 lines
12 KiB
Text
var mode = 0; #current mode
|
|
var displayed_screen = 0; #screenModeAndSettings
|
|
var page = 0; #current page
|
|
var blocked = 0; #boolean: 0 -> possible to cycle pages
|
|
var isOn = 0; #ON/OFF: 0 -> OFF
|
|
var freq = 1; #settimer frequency (in sec)
|
|
var screen = []; #array containing all screens
|
|
var line = []; #array containing the displayed lines
|
|
var routes = []; #array containing the preprogrammed tasks
|
|
var alt_unit_full_name = ["Feet", "Meters"];
|
|
var dist_unit_full_name = ["Nautic Miles", "Kilometers"];
|
|
var spd_unit_full_name = ["Knots", "KM/H"];
|
|
var alt_unit_short_name = ["ft", "m"];
|
|
var dist_unit_short_name = ["nm", "km"];
|
|
var spd_unit_short_name = ["kt", "km/h"];
|
|
var spd_unit = 0;
|
|
var dist_unit = 0;
|
|
var alt_unit = 0;
|
|
var startpos = nil; #geo.nas aircraft position
|
|
var waypointindex = 0; #step in actual task
|
|
var thresold_alert = [120, 60, 30, 15];
|
|
var thresold_alert_index = 1;
|
|
var thresold_next_waypoint = 5;
|
|
NOT_YET_IMPLEMENTED = [
|
|
"",
|
|
" NOT",
|
|
" YET",
|
|
"IMPLEMENTED",
|
|
""
|
|
];
|
|
var LINES = 5; #lines in display
|
|
var page_list = [
|
|
[0,0,0,0,0,0], #0 ModeAndSettings: 1 page for mode, 5 pages for settings
|
|
[1,2,3], #1 PositionMain, Odometers, WindInfos
|
|
[5,4,1,2,3,6,7], #2 AirportMain, NavigationMain, PositionMain, Odometers, WindInfos, AirportInfos, SearchAirport
|
|
[8,4,1,2,3,9], #3 TurnpointSelect, NavigationMain, PositionMain, Odometers, WindInfos, TurnpointInfos
|
|
[10,4,1,2,3,11,12], #4 TaskSelect, NavigationMain, PositionMain, Odometers, WindInfos, WaypointInfos, WaypointsList
|
|
[13] #5 Edit (special mode for editing waypoint, called from other modes)
|
|
];
|
|
|
|
#to ft m
|
|
var alt_conv = [[1.0000,0.3048], #from ft
|
|
[3.2808,1.0000]]; #from m
|
|
|
|
#to nm km m
|
|
var dist_conv = [[1.00000 ,1.852, 1852], #from nm
|
|
[0.53996 ,1.000, 1000], #from km
|
|
[0.00053996,0.001, 1.00]]; #from m
|
|
|
|
var gps_data = props.globals.getNode("/instrumentation/gps",1);
|
|
var gps_wp = gps_data.getNode("wp",1);
|
|
|
|
#### warps for buttons and knobs ########################################"
|
|
var right_knob = func(dir) { #manage right knob, depends of displayed screen
|
|
isOn > 0 or return;
|
|
screen[displayed_screen].right(dir);
|
|
refresh_display();
|
|
}
|
|
|
|
var enter_button = func() { #manage enter button, depends of displayed screen
|
|
isOn > 0 or return;
|
|
screen[displayed_screen].enter();
|
|
refresh_display();
|
|
}
|
|
|
|
var escape_button = func() { #manage escape button, depends of displayed screen
|
|
isOn > 0 or return;
|
|
screen[displayed_screen].escape();
|
|
refresh_display();
|
|
}
|
|
|
|
var start_button = func() { #manage start button, depends of displayed screen
|
|
isOn > 0 or return;
|
|
screen[displayed_screen].start();
|
|
refresh_display();
|
|
}
|
|
|
|
var left_knob = func(dir) { #manage left button, cycle in mode's pages if not blocked
|
|
isOn > 0 or return;
|
|
if (blocked == 0) {
|
|
if (displayed_screen == 13 and dir) {
|
|
mode = screenEdit.previous_mode;
|
|
page = screenEdit.previous_page;
|
|
}
|
|
page = cycle(size(page_list[mode]), page, dir);
|
|
displayed_screen = page_list[mode][page];
|
|
}
|
|
refresh_display();
|
|
}
|
|
|
|
var select_mode = func(dir) { #manage mode knob, cycle into available modes
|
|
isOn > 0 or return;
|
|
blocked = 0;
|
|
if (displayed_screen != 0) {
|
|
displayed_screen = 0; #screenModeAndSettings
|
|
page = 0;
|
|
mode = 0;
|
|
}
|
|
elsif (page == 0)
|
|
screen[0].mode_ = cycle(size(screen[0].available_modes), screen[0].mode_, dir);
|
|
refresh_display();
|
|
}
|
|
|
|
var power_knob = func() { #manage POWER knob
|
|
if (arg[0] > 0 and isOn < 11) isOn += 1;
|
|
elsif (arg[0] < 0 and isOn > 0) isOn -= 1;
|
|
else return;
|
|
props.globals.getNode("/instrumentation/zkv500/power",1).setIntValue(isOn);
|
|
var light = 0;
|
|
if (isOn > 0 and getprop("instrumentation/gps/serviceable") != 0)
|
|
light = (isOn - 1)/20;
|
|
props.globals.getNode("/instrumentation/zkv500/retro-light").setDoubleValue(light);
|
|
refresh_display();
|
|
}
|
|
|
|
### useful funcs #########################################################
|
|
var display = func () { #display the array line[]
|
|
for (var i = 0; i < LINES; i += 1) line[i].setValue(arg[0][i]);
|
|
}
|
|
|
|
var browse = func (entries_nbr, index_pointer, index_page,dir) {
|
|
#browse multipaged entries, returns [pointer in page, page]
|
|
nl = entries_nbr - (index_page * LINES) > LINES ? LINES : math.mod(entries_nbr - (index_page * LINES), LINES);
|
|
if (index_pointer + 1 == nl) {
|
|
np = int(entries_nbr / LINES) + (math.mod(entries_nbr,LINES) ? 1 : 0);
|
|
index_page = cycle(np, index_page, dir);
|
|
}
|
|
index_pointer = cycle(nl, index_pointer, dir);
|
|
return [index_pointer, index_page];
|
|
}
|
|
|
|
var cycle = func (entries_nbr, actual_entrie, dir) {
|
|
#cycle through entries, return entry index
|
|
entries_nbr -= 1;
|
|
if (dir == 1 and actual_entrie == entries_nbr) return 0;
|
|
elsif (dir == -1 and actual_entrie == 0) return entries_nbr;
|
|
else return actual_entrie + dir;
|
|
}
|
|
|
|
var refresh_display = func() { #refresh displayed lines, settimer if necessary
|
|
screen[displayed_screen].lines();
|
|
if (isOn and 0 < displayed_screen < 5 ) settimer(func { refresh_display(); }, freq, 1);
|
|
}
|
|
|
|
var seconds_to_string = func (time) { #converts secs (double) in string "hh:mm:ss"
|
|
var hh = int(time / 3600);
|
|
if (hh > 100) return "--:--:--";
|
|
var mm = int((time - (hh * 3600)) / 60);
|
|
var ss = int(time - (hh * 3600 + mm * 60));
|
|
return sprintf("%02d:%02d:%02d", hh, mm, ss);
|
|
}
|
|
|
|
### route management ######################################################
|
|
var list_routes = func { #load preprogrammed tasks
|
|
routes = [];
|
|
var path = getprop("/sim/fg-home") ~ "/Routes";
|
|
var s = io.stat(path);
|
|
if (s != nil and io.isdir(s[2])) {
|
|
foreach (var file; directory(path))
|
|
if (file[0] != 46) append(routes, file);
|
|
# size(routes) != 0 or return;
|
|
# routes = sort(routes, func(a,b) {
|
|
# num(a[1]) == nil or num(b[1]) == nil ? cmp(a[1], b[1]) : a[1] - b[1];
|
|
# });
|
|
# print(size(routes));
|
|
# foreach (var r; routes) print (r ~ ":" ~ r[0]);
|
|
}
|
|
return size(routes);
|
|
}
|
|
|
|
var add_waypoint = func (ID, name, type, coord) { #add a waypoint to a route
|
|
var waypoint = gps_data.getNode("route/Waypoint["~screenWaypointsList.n~"]/",1);
|
|
screenWaypointsList.n += 1;
|
|
waypoint.getNode("ID",1).setValue(ID);
|
|
waypoint.getNode("latitude-deg",1).setDoubleValue(coord[0]);
|
|
waypoint.getNode("longitude-deg",1).setDoubleValue(coord[1]);
|
|
waypoint.getNode("altitude-ft",1).setDoubleValue(coord[2]*alt_conv[1][0]);
|
|
waypoint.getNode("name",1).setValue(name);
|
|
waypoint.getNode("desc",1).setValue("no infos");
|
|
waypoint.getNode("waypoint-type",1).setValue(type);
|
|
}
|
|
|
|
var save_route = func { #save the route
|
|
screenWaypointsList.n != 0 or return;
|
|
var first_id = gps_data.getNode("route/Waypoint/ID").getValue();
|
|
var last_id = gps_data.getNode("route/Waypoint["~(screenWaypointsList.n - 1)~"]/ID").getValue();
|
|
var path = getprop("/sim/fg-home") ~ "/Export/"~first_id~"-"~last_id~".xml";
|
|
var args = props.Node.new({ filename : path });
|
|
var export = args.getNode("data", 1);
|
|
props.copy(gps_data.getNode("route"), export);
|
|
fgcommand("savexml", args);
|
|
}
|
|
|
|
var waypointAlert = func { #alert pilot about waypoint approach
|
|
mode > 0 or return;
|
|
var ttw = gps_wp.getNode("wp[1]/TTW",1).getValue();
|
|
var ttw_secs = 9999;
|
|
if (string.isdigit(ttw[0]))
|
|
ttw_secs = num(substr(ttw,0,2))*3600 + num(substr(ttw,3,2))*60 + num(substr(ttw,6,2));
|
|
|
|
if (ttw_secs < thresold_alert[thresold_alert_index])
|
|
gps_data.getNode("waypoint-alert",1).setBoolValue(1);
|
|
else
|
|
gps_data.getNode("waypoint-alert",1).setBoolValue(0);
|
|
|
|
if (mode == 4 and ttw_secs < thresold_next_waypoint)
|
|
screenNavigationMain.nextWaypoint();
|
|
}
|
|
|
|
### turnpoints management ######################################################
|
|
var load_bookmarks = func { #load turnpoints
|
|
var n = 0;
|
|
gps_data.getNode("bookmarks",1).removeChildren("bookmark");
|
|
var file = getprop("/sim/fg-home") ~ "/Export/bookmarks.xml";
|
|
var s = io.stat(file);
|
|
if (s != nil) {
|
|
fgcommand("loadxml", props.Node.new({
|
|
"filename": file,
|
|
"targetnode": "/instrumentation/gps/bookmarks"
|
|
}));
|
|
foreach (var c ;props.globals.getNode("/instrumentation/gps/bookmarks").getChildren("bookmark")) n += 1;
|
|
}
|
|
else
|
|
print(file ~ " not found...");
|
|
return n;
|
|
}
|
|
|
|
var save_bookmarks = func { #save turnpoints
|
|
var path = getprop("/sim/fg-home") ~ "/Export/bookmarks.xml";
|
|
var args = props.Node.new({ filename : path });
|
|
var export = args.getNode("data", 1);
|
|
props.copy(gps_data.getNode("bookmarks"), export);
|
|
fgcommand("savexml", args);
|
|
}
|
|
|
|
var add_bookmark = func (ID, name, type, coord) { #add turnpoint
|
|
var bookmark = gps_data.getNode("bookmarks/bookmark["~screenTurnpointSelect.n~"]/",1);
|
|
screenTurnpointSelect.n += 1;
|
|
bookmark.getNode("ID",1).setValue(ID);
|
|
bookmark.getNode("latitude-deg",1).setDoubleValue(coord[0]);
|
|
bookmark.getNode("longitude-deg",1).setDoubleValue(coord[1]);
|
|
bookmark.getNode("altitude-ft",1).setDoubleValue(coord[2]*alt_conv[1][0]);
|
|
bookmark.getNode("desc",1).setValue("no infos");
|
|
bookmark.getNode("name",1).setValue(name);
|
|
bookmark.getNode("waypoint-type",1).setValue(type);
|
|
save_bookmarks();
|
|
}
|
|
|
|
var EditMode = func (length, title, start_command, numcar = 0) {
|
|
#special mode for editing simple text
|
|
screenEdit.previous_mode = mode;
|
|
screenEdit.previous_page = page;
|
|
mode = 5; #ID edition
|
|
page = 0;
|
|
screenEdit.init(length, title, start_command, numcar);
|
|
}
|
|
|
|
### initialisation stuff ###################################################
|
|
var load_screens = func {
|
|
var zkv500_dir = getprop("/sim/fg-root") ~ "/Aircraft/Instruments-3d/zkv500/";
|
|
io.load_nasal(zkv500_dir ~ "AirportScreens.nas","zkv500");
|
|
io.load_nasal(zkv500_dir ~ "TurnpointScreens.nas","zkv500");
|
|
io.load_nasal(zkv500_dir ~ "MainScreens.nas","zkv500");
|
|
io.load_nasal(zkv500_dir ~ "TaskScreens.nas","zkv500");
|
|
}
|
|
|
|
var organize_screens = func {
|
|
screen = []; #empty screens
|
|
append(screen, zkv500.screenModeAndSettings); #0
|
|
append(screen, zkv500.screenPositionMain); #1
|
|
append(screen, zkv500.screenOdometers); #2
|
|
append(screen, zkv500.screenWindInfos); #3
|
|
append(screen, zkv500.screenNavigationMain); #4
|
|
append(screen, zkv500.screenAirportMain); #5
|
|
append(screen, zkv500.screenAirportInfos); #6
|
|
append(screen, zkv500.screenSearchAirport); #7
|
|
append(screen, zkv500.screenTurnpointSelect); #8
|
|
append(screen, zkv500.screenTurnpointInfos); #9
|
|
append(screen, zkv500.screenTaskSelect); #10
|
|
append(screen, zkv500.screenWaypointInfos); #11
|
|
append(screen, zkv500.screenWaypointsList); #12
|
|
append(screen, zkv500.screenEdit); #13
|
|
}
|
|
|
|
var init_gps_variables = func {
|
|
mode = 0;
|
|
page = 0;
|
|
displayed_screen = 0; #screenModeAndSettings
|
|
blocked = 0; #unlock left_knob
|
|
#isOn = 0; #start OFF
|
|
startpos = nil; #unset start position
|
|
waypointindex = 0; #route waypoint index on beginning
|
|
}
|
|
|
|
var init_gps_props = func {
|
|
for (var i = 0; i < LINES; i += 1) {
|
|
append(line, props.globals.getNode("/instrumentation/zkv500/line[" ~ i ~ "]", 1));
|
|
line[i].setValue("");
|
|
}
|
|
props.globals.getNode("/instrumentation/zkv500/retro-light",1).setDoubleValue(0);
|
|
props.globals.getNode("/instrumentation/zkv500/power",1).setIntValue(0);
|
|
aircraft.light.new("/sim/model/gps/redled", [0.1, 0.1, 0.1, 0.7], "/instrumentation/gps/waypoint-alert");
|
|
aircraft.light.new("/sim/model/gps/greenled", [0.6, 0.3], "/instrumentation/gps/message-alert");
|
|
startpos = geo.Coord.new(geo.aircraft_position());
|
|
screenPositionMain.begin_time = props.globals.getNode("/sim/time/elapsed-sec",1).getValue();
|
|
setlistener("/instrumentation/gps/wp/wp[1]/TTW", waypointAlert, 0, 0);
|
|
setlistener("/instrumentation/gps/serviceable", func {
|
|
if (getprop("/instrumentation/gps/serviceable") == 0)
|
|
setprop("/instrumentation/zkv500/retro-light", 0);
|
|
elsif (isOn > 0)
|
|
setprop("/instrumentation/zkv500/retro-light", (isOn - 1)/20);
|
|
}, 0, 0);
|
|
}
|
|
|
|
var init = func() {
|
|
load_screens();
|
|
organize_screens();
|
|
init_gps_variables();
|
|
init_gps_props();
|
|
print("GPS... initialized");
|
|
}
|
|
|
|
setlistener("/sim/signals/fdm-initialized",init);
|