# A3XX Electronic Centralised Aircraft Monitoring System # Copyright (c) 2019 Jonathan Redpath (legoboyvdlp) var lines = [props.globals.getNode("ECAM/msg/line1", 1), props.globals.getNode("ECAM/msg/line2", 1), props.globals.getNode("ECAM/msg/line3", 1), props.globals.getNode("ECAM/msg/line4", 1), props.globals.getNode("ECAM/msg/line5", 1), props.globals.getNode("ECAM/msg/line6", 1), props.globals.getNode("ECAM/msg/line7", 1), props.globals.getNode("ECAM/msg/line8", 1)]; var linesCol = [props.globals.getNode("ECAM/msg/linec1", 1), props.globals.getNode("ECAM/msg/linec2", 1), props.globals.getNode("ECAM/msg/linec3", 1), props.globals.getNode("ECAM/msg/linec4", 1), props.globals.getNode("ECAM/msg/linec5", 1), props.globals.getNode("ECAM/msg/linec6", 1), props.globals.getNode("ECAM/msg/linec7", 1), props.globals.getNode("ECAM/msg/linec8", 1)]; var rightLines = [props.globals.getNode("ECAM/rightmsg/line1", 1), props.globals.getNode("ECAM/rightmsg/line2", 1), props.globals.getNode("ECAM/rightmsg/line3", 1), props.globals.getNode("ECAM/rightmsg/line4", 1), props.globals.getNode("ECAM/rightmsg/line5", 1), props.globals.getNode("ECAM/rightmsg/line6", 1), props.globals.getNode("ECAM/rightmsg/line7", 1), props.globals.getNode("ECAM/rightmsg/line8", 1)]; var rightLinesCol = [props.globals.getNode("ECAM/rightmsg/linec1", 1), props.globals.getNode("ECAM/rightmsg/linec2", 1), props.globals.getNode("ECAM/rightmsg/linec3", 1), props.globals.getNode("ECAM/rightmsg/linec4", 1), props.globals.getNode("ECAM/rightmsg/linec5", 1), props.globals.getNode("ECAM/rightmsg/linec6", 1), props.globals.getNode("ECAM/rightmsg/linec7", 1), props.globals.getNode("ECAM/rightmsg/linec8", 1)]; var statusLines = [props.globals.getNode("ECAM/status/line1", 1), props.globals.getNode("ECAM/status/line2", 1), props.globals.getNode("ECAM/status/line3", 1), props.globals.getNode("ECAM/status/line4", 1), props.globals.getNode("ECAM/status/line5", 1), props.globals.getNode("ECAM/status/line6", 1), props.globals.getNode("ECAM/status/line7", 1), props.globals.getNode("ECAM/status/line8", 1)]; var statusLinesCol = [props.globals.getNode("ECAM/status/linec1", 1), props.globals.getNode("ECAM/status/linec2", 1), props.globals.getNode("ECAM/status/linec3", 1), props.globals.getNode("ECAM/status/linec4", 1), props.globals.getNode("ECAM/status/linec5", 1), props.globals.getNode("ECAM/status/linec6", 1), props.globals.getNode("ECAM/status/linec7", 1), props.globals.getNode("ECAM/status/linec8", 1)]; var leftOverflow = props.globals.initNode("/ECAM/warnings/overflow-left", 0, "BOOL"); var rightOverflow = props.globals.initNode("/ECAM/warnings/overflow-right", 0, "BOOL"); var overflow = props.globals.initNode("/ECAM/warnings/overflow", 0, "BOOL"); var dc_ess = props.globals.getNode("systems/electrical/bus/dc-ess", 1); var lights = [props.globals.initNode("/ECAM/warnings/master-warning-light", 0, "BOOL"), props.globals.initNode("/ECAM/warnings/master-caution-light", 0, "BOOL")]; var aural = [props.globals.initNode("/sim/sound/warnings/crc", 0, "BOOL"), props.globals.initNode("/sim/sound/warnings/chime", 0, "BOOL"), props.globals.initNode("/sim/sound/warnings/cricket", 0, "BOOL")]; var warningFlash = props.globals.initNode("/ECAM/warnings/master-warning-flash", 0, "BOOL"); var lineIndex = 0; var rightLineIndex = 0; var statusIndex = 0; var flash = 0; var hasCleared = 0; var statusFlag = 0; var counter = 0; var noMainMsg = 0; var storeFirstWarning = nil; var warningNodes = { Logic: { crossbleedFault: props.globals.initNode("/ECAM/warnings/logic/crossbleed-fault"), crossbleedWai: props.globals.initNode("/ECAM/warnings/logic/crossbleed-wai"), bleed1LoTempUnsuc: props.globals.initNode("/ECAM/warnings/logic/bleed-1-lo-temp-unsucc"), bleed1LoTempXbleed: props.globals.initNode("/ECAM/warnings/logic/bleed-1-lo-temp-xbleed"), bleed1LoTempBleed: props.globals.initNode("/ECAM/warnings/logic/bleed-1-lo-temp-bleed"), bleed1LoTempPack: props.globals.initNode("/ECAM/warnings/logic/bleed-1-lo-temp-pack"), bleed1WaiAvail: props.globals.initNode("/ECAM/warnings/logic/bleed-1-wai-avail"), bleed2LoTempUnsuc: props.globals.initNode("/ECAM/warnings/logic/bleed-2-lo-temp-unsucc"), bleed2LoTempXbleed: props.globals.initNode("/ECAM/warnings/logic/bleed-2-lo-temp-xbleed"), bleed2LoTempBleed: props.globals.initNode("/ECAM/warnings/logic/bleed-2-lo-temp-bleed"), bleed2LoTempPack: props.globals.initNode("/ECAM/warnings/logic/bleed-2-lo-temp-pack"), bleed2WaiAvail: props.globals.initNode("/ECAM/warnings/logic/bleed-2-wai-avail"), }, Timers: { apuFaultOutput: props.globals.initNode("/ECAM/warnings/timer/apu-fault-output"), bleed1Fault: props.globals.initNode("/ECAM/warnings/timer/bleed-1-fault"), bleed1FaultOutput: props.globals.initNode("/ECAM/warnings/timer/bleed-1-fault-output"), bleed2Fault: props.globals.initNode("/ECAM/warnings/timer/bleed-2-fault"), bleed2FaultOutput: props.globals.initNode("/ECAM/warnings/timer/bleed-2-fault-output"), bleed1NotShutOutput: props.globals.initNode("/ECAM/warnings/timer/prv-1-not-shut-output"), bleed2NotShutOutput: props.globals.initNode("/ECAM/warnings/timer/prv-2-not-shut-output"), bleed1And2LoTemp: props.globals.initNode("/ECAM/warnings/timer/bleed-1-and-2-low-temp"), bleed1And2LoTempOutput: props.globals.initNode("/ECAM/warnings/timer/bleed-1-and-2-low-temp-output"), bleed1Off60Output: props.globals.initNode("/ECAM/warnings/logic/bleed-1-off-60-output"), bleed1Off5Output: props.globals.initNode("/ECAM/warnings/logic/bleed-1-off-5-output"), bleed2Off60Output: props.globals.initNode("/ECAM/warnings/logic/bleed-2-off-60-output"), bleed2Off5Output: props.globals.initNode("/ECAM/warnings/logic/bleed-2-off-5-output"), eng1AiceNotClsd: props.globals.initNode("/ECAM/warnings/timer/eng-aice-1-open-output"), eng2AiceNotClsd: props.globals.initNode("/ECAM/warnings/timer/eng-aice-2-open-output"), eng1AiceNotOpen: props.globals.initNode("/ECAM/warnings/timer/eng-aice-1-closed-output"), eng2AiceNotOpen: props.globals.initNode("/ECAM/warnings/timer/eng-aice-2-closed-output"), }, Flipflops: { bleed1LowTemp: props.globals.initNode("/ECAM/warnings/logic/bleed-1-low-temp-flipflop-output"), bleed2LowTemp: props.globals.initNode("/ECAM/warnings/logic/bleed-2-low-temp-flipflop-output"), }, }; var warning = { new: func(msg,colour = "g",aural = 9,light = 9,isMainMsg = 0,lastSubmsg = 0, sdPage = "nil", isMemo = 0) { var t = {parents:[warning]}; t.msg = msg; t.colour = colour; t.aural = aural; t.light = light; t.isMainMsg = isMainMsg; t.lastSubmsg = lastSubmsg; t.active = 0; t.noRepeat = 0; t.noRepeat2 = 0; t.clearFlag = 0; t.sdPage = sdPage; t.isMemo = isMemo; t.hasCalled = 0; t.wasActive = 0; return t }, write: func() { if (me.active == 0) { return; } me.wasActive = 1; lineIndex = 0; while (lineIndex <= 7 and lines[lineIndex].getValue() != "") { lineIndex = lineIndex + 1; # go to next line until empty line } if (lineIndex == 8) { leftOverflow.setBoolValue(1); } elsif (leftOverflow.getBoolValue()) { leftOverflow.setBoolValue(0); } if (lineIndex <= 7) { if (lines[lineIndex].getValue() == "" and me.msg != "") { # at empty line. Also checks if message is not blank to allow for some warnings with no displayed msg, eg stall lines[lineIndex].setValue(me.msg); linesCol[lineIndex].setValue(me.colour); } } }, warnlight: func() { if (me.light > 1) { return; } if (me.active == 0 and me.wasActive == 1) { lights[me.light].setBoolValue(0); me.wasActive = 0; } if (me.noRepeat == 1 or me.active == 0) { return; } lights[me.light].setBoolValue(1); me.noRepeat = 1; }, sound: func() { if (me.aural > 2) { return; } if (me.active == 0 and me.wasActive == 1) { aural[me.aural].setBoolValue(0); me.wasActive = 0; } if (me.noRepeat2 == 1 or me.active == 0) { return; } if (me.aural != 0) { aural[me.aural].setBoolValue(0); settimer(func() { aural[me.aural].setBoolValue(1); }, 0.15); } else { aural[me.aural].setBoolValue(1); } me.noRepeat2 = 1; }, callPage: func() { if (me.sdPage == "nil" or me.hasCalled == 1) { return; } libraries.SystemDisplay.failCall(me.sdPage); me.hasCalled = 1; } }; var memo = { new: func(msg,colour = "g") { var t = {parents:[memo]}; t.msg = msg; t.colour = colour; t.active = 0; return t }, write: func() { if (me.active == 1) { rightLineIndex = 0; while (rightLines[rightLineIndex].getValue() != "" and rightLineIndex <= 7) { rightLineIndex = rightLineIndex + 1; # go to next line until empty line } if (rightLineIndex > 7) { rightOverflow.setBoolValue(1); } elsif (rightOverflow.getBoolValue()) { rightOverflow.setBoolValue(0); } if (rightLines[rightLineIndex].getValue() == "" and rightLineIndex <= 7) { # at empty line rightLines[rightLineIndex].setValue(me.msg); rightLinesCol[rightLineIndex].setValue(me.colour); } } }, }; var status = { new: func(msg,colour) { var t = {parents:[status]}; t.msg = msg; t.colour = colour; t.active = 0; return t }, write: func() { if (me.active == 1) { statusIndex = 0; while (statusLines[statusIndex].getValue() != "" and statusIndex <= 7) { statusIndex = statusIndex + 1; # go to next line until empty line } if (statusLines[statusIndex].getValue() == "" and statusIndex <= 7) { # at empty line statusLines[rightLineIndex].setValue(me.msg); statusLinesCol[rightLineIndex].setValue(me.colour); } } }, }; var ECAM_controller = { _recallCounter: 0, _noneActive: 0, init: func() { ECAMloopTimer.start(); me.reset(); }, loop: func() { if ((systems.ELEC.Bus.acEss.getValue() >= 110 or systems.ELEC.Bus.ac2.getValue() >= 110) and !getprop("systems/acconfig/acconfig-running")) { # update FWC phases phaseLoop(); # check active messages messages_priority_3(); messages_priority_2(); messages_priority_1(); messages_priority_0(); messages_config_memo(); messages_memo(); messages_right_memo(); } else { foreach (var w; warnings.vector) { w.active = 0; } shutUpYou(); } # clear display momentarily for(var n = 0; n <= 7; n += 1) { lines[n].setValue(""); } for(var n = 0; n <= 7; n += 1) { rightLines[n].setValue(""); } # write to ECAM var counter = 0; if (!getprop("systems/acconfig/autoconfig-running")) { foreach (var w; warnings.vector) { if (w.active == 1) { if (counter < 9) { w.write(); counter += 1; } w.warnlight(); w.sound(); } elsif (w.wasActive == 1) { w.warnlight(); w.sound(); } } } if (lines[0].getValue() == "" and flash == 0) { # disable left memos if a warning exists. Warnings are processed first, so this stops leftmemos if line1 is not empty foreach (var c; configmemos.vector) { c.write(); } } if (lines[0].getValue() == "" and flash == 0) { # disable left memos if a warning exists. Warnings are processed first, so this stops leftmemos if line1 is not empty foreach (var l; leftmemos.vector) { l.write(); } } foreach (var sL; specialLines.vector) { sL.write(); } foreach (var sF; secondaryFailures.vector) { sF.write(); } foreach (var m; memos.vector) { m.write(); } if (leftOverflow.getBoolValue() == 1 or leftOverflow.getBoolValue() == 1) { overflow.setBoolValue(1); } elsif (leftOverflow.getBoolValue() == 0 and leftOverflow.getBoolValue() == 0) { overflow.setBoolValue(0); } }, reset: func() { foreach (var w; warnings.vector) { if (w.active == 1) { w.active = 0; } } foreach (var l; configmemos.vector) { if (l.active == 1) { l.active = 0; } } foreach (var l; leftmemos.vector) { if (l.active == 1) { l.active = 0; } } foreach (var sL; specialLines.vector) { if (sL.active == 1) { sL.active = 0; } } foreach (var sF; secondaryFailures.vector) { if (sF.active == 1) { sF.active = 0; } } foreach (var m; memos.vector) { if (m.active == 1) { m.active = 0; } } }, clear: func() { hasCleared = 0; counter = 0; noMainMsg = 0; storeFirstWarning = nil; # first go through the first eight, see how many mainMsg there are foreach (var w; warnings.vector) { if (counter >= 8) { break; } if (w.active == 1 and w.clearFlag != 1 and w.isMemo != 1) { counter += 1; if (w.isMainMsg == 1) { if (noMainMsg == 0) { storeFirstWarning = w; } noMainMsg += 1; } } } # then, if there is an overflow and noMainMsg == 1, we clear the first shown ones if (leftOverflow.getBoolValue() and noMainMsg == 1) { counter = 0; foreach (var w; warnings.vector) { if (counter >= 8) { break; } if (w.active == 1 and w.clearFlag != 1 and w.isMemo != 1) { counter += 1; if (w.isMainMsg == 1) { continue; } w.clearFlag = 1; hasCleared = 1; statusFlag = 1; } } } # else, we clear the first mainMsg stored else { if (storeFirstWarning != nil) { if (storeFirstWarning.active == 1 and storeFirstWarning.clearFlag != 1 and storeFirstWarning.isMainMsg == 1 and storeFirstWarning.isMemo != 1) { storeFirstWarning.clearFlag = 1; hasCleared = 1; statusFlag = 1; } } } if (statusFlag == 1 and lines[0].getValue() == "") { libraries.SystemDisplay.manCall("sts"); statusFlag = 0; } }, recall: func() { me._noneActive = 1; me._recallCounter = 0; foreach (var w; warnings.vector) { if (w.clearFlag == 1) { w.noRepeat = 0; w.clearFlag = 0; me._recallCounter += 1; } if (w.active == 1) { me._noneActive = 0; } } if (me._recallCounter == 0 and me._noneActive) { FWC.Btn.recallStsNormal.setValue(1); settimer(func() { if (FWC.Btn.recallStsNormal.getValue() == 1) { # catch unexpected error, trying something new here FWC.Btn.recallStsNormal.setValue(0); } else { die("Exception in ECAM-controller.nas, line 316"); } }, 0.1); } }, warningReset: func(warning) { if (warning.aural != 9 and warning.active == 1) { aural[warning.aural].setBoolValue(0); } warning.active = 0; warning.noRepeat = 0; warning.noRepeat2 = 0; # don't set .wasActive to 0, warnlight / sound funcs do that }, }; setlistener("/systems/electrical/bus/dc-ess", func { if (dc_ess.getValue() < 25) { ECAM_controller.reset(); } }, 0, 0); var ECAMloopTimer = maketimer(0.15, func { ECAM_controller.loop(); }); # Flash Master Warning Light var shutUpYou = func() { lights[0].setBoolValue(0); } var warnTimer = maketimer(0.25, func { if (!lights[0].getBoolValue()) { warnTimer.stop(); warningFlash.setBoolValue(0); } else if (!warningFlash.getBoolValue()) { warningFlash.setBoolValue(1); } else { warningFlash.setBoolValue(0); } }); setlistener("/ECAM/warnings/master-warning-light", func { if (lights[0].getBoolValue()) { warningFlash.setBoolValue(0); warnTimer.start(); } else { warnTimer.stop(); warningFlash.setBoolValue(0); } }, 0, 0);