# Nasal files to be loaded at start # the order could be important as some files need other one to be loaded first files_to_load = [ 'lib.nas', # some useful functions, should stay first loaded 'radios.nas', # all about radios COMM, NAV, XPDR 'knobs.nas', # handles knobs 'buttons.nas', # handles knobs and buttons 'softkeys.nas',# handles softkeys and menus items 'maps/route.nas', 'maps/navaids.nas', 'maps/tiles.nas', 'maps/tcas.nas', 'maps/topo.nas', 'maps/track.nas', 'map.nas', # moves the maps 'display.nas', 'menu.nas', # manage windows 'core.nas', # the scheduler and switch on button 'afcs.nas', # Automatic Flight Control System 'annunciations.nas', # in flight tests ]; # 'routes.nas', # manages flightplans and routes # 'display.nas', # display messages and popups # 'infos.nas', # get informations from environment # 'mud.nas', # displays simple embedded GUI (Multi-Use Display) var flightdeck = {}; var autopilot = {}; var annunciations = {}; var units = { speed : { from_kt : 1, from_kmh : MPS2KT * 3.6, }, altitude : { from_ft : 1, from_m : M2FT, }, vspeed : { from_fpm : 1, from_mpm : M2FT, }, distance : { from_m : M2NM / 1000, from_nm : 1, }, temperature : { from_C : func (c) return c, from_F : func (f) return (f -32) / 1.8, }, volume: { from_l: 1, from_gal: 3.78541178, from_m3: 1000, }, pressure: { from_inhg: 1, from_hpa: 0.02953, }, }; var data = { # set of data common to all devices roll : 0, pitch : 0, vsi : 0, ias : 0, alt : 0, hdg : 0, trk : 0, wow : 1, lat : 0, lon : 0, aoa : 0, time : '23:59:59', fpSize : 0, tcas: [], tcas_level: 0, Vspeeds : {}, settings: { units: { pressure: 'inhg', altitude: 'ft', speed: 'knots', distance: 'nm', vertical: 'fpm', }, time: { label: 'GMT', actual: ' GMT >', }, }, timers : { '20Hz': maketimer ( 0.05, func { data.roll = getprop('/orientation/roll-deg'); data.pitch = getprop('orientation/pitch-deg'); data.vsi = getprop('/instrumentation/vertical-speed-indicator/indicated-speed-fpm') * units.vspeed.from_fpm; data.ias = getprop('/velocities/airspeed-kt') * units.speed.from_kt; data.alt = getprop('/instrumentation/altimeter/indicated-altitude-ft') * units.altitude.from_ft; data.hdg = getprop('/orientation/heading-deg'); data.trk = data.wow ? data.hdg : getprop('/orientation/track-deg'); data.aoa = getprop('/orientation/alpha-deg'); } ), '1Hz': maketimer ( 1, func { data.wow = getprop('/gear/gear/wow'); data.lat = getprop('/position/latitude-deg'); data.lon = getprop('/position/longitude-deg'); } ), }, listeners : {}, }; var zkv = cdi = radios = alerts = infos = cursors = afcs = eis = misc = nil; var init_props = func { zkv = props.globals.getNode('/instrumentation/zkv1000',1); foreach (var d; zkv.getChildren()) if (d.getNode('status') != nil) flightdeck[d.getName()] = nil; zkv.getNode('emission',1).setDoubleValue(0.5); zkv.getNode('body-emission',1).setDoubleValue(0.0); zkv.getNode('body-texture',1).setValue(''); zkv.getNode('display-brightness-norm',1).setDoubleValue(0.5); zkv.getNode('lightmap',1).setIntValue(0); if (zkv.getNode('size-factor').getValue() == nil) zkv.getNode('size-factor',1).setDoubleValue(1.0); if (zkv.getValue('flightplans') != nil and io.stat(zkv.getValue('flightplans')) == "dir") data.flightplans = zkv.getValue('flightplans'); else data.flightplans = getprop('/sim/fg-home') ~ '/Export'; radios = zkv.getNode('radios', 1); radios.getNode('nav1-selected',1).setIntValue(0); radios.getNode('nav1-standby',1).setIntValue(0); radios.getNode('nav2-selected',1).setIntValue(0); radios.getNode('nav2-standby',1).setIntValue(0); radios.getNode('nav-tune',1).setIntValue(0); radios.getNode('nav-freq-mhz',1).alias('/instrumentation/nav/frequencies/standby-mhz'); radios.getNode('comm1-selected',1).setIntValue(1); radios.getNode('comm2-selected',1).setIntValue(0); radios.getNode('comm-tune',1).setIntValue(0); radios.getNode('comm-freq-mhz',1).alias('/instrumentation/comm/frequencies/standby-mhz'); radios.getNode('xpdr-mode',1).setValue('GND'); radios.getNode('brg1-source',1).setValue('OFF'); radios.getNode('brg2-source',1).setValue('OFF'); cdi = zkv.getNode('cdi', 1); cdi.getNode('source', 1).setValue('OFF'); cdi.getNode('no-flag', 1).setBoolValue(0); cdi.getNode('FROM-flag', 1).alias('no-flag'); cdi.getNode('TO-flag', 1).alias('no-flag'); cdi.getNode('course', 1); cdi.getNode('course-deflection', 1); cdi.getNode('radial', 1); cdi.getNode('in-range', 1); alerts = zkv.getNode('alerts',1); alerts.getNode('traffic-proximity',1).setIntValue(0); alerts.getNode('marker-beacon', 1).setIntValue(0); alerts.getNode('warning', 1).setBoolValue(0); alerts.getNode('alert', 1).setBoolValue(0); alerts.getNode('message', 1).setValue(''); foreach (var v; alerts.getChildren()) if (string.match(v.getName(), 'V[a-z0-9]*')) data.Vspeeds[v.getName()] = v.getValue(); var aoa = alerts.getValue('stall-aoa'); data['stall-aoa'] = (aoa == nil or aoa == 0) ? 9999 : aoa; aoa = alerts.getValue('approach-aoa'); if (aoa != nil) data['approach-aoa'] = aoa; afcs = zkv.getNode('afcs',1); afcs.getNode('fd-bars-visible',1).setBoolValue(0); afcs.getNode('alt-bug-visible',1).setBoolValue(0); afcs.getNode('heading-bug-deg',1).setDoubleValue(int(getprop('/orientation/heading-deg'))); afcs.getNode('target-pitch-deg',1).setDoubleValue(0.0); afcs.getNode('selected-alt-ft',1).setIntValue(2000); afcs.getNode('selected-alt-ft-diff',1).setDoubleValue(0.0); afcs.getNode('selected-ias-kt-diff',1).setDoubleValue(0.0); afcs.getNode('vertical-speed-fpm',1).setDoubleValue(0.0); afcs.getNode('roll-armed', 1).setBoolValue(0); afcs.getNode('pitch-armed', 1).setBoolValue(0); afcs.getNode('roll-armed-mode-text',1).setValue(''); afcs.getNode('roll-active-mode-text',1).setValue(''); afcs.getNode('roll-armed-mode',1).setIntValue(0); afcs.getNode('roll-active-mode',1).setIntValue(0); afcs.getNode('roll-active-mode-blink',1).setBoolValue(0); afcs.getNode('pit-armed-mode-text',1).setValue(''); afcs.getNode('pit-active-mode-text',1).setValue(''); afcs.getNode('pit-armed-mode',1).setIntValue(0); afcs.getNode('pit-active-mode',1).setIntValue(0); afcs.getNode('pit-active-mode-blink',1).setBoolValue(0); afcs.getNode('route',1); data.flightplans = getprop('/sim/fg-home') ~ '/Export'; misc = zkv.getNode('misc',1); misc.getNode('alt-setting-inhg',1).setDoubleValue(getprop('/instrumentation/altimeter/setting-inhg')); eis = zkv.getNode('eis',1); eis.getNode('fuel-qty-at-start', 1).setValue( getprop('/consumables/fuel/tank/level-gal_us') + getprop('/consumables/fuel/tank/level-gal_us')); var tiles_defaults = { # see https://www.wikimedia.org/wiki/Maps server: 'maps.wikimedia.org', type: 'osm-intl', apikey: '', format: 'png', template: 'https://{server}/{type}/{z}/{x}/{y}.{format}{apikey}', }; foreach (var v; keys(tiles_defaults)) { var val = getprop('/sim/online-tiles-' ~ v); data['tiles-' ~ v] = val != nil ? val : tiles_defaults[v]; } props.globals.getNode('/instrumentation/transponder/id-code',1).setIntValue(1200); props.globals.getNode('/instrumentation/transponder/serviceable',1).setBoolValue(1); props.globals.getNode('/autopilot/settings/heading-bug-deg', 1).alias(afcs.getNode('heading-bug-deg').getPath()); props.globals.getNode('/autopilot/settings/target-alt-ft',1).alias(afcs.getNode('selected-alt-ft').getPath()); props.globals.getNode('/autopilot/settings/target-speed-kt',1).setDoubleValue(0.0); props.globals.getNode('/autopilot/settings/vertical-speed-fpm',1).setDoubleValue(0.0); props.globals.getNode('/autopilot/internal/target-pitch-deg',1).setDoubleValue(0.0); props.globals.getNode('/autopilot/internal/flc-altitude-pitch-deg',1).setDoubleValue(0.0); props.globals.getNode('/autopilot/internal/flc-airspeed-pitch-deg',1).setDoubleValue(0.0); props.globals.getNode('/autopilot/internal/target-roll-deg',1).setDoubleValue(0.0); props.globals.getNode('/autopilot/locks/pitch',1).setValue(''); props.globals.getNode('/autopilot/locks/roll',1).setValue(''); props.globals.getNode('/autopilot/locks/passive-mode', 1).setIntValue(1); data.zkv1000_reldir = io.dirname(getprop('/nasal/zkv1000/file')); data.zkv1000_dir = string.normpath( io.dirname(getprop('/sim/aircraft-dir')) ~ '/' ~ string.replace(data.zkv1000_reldir, split('/', data.zkv1000_reldir)[0], '') ) ~ '/'; } var load_nasal = func { var nasal_dir = data.zkv1000_dir ~ 'Nasal/'; for (var i = 0; i < size(files_to_load); i += 1) io.load_nasal(nasal_dir ~ files_to_load[i], 'zkv1000'); files_to_load = nil; } var load_multikey = func { fgcommand('loadxml', props.Node.new({ 'filename': data.zkv1000_dir ~ 'Systems/multikey.xml', 'targetnode': '/input/keyboard/' })); multikey.init(); } var load_settings = func { var settings_file = getprop('/sim/fg-home') ~ '/aircraft-data/zkv1000.xml'; if (io.stat(settings_file) != nil) { fgcommand('loadxml', props.Node.new({ filename: settings_file, targetnode: zkv.getNode('save', 1).getPath() })); var xmlsettings = zkv.getNode('save/' ~ getprop('/sim/aircraft')); if (xmlsettings != nil) { foreach (var domain; keys(units)) foreach (var from; keys(units[domain])) { if (xmlsettings.getNode(domain) != nil) { var unit_value = xmlsettings.getNode(domain).getValue(from); if (unit_value != nil) { if (typeof(units[domain][from]) == 'scalar') units[domain][from] = unit_value; if (typeof(units[domain][from]) == 'func') { units[domain][from] = compile(unit_value); units[domain][from](0); } } } } data.settings.units.pressure = xmlsettings.getNode('pressure').getValue(); } zkv.getNode('save').remove(); } foreach (var V; keys(data.Vspeeds)) data.Vspeeds[V] *= units.speed.from_kt; } var zkv1000_init = func { removelistener(init); init_props(); load_settings(); load_multikey(); load_nasal(); msg('loaded'); if (zkv.getValue('auto-power')) { if (zkv.getNode('serviceable',1).getAttribute("alias") == 1) { data.timers.serviceable = maketimer(1, func { if (zkv.getNode('serviceable').getValue() != 0) zkv1000.powerOn(); }); data.timers.serviceable.start(); } else { var prop = zkv.getNode('serviceable',1).getPath(); data.listeners[prop] = setlistener(prop, zkv1000.powerOn, 0, 0); } } } var init = setlistener('/sim/signals/fdm-initialized', zkv1000_init, 0, 0);