2019-10-14 16:48:35 +00:00
# A3XX FMGC Autopilot
# Based off IT-AUTOFLIGHT System Controller V4.0.X
2021-04-23 17:21:07 +00:00
# Copyright (c) 2021 Josh Davidson (Octal450)
2019-10-14 16:48:35 +00:00
# Initialize all used variables and property nodes
# Sim
2020-07-08 20:14:55 +00:00
var Controls = {
aileron: props.globals.getNode("/controls/flight/aileron", 1),
2021-04-28 23:09:15 +00:00
aileron2: props.globals.getNode("/controls/flight/aileron[1]", 1),
2020-07-08 20:14:55 +00:00
elevator: props.globals.getNode("/controls/flight/elevator", 1),
2021-04-28 23:09:15 +00:00
elevator2: props.globals.getNode("/controls/flight/elevator[1]", 1),
2020-07-08 20:14:55 +00:00
rudder: props.globals.getNode("/controls/flight/rudder", 1),
2021-04-28 23:09:15 +00:00
rudder2: props.globals.getNode("/controls/flight/rudder[1]", 1),
2019-10-14 16:48:35 +00:00
};
var FPLN = {
2019-12-29 21:37:52 +00:00
active: props.globals.getNode("/FMGC/flightplan[2]/active", 1),
2019-10-14 16:48:35 +00:00
activeTemp: 0,
currentCourse: 0,
2020-07-09 01:12:02 +00:00
currentWp: props.globals.getNode("/FMGC/flightplan[2]/current-wp", 1),
currentWpTemp: 0,
2019-10-14 16:48:35 +00:00
deltaAngle: 0,
deltaAngleRad: 0,
distCoeff: 0,
maxBank: 0,
maxBankLimit: 0,
nextCourse: 0,
R: 0,
radius: 0,
turnDist: 0,
2019-12-29 21:37:52 +00:00
wp0Dist: props.globals.getNode("/FMGC/flightplan[2]/current-leg-dist", 1),
2019-10-14 16:48:35 +00:00
wpFlyFrom: 0,
wpFlyTo: 0,
};
2020-07-08 20:14:55 +00:00
var Gear = {
wow0: props.globals.getNode("/gear/gear[0]/wow", 1),
wow1: props.globals.getNode("/gear/gear[1]/wow", 1),
wow1Temp: 1,
wow2: props.globals.getNode("/gear/gear[2]/wow", 1),
wow2Temp: 1,
};
2019-10-14 16:48:35 +00:00
var Misc = {
2020-07-08 20:14:55 +00:00
elapsedSec: props.globals.getNode("/sim/time/elapsed-sec", 1),
fbwLaw: props.globals.getNode("/it-fbw/law", 1),
flapNorm: props.globals.getNode("/surface-positions/flap-pos-norm", 1),
pfdHeadingScale: props.globals.getNode("/instrumentation/pfd/heading-scale", 1),
};
var Position = {
gearAglFtTemp: 0,
gearAglFt: props.globals.getNode("/position/gear-agl-ft", 1),
indicatedAltitudeFt: props.globals.getNode("/instrumentation/altimeter/indicated-altitude-ft", 1),
indicatedAltitudeFtTemp: 0,
};
var Radio = {
gsDefl: [props.globals.getNode("/instrumentation/nav[0]/gs-needle-deflection-norm", 1), props.globals.getNode("instrumentation/nav[1]/gs-needle-deflection-norm", 1)],
gsDeflTemp: 0,
inRange: [props.globals.getNode("/instrumentation/nav[0]/in-range", 1), props.globals.getNode("instrumentation/nav[1]/in-range", 1)],
inRangeTemp: 0,
locDefl: [props.globals.getNode("/instrumentation/nav[0]/heading-needle-deflection-norm", 1), props.globals.getNode("instrumentation/nav[1]/heading-needle-deflection-norm", 1)],
locDeflTemp: 0,
radioSel: 0,
signalQuality: [props.globals.getNode("/instrumentation/nav[0]/signal-quality-norm", 1), props.globals.getNode("instrumentation/nav[1]/signal-quality-norm", 1)],
signalQualityTemp: 0,
};
var Velocities = {
airspeedKt: props.globals.getNode("/velocities/airspeed-kt", 1), # Only used for gain scheduling
groundspeedKt: props.globals.getNode("/velocities/groundspeed-kt", 1),
groundspeedMps: 0,
indicatedAirspeedKt: props.globals.getNode("/instrumentation/airspeed-indicator/indicated-speed-kt", 1),
indicatedMach: props.globals.getNode("/instrumentation/airspeed-indicator/indicated-mach", 1),
trueAirspeedKt: props.globals.getNode("/instrumentation/airspeed-indicator/true-speed-kt", 1),
trueAirspeedKtTemp: 0,
2019-10-14 16:48:35 +00:00
};
# IT-AUTOFLIGHT
2020-12-22 18:41:07 +00:00
var Fd = {
pitchBar: props.globals.initNode("/it-autoflight/fd/pitch-bar", 0, "DOUBLE"),
rollBar: props.globals.initNode("/it-autoflight/fd/roll-bar", 0, "DOUBLE"),
};
2019-10-14 16:48:35 +00:00
var Input = {
alt: props.globals.initNode("/it-autoflight/input/alt", 10000, "INT"),
ap1: props.globals.initNode("/it-autoflight/input/ap1", 0, "BOOL"),
ap2: props.globals.initNode("/it-autoflight/input/ap2", 0, "BOOL"),
athr: props.globals.initNode("/it-autoflight/input/athr", 0, "BOOL"),
altDiff: 0,
2020-12-22 18:41:07 +00:00
bankLimitSw: props.globals.initNode("/it-autoflight/input/bank-limit-sw", 0, "INT"),
bankLimitSwTemp: 0,
2019-10-14 16:48:35 +00:00
fd1: props.globals.initNode("/it-autoflight/input/fd1", 1, "BOOL"),
fd2: props.globals.initNode("/it-autoflight/input/fd2", 1, "BOOL"),
fpa: props.globals.initNode("/it-autoflight/input/fpa", 0, "DOUBLE"),
2020-07-08 20:14:55 +00:00
fpaAbs: props.globals.initNode("/it-autoflight/input/fpa-abs", 0, "DOUBLE"), # Set by property rule
2019-10-14 16:48:35 +00:00
hdg: props.globals.initNode("/it-autoflight/input/hdg", 0, "INT"),
2020-04-18 19:48:47 +00:00
hdgCalc: 0,
2020-07-09 01:12:02 +00:00
kts: props.globals.initNode("/it-autoflight/input/kts", 100, "INT"),
2019-10-14 16:48:35 +00:00
ktsMach: props.globals.initNode("/it-autoflight/input/kts-mach", 0, "BOOL"),
lat: props.globals.initNode("/it-autoflight/input/lat", 5, "INT"),
latTemp: 5,
2020-07-09 01:12:02 +00:00
mach: props.globals.initNode("/it-autoflight/input/mach", 0.5, "DOUBLE"),
2019-10-14 16:48:35 +00:00
toga: props.globals.initNode("/it-autoflight/input/toga", 0, "BOOL"),
trk: props.globals.initNode("/it-autoflight/input/trk", 0, "BOOL"),
trueCourse: props.globals.initNode("/it-autoflight/input/true-course", 0, "BOOL"),
vs: props.globals.initNode("/it-autoflight/input/vs", 0, "INT"),
2020-07-08 20:14:55 +00:00
vsAbs: props.globals.initNode("/it-autoflight/input/vs-abs", 0, "INT"), # Set by property rule
2019-10-14 16:48:35 +00:00
vert: props.globals.initNode("/it-autoflight/input/vert", 7, "INT"),
vertTemp: 7,
};
var Internal = {
alt: props.globals.initNode("/it-autoflight/internal/alt", 10000, "INT"),
altCaptureActive: 0,
altDiff: 0,
altTemp: 0,
altPredicted: props.globals.initNode("/it-autoflight/internal/altitude-predicted", 0, "DOUBLE"),
bankLimit: props.globals.initNode("/it-autoflight/internal/bank-limit", 30, "INT"),
bankLimitAuto: 30,
2020-07-09 01:12:02 +00:00
captVs: 0,
2020-04-18 19:48:47 +00:00
driftAngle: props.globals.initNode("/it-autoflight/internal/drift-angle-deg", 0, "DOUBLE"),
2020-12-22 18:41:07 +00:00
driftAngleTemp: 0,
2019-10-14 16:48:35 +00:00
flchActive: 0,
fpa: props.globals.initNode("/it-autoflight/internal/fpa", 0, "DOUBLE"),
hdgErrorDeg: props.globals.initNode("/it-autoflight/internal/heading-error-deg", 0, "DOUBLE"),
hdgPredicted: props.globals.initNode("/it-autoflight/internal/heading-predicted", 0, "DOUBLE"),
lnavAdvanceNm: props.globals.initNode("/it-autoflight/internal/lnav-advance-nm", 0, "DOUBLE"),
2020-07-09 01:12:02 +00:00
minVs: props.globals.initNode("/it-autoflight/internal/min-vs", -500, "INT"),
maxVs: props.globals.initNode("/it-autoflight/internal/max-vs", 500, "INT"),
2019-10-14 16:48:35 +00:00
vs: props.globals.initNode("/it-autoflight/internal/vert-speed-fpm", 0, "DOUBLE"),
vsTemp: 0,
};
var Output = {
ap1: props.globals.initNode("/it-autoflight/output/ap1", 0, "BOOL"),
ap1Temp: 0,
ap2: props.globals.initNode("/it-autoflight/output/ap2", 0, "BOOL"),
ap2Temp: 0,
apprArm: props.globals.initNode("/it-autoflight/output/appr-armed", 0, "BOOL"),
athr: props.globals.initNode("/it-autoflight/output/athr", 0, "BOOL"),
athrTemp: 0,
fd1: props.globals.initNode("/it-autoflight/output/fd1", 1, "BOOL"),
fd1Temp: 0,
fd2: props.globals.initNode("/it-autoflight/output/fd2", 1, "BOOL"),
fd2Temp: 0,
lat: props.globals.initNode("/it-autoflight/output/lat", 5, "INT"),
latTemp: 5,
lnavArm: props.globals.initNode("/it-autoflight/output/lnav-armed", 0, "BOOL"),
locArm: props.globals.initNode("/it-autoflight/output/loc-armed", 0, "BOOL"),
thrMode: props.globals.initNode("/it-autoflight/output/thr-mode", 2, "INT"),
vert: props.globals.initNode("/it-autoflight/output/vert", 7, "INT"),
vertTemp: 7,
};
var Text = {
lat: props.globals.initNode("/it-autoflight/mode/lat", "T/O", "STRING"),
thr: props.globals.initNode("/it-autoflight/mode/thr", "PITCH", "STRING"),
vert: props.globals.initNode("/it-autoflight/mode/vert", "T/O CLB", "STRING"),
2020-04-19 04:01:55 +00:00
vertTemp: "T/O CLB",
2019-10-14 16:48:35 +00:00
};
2020-12-22 18:41:07 +00:00
var Settings = {
2020-07-08 20:14:55 +00:00
reducAglFt: props.globals.initNode("/it-autoflight/settings/accel-agl-ft", 1500, "INT"), # Changable from MCDU, eventually set to 1500 above runway
2019-10-14 16:48:35 +00:00
};
var Sound = {
2021-04-22 22:03:06 +00:00
apOff: props.globals.initNode("/it-autoflight/sound/apoffsound", 0, "BOOL"), # Is this still needed??? -JD
2019-10-14 16:48:35 +00:00
enableApOff: 0,
};
# A3XX Custom
var Custom = {
apFdOn: 0,
2020-07-08 20:14:55 +00:00
hdgTime: props.globals.getNode("/modes/fcu/hdg-time", 1),
ndTrkSel: [props.globals.getNode("/instrumentation/efis[0]/trk-selected", 1), props.globals.getNode("/instrumentation/efis[1]/trk-selected", 1)],
2019-10-14 16:48:35 +00:00
showHdg: props.globals.initNode("/it-autoflight/custom/show-hdg", 1, "BOOL"),
trkFpa: props.globals.initNode("/it-autoflight/custom/trk-fpa", 0, "BOOL"),
Input: {
2020-07-08 20:14:55 +00:00
spdManaged: props.globals.getNode("/it-autoflight/input/spd-managed", 1),
2019-10-14 16:48:35 +00:00
},
Output: {
fmaPower: props.globals.initNode("/it-autoflight/output/fma-pwr", 0, "BOOL"),
2021-06-16 11:39:59 +00:00
vsFCU: props.globals.initNode("/it-autoflight/output/vs-fcu-display", "", "STRING"),
2019-10-14 16:48:35 +00:00
},
Sound: {
athrOff: props.globals.initNode("/it-autoflight/sound/athrsound", 0, "BOOL"),
enableAthrOff: 0,
},
2021-04-22 21:44:17 +00:00
ThrLock: props.globals.getNode("/fdm/jsbsim/fadec/thr-locked", 1)
2019-10-14 16:48:35 +00:00
};
var ITAF = {
init: func() {
Custom.ndTrkSel[0].setBoolValue(0);
Custom.ndTrkSel[1].setBoolValue(0);
Custom.trkFpa.setBoolValue(0);
Input.ktsMach.setBoolValue(0);
Input.ap1.setBoolValue(0);
Input.ap2.setBoolValue(0);
Input.athr.setBoolValue(0);
Input.fd1.setBoolValue(1);
Input.fd2.setBoolValue(1);
Input.hdg.setValue(360);
Input.alt.setValue(10000);
Input.vs.setValue(0);
2021-06-16 11:39:59 +00:00
Custom.Output.vsFCU.setValue(left(sprintf("%+05.0f",0),3));
2019-10-14 16:48:35 +00:00
Input.fpa.setValue(0);
Input.lat.setValue(9);
Input.vert.setValue(9);
Input.trk.setBoolValue(0);
Input.trueCourse.setBoolValue(0);
Input.toga.setBoolValue(0);
Custom.Input.spdManaged.setBoolValue(0);
Output.ap1.setBoolValue(0);
Output.ap2.setBoolValue(0);
Output.athr.setBoolValue(0);
Output.fd1.setBoolValue(1);
Output.fd2.setBoolValue(1);
2020-12-22 18:41:07 +00:00
me.updateLnavArm(0);
me.updateLocArm(0);
me.updateApprArm(0);
2019-10-14 16:48:35 +00:00
Output.thrMode.setValue(0);
Output.lat.setValue(9);
Output.vert.setValue(9);
2020-07-09 01:12:02 +00:00
Internal.minVs.setValue(-500);
Internal.maxVs.setValue(500);
2019-10-14 16:48:35 +00:00
Internal.bankLimit.setValue(30);
Internal.bankLimitAuto = 30;
Internal.alt.setValue(10000);
Internal.altCaptureActive = 0;
2020-07-09 01:12:02 +00:00
Input.kts.setValue(100);
2019-10-14 16:48:35 +00:00
Input.mach.setValue(0.5);
Text.thr.setValue("THRUST");
2020-12-22 18:41:07 +00:00
updateFma.arm();
me.updateLatText("");
me.updateVertText("");
2019-10-14 16:48:35 +00:00
Custom.showHdg.setBoolValue(1);
Custom.Output.fmaPower.setBoolValue(1);
ManagedSPD.stop();
loopTimer.start();
slowLoopTimer.start();
},
loop: func() {
Output.latTemp = Output.lat.getValue();
Output.vertTemp = Output.vert.getValue();
# VOR/ILS Revision
if (Output.latTemp == 2 or Output.vertTemp == 2 or Output.vertTemp == 6) {
me.checkRadioRevision(Output.latTemp, Output.vertTemp);
}
Gear.wow1Temp = Gear.wow1.getBoolValue();
Gear.wow2Temp = Gear.wow2.getBoolValue();
Output.ap1Temp = Output.ap1.getBoolValue();
Output.ap2Temp = Output.ap2.getBoolValue();
Output.latTemp = Output.lat.getValue();
Output.vertTemp = Output.vert.getValue();
Text.vertTemp = Text.vert.getValue();
Position.gearAglFtTemp = Position.gearAglFt.getValue();
Internal.vsTemp = Internal.vs.getValue();
Position.indicatedAltitudeFtTemp = Position.indicatedAltitudeFt.getValue();
2021-06-16 13:59:37 +00:00
# Update VLS / VMAX for autothrust
FMGCNodes.vmax.setValue(FMGCInternal.maxspeed);
FMGCNodes.vlsMin.setValue(FMGCInternal.vls_min);
2019-10-14 16:48:35 +00:00
# LNAV Engagement
if (Output.lnavArm.getBoolValue()) {
2020-12-22 18:41:07 +00:00
me.checkLnav(1);
2019-10-14 16:48:35 +00:00
}
# VOR/LOC or ILS/LOC Capture
if (Output.locArm.getBoolValue()) {
2020-12-22 18:41:07 +00:00
me.checkLoc(1);
2019-10-14 16:48:35 +00:00
}
# G/S Capture
if (Output.apprArm.getBoolValue()) {
2020-12-22 18:41:07 +00:00
me.checkAppr(1);
2019-10-14 16:48:35 +00:00
}
# Autoland Logic
if (Output.latTemp == 2) {
2020-07-19 22:58:21 +00:00
if (Position.gearAglFtTemp <= 50) { # ALIGN
2019-10-14 16:48:35 +00:00
me.setLatMode(4);
}
}
if (Output.vertTemp == 2) {
2020-07-19 22:58:21 +00:00
if (Position.gearAglFtTemp <= 400 and Position.gearAglFtTemp >= 5) {
2020-12-22 18:41:07 +00:00
me.updateVertText("LAND");
2020-07-19 22:58:21 +00:00
2021-07-26 00:12:22 +00:00
if (Position.gearAglFtTemp <= 50) {
2020-07-19 22:58:21 +00:00
me.setVertMode(6);
}
2019-10-14 16:48:35 +00:00
}
} else if (Output.vertTemp == 6) {
2020-12-22 18:41:07 +00:00
if (Gear.wow1Temp and Gear.wow2Temp and Text.vert.getValue() != "ROLLOUT") {
me.updateLatText("RLOU");
me.updateVertText("ROLLOUT");
2019-10-14 16:48:35 +00:00
}
}
# FLCH Engagement
if (Text.vertTemp == "T/O CLB") {
2020-12-22 18:41:07 +00:00
me.checkFlch(Settings.reducAglFt.getValue());
2019-10-14 16:48:35 +00:00
}
# Altitude Capture/Sync Logic
if (Output.vertTemp != 0) {
Internal.alt.setValue(Input.alt.getValue());
}
Internal.altTemp = Internal.alt.getValue();
Internal.altDiff = Internal.altTemp - Position.indicatedAltitudeFtTemp;
2020-04-21 00:50:22 +00:00
if (Output.vertTemp != 0 and Output.vertTemp != 2 and Output.vertTemp != 6 and Output.vertTemp != 9) {
2020-07-09 01:12:02 +00:00
Internal.captVs = math.clamp(math.round(abs(Internal.vs.getValue()) / 5, 100), 50, 2500); # Capture limits
2019-10-14 16:48:35 +00:00
Custom.apFdOn = Output.ap1Temp or Output.ap2Temp or Output.fd1.getBoolValue() or Output.fd2.getBoolValue();
2020-07-09 01:12:02 +00:00
if (abs(Internal.altDiff) <= Internal.captVs and !Gear.wow1Temp and !Gear.wow2Temp and Custom.apFdOn) {
2019-10-14 16:48:35 +00:00
if (Internal.altTemp >= Position.indicatedAltitudeFtTemp and Internal.vsTemp >= -25) { # Don't capture if we are going the wrong way
me.setVertMode(3);
} else if (Internal.altTemp < Position.indicatedAltitudeFtTemp and Internal.vsTemp <= 25) { # Don't capture if we are going the wrong way
me.setVertMode(3);
}
}
}
# Altitude Hold Min/Max Reset
if (Internal.altCaptureActive) {
2020-12-22 18:41:07 +00:00
if (abs(Internal.altDiff) <= 20 and Text.vert.getValue() != "ALT HLD") {
2019-10-14 16:48:35 +00:00
me.resetClimbRateLim();
2020-12-22 18:41:07 +00:00
me.updateVertText("ALT HLD");
2019-10-14 16:48:35 +00:00
}
}
# Thrust Mode Selector
2020-12-22 18:41:07 +00:00
me.updateThrustMode();
2019-10-14 16:48:35 +00:00
# Custom Stuff Below
# Heading Sync
if (!Custom.showHdg.getBoolValue()) {
Input.hdg.setValue(Misc.pfdHeadingScale.getValue());
}
# Preselect Heading
if (Output.latTemp != 0 and Output.latTemp != 9) { # Modes that always show HDG
if (Custom.hdgTime.getValue() + 45 >= Misc.elapsedSec.getValue()) {
2020-12-22 18:41:07 +00:00
Custom.showHdg.setBoolValue(1);
2019-10-14 16:48:35 +00:00
} else {
2020-12-22 18:41:07 +00:00
Custom.showHdg.setBoolValue(0);
2019-10-14 16:48:35 +00:00
}
}
# Misc
if (Output.ap1Temp == 1 or Output.ap2Temp == 1) { # Trip AP off
2021-04-28 23:09:15 +00:00
if (abs(Controls.aileron.getValue()) >= 0.2 or abs(Controls.elevator.getValue()) >= 0.2 or abs(Controls.rudder.getValue()) >= 0.2 or abs(Controls.aileron2.getValue()) >= 0.2 or abs(Controls.elevator2.getValue()) >= 0.2 or abs(Controls.rudder2.getValue()) >= 0.2) {
2019-10-27 20:54:00 +00:00
fcu.apOff("hard", 0);
2019-10-14 16:48:35 +00:00
}
}
2021-05-16 22:31:08 +00:00
if ((systems.FADEC.n1Mode[0].getValue() > 0 or systems.FADEC.n1Mode[1].getValue() > 0) and Output.athr.getBoolValue()) {
fcu.athrOff("hard");
}
2019-10-14 16:48:35 +00:00
},
slowLoop: func() {
2020-04-18 05:12:04 +00:00
Velocities.trueAirspeedKtTemp = Velocities.trueAirspeedKt.getValue();
2019-10-14 16:48:35 +00:00
FPLN.activeTemp = FPLN.active.getValue();
2020-07-09 01:12:02 +00:00
FPLN.currentWpTemp = FPLN.currentWp.getValue();
2019-10-14 16:48:35 +00:00
# Bank Limit
2020-04-18 05:12:04 +00:00
if (Velocities.trueAirspeedKtTemp >= 420) {
2019-10-14 16:48:35 +00:00
Internal.bankLimitAuto = 15;
2020-04-18 05:12:04 +00:00
} else if (Velocities.trueAirspeedKtTemp >= 340) {
2019-10-14 16:48:35 +00:00
Internal.bankLimitAuto = 20;
} else {
2020-07-19 22:58:21 +00:00
Internal.bankLimitAuto = 30;
2019-10-14 16:48:35 +00:00
}
Internal.bankLimit.setValue(Internal.bankLimitAuto);
# If in LNAV mode and route is not longer active, switch to HDG HLD
if (Output.lat.getValue() == 1) { # Only evaulate the rest of the condition if we are in LNAV mode
2020-02-02 20:34:27 +00:00
if (flightPlanController.num[2].getValue() == 0 or !FPLN.active.getBoolValue()) {
2019-10-14 16:48:35 +00:00
me.setLatMode(3);
}
}
# Waypoint Advance Logic
2020-02-02 20:34:27 +00:00
if (flightPlanController.num[2].getValue() > 0 and FPLN.activeTemp == 1) {
2020-07-09 01:12:02 +00:00
if ((FPLN.currentWpTemp + 1) < flightPlanController.num[2].getValue()) {
2020-04-18 05:12:04 +00:00
Velocities.groundspeedMps = Velocities.groundspeedKt.getValue() * 0.5144444444444;
2020-07-09 01:12:02 +00:00
FPLN.wpFlyFrom = FPLN.currentWpTemp;
2019-10-14 16:48:35 +00:00
if (FPLN.wpFlyFrom < 0) {
FPLN.wpFlyFrom = 0;
}
2020-05-15 22:53:33 +00:00
FPLN.currentCourse = fmgc.wpCourse[2][FPLN.wpFlyFrom].getValue();
2020-07-09 01:12:02 +00:00
FPLN.wpFlyTo = FPLN.currentWpTemp + 1;
2020-05-15 22:53:33 +00:00
FPLN.nextCourse = fmgc.wpCourse[2][FPLN.wpFlyTo].getValue();
2019-10-14 16:48:35 +00:00
FPLN.maxBankLimit = Internal.bankLimit.getValue();
FPLN.deltaAngle = math.abs(geo.normdeg180(FPLN.currentCourse - FPLN.nextCourse));
FPLN.maxBank = FPLN.deltaAngle * 1.5;
if (FPLN.maxBank > FPLN.maxBankLimit) {
FPLN.maxBank = FPLN.maxBankLimit;
}
2020-04-18 05:12:04 +00:00
FPLN.radius = (Velocities.groundspeedMps * Velocities.groundspeedMps) / (9.81 * math.tan(FPLN.maxBank / 57.2957795131));
2019-10-14 16:48:35 +00:00
FPLN.deltaAngleRad = (180 - FPLN.deltaAngle) / 114.5915590262;
FPLN.R = FPLN.radius / math.sin(FPLN.deltaAngleRad);
FPLN.distCoeff = FPLN.deltaAngle * -0.011111 + 2;
if (FPLN.distCoeff < 1) {
FPLN.distCoeff = 1;
}
FPLN.turnDist = math.cos(FPLN.deltaAngleRad) * FPLN.R * FPLN.distCoeff / 1852;
if (Gear.wow0.getBoolValue() and FPLN.turnDist < 1) {
FPLN.turnDist = 1;
}
Internal.lnavAdvanceNm.setValue(FPLN.turnDist);
2020-04-23 17:21:59 +00:00
# Advance logic done by flightplan controller
2020-05-15 22:53:33 +00:00
if (FPLN.wp0Dist.getValue() <= FPLN.turnDist and !Gear.wow1.getBoolValue()) {
2020-04-23 17:21:59 +00:00
flightPlanController.autoSequencing();
2019-10-14 16:48:35 +00:00
}
2020-06-14 17:30:58 +00:00
2020-07-09 01:12:02 +00:00
#if (FPLN.wp0Dist.getValue() <= FPLN.turnDist and !Gear.wow1.getBoolValue() and fmgc.flightPlanController.flightplans[2].getWP(FPLN.currentWpTemp).fly_type == "flyBy") {
2020-06-14 17:30:58 +00:00
# flightPlanController.autoSequencing();
#} elsif (FPLN.wp0Dist.getValue() <= 0.1) {
# flightPlanController.autoSequencing();
#}
2019-10-14 16:48:35 +00:00
}
}
},
ap1Master: func(s) {
if (s == 1) {
2020-11-28 17:33:44 +00:00
if (Output.vert.getValue() != 6 and !Gear.wow1.getBoolValue() and !Gear.wow2.getBoolValue() and systems.ELEC.Bus.acEss.getValue() >= 110 and fbw.FBW.apOff == 0 and Position.gearAglFt.getValue() >= 100) {
2019-10-14 16:48:35 +00:00
me.revertBasicMode();
Output.ap1.setBoolValue(1);
Output.latTemp = Output.lat.getValue();
if (Output.ap2.getBoolValue() and !Output.apprArm.getBoolValue() and Output.latTemp != 2 and Output.latTemp != 4) {
me.ap2Master(0);
}
Sound.enableApOff = 1;
Sound.apOff.setBoolValue(0);
}
} else {
Output.ap1.setBoolValue(0);
me.apOffFunction();
}
Output.ap1Temp = Output.ap1.getBoolValue();
if (Input.ap1.getBoolValue() != Output.ap1Temp) {
Input.ap1.setBoolValue(Output.ap1Temp);
}
},
ap2Master: func(s) {
if (s == 1) {
2020-11-28 17:33:44 +00:00
if (Output.vert.getValue() != 6 and !Gear.wow1.getBoolValue() and !Gear.wow2.getBoolValue() and systems.ELEC.Bus.acEss.getValue() >= 110 and fbw.FBW.apOff == 0 and Position.gearAglFt.getValue() >= 100) {
2019-10-14 16:48:35 +00:00
me.revertBasicMode();
Output.ap2.setBoolValue(1);
Output.latTemp = Output.lat.getValue();
if (Output.ap1.getBoolValue() and !Output.apprArm.getBoolValue() and Output.latTemp != 2 and Output.latTemp != 4) {
me.ap1Master(0);
}
Sound.enableApOff = 1;
Sound.apOff.setBoolValue(0);
}
} else {
Output.ap2.setBoolValue(0);
me.apOffFunction();
}
Output.ap2Temp = Output.ap2.getBoolValue();
if (Input.ap2.getBoolValue() != Output.ap2Temp) {
Input.ap2.setBoolValue(Output.ap2Temp);
}
},
apOffFunction: func() {
if (!Output.ap1.getBoolValue() and !Output.ap2.getBoolValue()) { # Only do if both APs are off
me.updateFma();
if (Sound.enableApOff) {
Sound.apOff.setBoolValue(1);
Sound.enableApOff = 0;
}
}
},
athrMaster: func(s) {
if (s == 1) {
2021-05-16 22:31:08 +00:00
if (systems.ELEC.Bus.acEss.getValue() >= 110 and !pts.FMGC.CasCompare.casRejectAll.getBoolValue() and fbw.FBW.apOff == 0 and systems.FADEC.n1Mode[0].getValue() == 0 and systems.FADEC.n1Mode[1].getValue() == 0) {
2019-10-14 16:48:35 +00:00
Output.athr.setBoolValue(1);
Custom.ThrLock.setValue(0);
Custom.Sound.enableAthrOff = 1;
Custom.Sound.athrOff.setBoolValue(0);
}
} else {
Output.athr.setBoolValue(0);
if (Custom.Sound.enableAthrOff) {
Custom.Sound.athrOff.setBoolValue(1);
Custom.Sound.enableAthrOff = 0;
}
}
Output.athrTemp = Output.athr.getBoolValue();
if (Input.athr.getBoolValue() != Output.athrTemp) {
Input.athr.setBoolValue(Output.athrTemp);
}
},
fd1Master: func(s) {
if (s == 1) {
Output.fd1.setBoolValue(1);
me.updateFma();
} else {
Output.fd1.setBoolValue(0);
if (!Output.fd2.getBoolValue()) {
me.updateFma();
}
}
Output.fd1Temp = Output.fd1.getBoolValue();
if (Input.fd1.getBoolValue() != Output.fd1Temp) {
Input.fd1.setBoolValue(Output.fd1Temp);
}
},
fd2Master: func(s) {
if (s == 1) {
Output.fd2.setBoolValue(1);
me.updateFma();
} else {
Output.fd2.setBoolValue(0);
if (!Output.fd1.getBoolValue()) {
me.updateFma();
}
}
Output.fd2Temp = Output.fd2.getBoolValue();
if (Input.fd2.getBoolValue() != Output.fd2Temp) {
Input.fd2.setBoolValue(Output.fd2Temp);
}
},
setLatMode: func(n) {
Output.vertTemp = Output.vert.getValue();
if (n == 0) { # HDG SEL
2020-12-22 18:41:07 +00:00
me.updateLnavArm(0);
me.updateLocArm(0);
me.updateApprArm(0);
2019-10-14 16:48:35 +00:00
Output.lat.setValue(0);
Custom.showHdg.setBoolValue(1);
2020-12-22 18:41:07 +00:00
me.updateLatText("HDG");
2019-10-14 16:48:35 +00:00
if (Output.vertTemp == 2 or Output.vertTemp == 6) { # Also cancel G/S or FLARE if active
me.setVertMode(1);
}
} else if (n == 1) { # LNAV
2020-12-22 18:41:07 +00:00
me.updateLocArm(0);
me.updateApprArm(0);
me.checkLnav(0);
2019-10-14 16:48:35 +00:00
} else if (n == 2) { # VOR/LOC
2020-12-22 18:41:07 +00:00
me.updateLnavArm(0);
me.checkLoc(0);
2019-10-14 16:48:35 +00:00
} else if (n == 3) { # HDG HLD
2020-12-22 18:41:07 +00:00
me.updateLnavArm(0);
me.updateLocArm(0);
me.updateApprArm(0);
2020-07-09 01:12:02 +00:00
me.syncHdg();
2019-10-14 16:48:35 +00:00
Output.lat.setValue(0);
Custom.showHdg.setBoolValue(1);
2020-12-22 18:41:07 +00:00
me.updateLatText("HDG");
2019-10-14 16:48:35 +00:00
if (Output.vertTemp == 2 or Output.vertTemp == 6) { # Also cancel G/S or FLARE if active
me.setVertMode(1);
}
} else if (n == 4) { # ALIGN
2020-12-22 18:41:07 +00:00
me.updateLnavArm(0);
me.updateLocArm(0);
me.updateApprArm(0);
2019-10-14 16:48:35 +00:00
Output.lat.setValue(4);
Custom.showHdg.setBoolValue(0);
2020-12-22 18:41:07 +00:00
me.updateLatText("ALGN");
2019-10-14 16:48:35 +00:00
} else if (n == 5) { # RWY
2020-12-22 18:41:07 +00:00
me.updateLnavArm(0);
me.updateLocArm(0);
me.updateApprArm(0);
2019-10-14 16:48:35 +00:00
Output.lat.setValue(5);
Custom.showHdg.setBoolValue(0);
2020-12-22 18:41:07 +00:00
me.updateLatText("T/O");
2019-10-14 16:48:35 +00:00
} else if (n == 9) { # NONE
2020-12-22 18:41:07 +00:00
me.updateLocArm(0);
2019-10-14 16:48:35 +00:00
Output.lat.setValue(9);
Custom.showHdg.setBoolValue(1);
2020-12-22 18:41:07 +00:00
me.updateLatText("");
2019-10-14 16:48:35 +00:00
}
},
setLatArm: func(n) {
if (n == 0) {
2020-12-22 18:41:07 +00:00
me.updateLnavArm(0);
2019-10-14 16:48:35 +00:00
Custom.showHdg.setBoolValue(1);
} else if (n == 1) {
2020-02-02 20:34:27 +00:00
if (flightPlanController.num[2].getValue() > 0 and FPLN.active.getBoolValue()) {
2020-12-22 18:41:07 +00:00
me.updateLnavArm(1);
2019-10-14 16:48:35 +00:00
Custom.showHdg.setBoolValue(0);
}
} else if (n == 3) {
2020-07-09 01:12:02 +00:00
me.syncHdg();
2020-12-22 18:41:07 +00:00
me.updateLnavArm(0);
2019-10-14 16:48:35 +00:00
Custom.showHdg.setBoolValue(1);
}
},
setVertMode: func(n) {
Input.altDiff = Input.alt.getValue() - Position.indicatedAltitudeFt.getValue();
if (n == 0) { # ALT HLD
Internal.flchActive = 0;
Internal.altCaptureActive = 0;
2020-12-22 18:41:07 +00:00
me.updateApprArm(0);
2019-10-14 16:48:35 +00:00
Output.vert.setValue(0);
me.resetClimbRateLim();
2020-12-22 18:41:07 +00:00
me.updateVertText("ALT HLD");
2020-07-09 01:12:02 +00:00
me.syncAlt();
2020-12-22 18:41:07 +00:00
me.updateThrustMode();
2019-10-14 16:48:35 +00:00
} else if (n == 1) { # V/S
if (abs(Input.altDiff) >= 25) {
Internal.flchActive = 0;
Internal.altCaptureActive = 0;
2020-12-22 18:41:07 +00:00
me.updateApprArm(0);
2019-10-14 16:48:35 +00:00
Output.vert.setValue(1);
2020-12-22 18:41:07 +00:00
me.updateVertText("V/S");
2020-07-09 01:12:02 +00:00
me.syncVs();
2020-12-22 18:41:07 +00:00
me.updateThrustMode();
2019-10-14 16:48:35 +00:00
} else {
2020-12-22 18:41:07 +00:00
me.updateApprArm(0);
2019-10-14 16:48:35 +00:00
}
} else if (n == 2) { # G/S
2020-12-22 18:41:07 +00:00
me.updateLnavArm(0);
me.checkLoc(0);
me.checkAppr(0);
2019-10-14 16:48:35 +00:00
} else if (n == 3) { # ALT CAP
Internal.flchActive = 0;
Output.vert.setValue(0);
me.setClimbRateLim();
Internal.altCaptureActive = 1;
2020-12-22 18:41:07 +00:00
me.updateVertText("ALT CAP");
me.updateThrustMode();
2019-10-14 16:48:35 +00:00
} else if (n == 4) { # FLCH
2020-12-22 18:41:07 +00:00
me.updateApprArm(0);
2019-12-29 21:37:52 +00:00
Output.vert.setValue(1);
Internal.alt.setValue(Input.alt.getValue());
Internal.altDiff = Internal.alt.getValue() - Position.indicatedAltitudeFt.getValue();
if (abs(Internal.altDiff) >= 250) { # SPD CLB or SPD DES
2019-10-14 16:48:35 +00:00
Internal.altCaptureActive = 0;
Output.vert.setValue(4);
Internal.flchActive = 1;
2021-04-22 22:03:06 +00:00
Internal.alt.setValue(Input.alt.getValue());
2020-12-22 18:41:07 +00:00
me.updateThrustMode();
2019-10-14 16:48:35 +00:00
} else { # ALT CAP
Internal.flchActive = 0;
Internal.alt.setValue(Input.alt.getValue());
Internal.altCaptureActive = 1;
Output.vert.setValue(0);
2020-12-22 18:41:07 +00:00
me.updateVertText("ALT CAP");
me.updateThrustMode();
2019-10-14 16:48:35 +00:00
}
} else if (n == 5) { # FPA
if (abs(Input.altDiff) >= 25) {
Internal.flchActive = 0;
Internal.altCaptureActive = 0;
2020-12-22 18:41:07 +00:00
me.updateApprArm(0);
2019-10-14 16:48:35 +00:00
Output.vert.setValue(5);
2020-12-22 18:41:07 +00:00
me.updateVertText("FPA");
2020-07-09 01:12:02 +00:00
me.syncFpa();
2020-12-22 18:41:07 +00:00
me.updateThrustMode();
2019-10-14 16:48:35 +00:00
} else {
2020-12-22 18:41:07 +00:00
me.updateApprArm(0);
2019-10-14 16:48:35 +00:00
}
} else if (n == 6) { # FLARE/ROLLOUT
Internal.flchActive = 0;
Internal.altCaptureActive = 0;
2020-12-22 18:41:07 +00:00
me.updateApprArm(0);
2019-10-14 16:48:35 +00:00
Output.vert.setValue(6);
2021-07-26 00:12:22 +00:00
me.updateVertText("FLARE");
2021-07-26 00:14:14 +00:00
me.updateThrustMode();
2019-10-14 16:48:35 +00:00
} else if (n == 7) { # T/O CLB or G/A CLB, text is set by TOGA selector
Internal.flchActive = 0;
Internal.altCaptureActive = 0;
2020-12-22 18:41:07 +00:00
me.updateApprArm(0);
2019-10-14 16:48:35 +00:00
Output.vert.setValue(7);
2020-12-22 18:41:07 +00:00
me.updateThrustMode();
2019-10-14 16:48:35 +00:00
} else if (n == 9) { # NONE
Internal.flchActive = 0;
Internal.altCaptureActive = 0;
2020-12-22 18:41:07 +00:00
me.updateApprArm(0);
2019-10-14 16:48:35 +00:00
Output.vert.setValue(9);
2020-12-22 18:41:07 +00:00
me.updateVertText("");
me.updateThrustMode();
2019-10-14 16:48:35 +00:00
}
},
2020-12-22 18:41:07 +00:00
updateThrustMode: func() {
Output.vertTemp = Output.vert.getValue();
if (Output.athr.getBoolValue() and Output.vertTemp != 7 and (Output.ap1.getBoolValue() or Output.ap2.getBoolValue()) and Position.gearAglFt.getValue() <= 30 and (Output.vertTemp == 2 or Output.vertTemp == 6)) {
# Manual says 40 feet - but video reference shows 30!
Output.thrMode.setValue(1);
Text.thr.setValue("RETARD");
} else if (Output.vertTemp == 4) {
if (Internal.alt.getValue() >= Position.indicatedAltitudeFt.getValue()) {
Output.thrMode.setValue(2);
Text.thr.setValue("PITCH");
2021-04-22 22:03:06 +00:00
if (Internal.flchActive and Text.vert.getValue() != "SPD CLB") {
2020-12-22 18:41:07 +00:00
me.updateVertText("SPD CLB");
}
} else {
Output.thrMode.setValue(1);
Text.thr.setValue("PITCH");
2021-04-22 22:03:06 +00:00
if (Internal.flchActive and Text.vert.getValue() != "SPD DES") {
2020-12-22 18:41:07 +00:00
me.updateVertText("SPD DES");
}
}
} else if (Output.vertTemp == 7) {
Output.thrMode.setValue(2);
Text.thr.setValue("PITCH");
} else {
Output.thrMode.setValue(0);
Text.thr.setValue("THRUST");
2019-10-14 16:48:35 +00:00
}
},
2020-12-22 18:41:07 +00:00
activateLnav: func() {
2019-10-14 16:48:35 +00:00
if (Output.lat.getValue() != 1) {
2020-12-22 18:41:07 +00:00
me.updateLnavArm(0);
me.updateLocArm(0);
me.updateApprArm(0);
2019-10-14 16:48:35 +00:00
Output.lat.setValue(1);
Custom.showHdg.setBoolValue(0);
2020-12-22 18:41:07 +00:00
me.updateLatText("LNAV");
2019-10-14 16:48:35 +00:00
if (Output.vertTemp == 2 or Output.vertTemp == 6) { # Also cancel G/S or FLARE if active
me.setVertMode(1);
}
}
},
2020-12-22 18:41:07 +00:00
activateLoc: func() {
2019-10-14 16:48:35 +00:00
if (Output.lat.getValue() != 2) {
2020-12-22 18:41:07 +00:00
me.updateLnavArm(0);
me.updateLocArm(0);
2019-10-14 16:48:35 +00:00
Output.lat.setValue(2);
Custom.showHdg.setBoolValue(0);
2020-12-22 18:41:07 +00:00
me.updateLatText("LOC");
2019-10-14 16:48:35 +00:00
}
},
2020-12-22 18:41:07 +00:00
activateGs: func() {
2019-10-14 16:48:35 +00:00
if (Output.vert.getValue() != 2) {
Internal.flchActive = 0;
Internal.altCaptureActive = 0;
2020-12-22 18:41:07 +00:00
me.updateApprArm(0);
2019-10-14 16:48:35 +00:00
Output.vert.setValue(2);
2020-12-22 18:41:07 +00:00
me.updateVertText("G/S");
me.updateThrustMode();
2019-10-14 16:48:35 +00:00
}
},
2020-12-22 18:41:07 +00:00
checkLnav: func(t) {
2021-04-22 22:03:06 +00:00
FPLN.activeTemp = FPLN.active.getBoolValue();
if (flightPlanController.num[2].getValue() > 0 and FPLN.activeTemp and Position.gearAglFt.getValue() >= 30) {
2020-12-22 18:41:07 +00:00
me.activateLnav();
2021-04-22 22:03:06 +00:00
} else if (FPLN.activeTemp and Output.lat.getValue() != 1 and t != 1) {
2020-12-22 18:41:07 +00:00
me.updateLnavArm(1);
2019-10-14 16:48:35 +00:00
}
},
2020-12-22 18:41:07 +00:00
checkFlch: func(a) {
2019-10-14 16:48:35 +00:00
if (Position.indicatedAltitudeFt.getValue() >= a and a != 0 and !Gear.wow1.getBoolValue() and !Gear.wow2.getBoolValue()) {
me.setVertMode(4);
}
},
2020-12-22 18:41:07 +00:00
checkLoc: func(t) {
2019-10-14 16:48:35 +00:00
if (Radio.inRange[Radio.radioSel].getBoolValue()) { # # Only evaulate the rest of the condition unless we are in range
Radio.locDeflTemp = Radio.locDefl[Radio.radioSel].getValue();
Radio.signalQualityTemp = Radio.signalQuality[Radio.radioSel].getValue();
if (abs(Radio.locDeflTemp) <= 0.95 and Radio.locDeflTemp != 0 and Radio.signalQualityTemp >= 0.99) {
2020-12-22 18:41:07 +00:00
me.activateLoc();
2019-10-14 16:48:35 +00:00
} else if (t != 1) { # Do not do this if loop calls it
if (Output.lat.getValue() != 2) {
2020-12-22 18:41:07 +00:00
me.updateLnavArm(0);
me.updateLocArm(1);
2019-10-14 16:48:35 +00:00
}
}
2021-04-22 22:03:06 +00:00
} else {
Radio.signalQuality[Radio.radioSel].setValue(0); # Prevent bad behavior due to FG not updating it when not in range
me.updateLocArm(0);
2019-10-14 16:48:35 +00:00
}
},
2020-12-22 18:41:07 +00:00
checkAppr: func(t) {
2019-10-14 16:48:35 +00:00
if (Radio.inRange[Radio.radioSel].getBoolValue()) { # # Only evaulate the rest of the condition unless we are in range
Radio.gsDeflTemp = Radio.gsDefl[Radio.radioSel].getValue();
if (abs(Radio.gsDeflTemp) <= 0.2 and Radio.gsDeflTemp != 0 and Output.lat.getValue() == 2) { # Only capture if LOC is active
2020-12-22 18:41:07 +00:00
me.activateGs();
2019-10-14 16:48:35 +00:00
} else if (t != 1) { # Do not do this if loop calls it
if (Output.vert.getValue() != 2) {
2020-12-22 18:41:07 +00:00
me.updateApprArm(1);
2019-10-14 16:48:35 +00:00
}
}
2021-04-22 22:03:06 +00:00
} else {
Radio.signalQuality[Radio.radioSel].setValue(0); # Prevent bad behavior due to FG not updating it when not in range
me.updateApprArm(0);
2019-10-14 16:48:35 +00:00
}
},
checkRadioRevision: func(l, v) { # Revert mode if signal lost
Radio.inRangeTemp = Radio.inRange[Radio.radioSel].getBoolValue();
if (!Radio.inRangeTemp) {
if (l == 4 or v == 6) {
me.ap1Master(0);
me.ap2Master(0);
me.setLatMode(3);
me.setVertMode(1);
} else {
me.setLatMode(3); # Also cancels G/S if active
}
}
},
setClimbRateLim: func() {
Internal.vsTemp = Internal.vs.getValue();
if (Internal.alt.getValue() >= Position.indicatedAltitudeFt.getValue()) {
2020-07-09 01:12:02 +00:00
Internal.maxVs.setValue(math.round(Internal.vsTemp));
Internal.minVs.setValue(-500);
2019-10-14 16:48:35 +00:00
} else {
2020-07-09 01:12:02 +00:00
Internal.maxVs.setValue(500);
Internal.minVs.setValue(math.round(Internal.vsTemp));
2019-10-14 16:48:35 +00:00
}
},
resetClimbRateLim: func() {
2020-07-09 01:12:02 +00:00
Internal.minVs.setValue(-500);
Internal.maxVs.setValue(500);
2019-10-14 16:48:35 +00:00
},
takeoffGoAround: func() {
Output.vertTemp = Output.vert.getValue();
2020-04-18 05:12:04 +00:00
if ((Output.vertTemp == 2 or Output.vertTemp == 6) and Velocities.indicatedAirspeedKt.getValue() >= 80) {
2020-07-08 20:14:55 +00:00
me.setLatMode(3);
me.setVertMode(7); # Must be before kicking AP off
2020-12-22 18:41:07 +00:00
me.updateVertText("G/A CLB");
2020-07-09 01:12:02 +00:00
me.syncKtsGa();
2019-10-14 16:48:35 +00:00
if (Gear.wow1.getBoolValue() or Gear.wow2.getBoolValue()) {
me.ap1Master(0);
me.ap2Master(0);
}
} else if (Gear.wow1Temp or Gear.wow2Temp) {
me.athrMaster(1);
2020-04-18 05:12:04 +00:00
if (Output.lat.getValue() != 5) { # Don't accidently disarm LNAV
me.setLatMode(5);
}
2019-10-14 16:48:35 +00:00
me.setVertMode(7);
2020-12-22 18:41:07 +00:00
me.updateVertText("T/O CLB");
2019-10-14 16:48:35 +00:00
}
},
2020-07-09 01:12:02 +00:00
syncKts: func() {
2021-06-16 12:36:36 +00:00
Input.kts.setValue(math.clamp(math.round(Velocities.indicatedAirspeedKt.getValue()), 100, 399));
2019-10-14 16:48:35 +00:00
},
2020-07-09 01:12:02 +00:00
syncKtsGa: func() { # Same as syncKts, except doesn't go below V2
2021-06-16 12:36:36 +00:00
Input.kts.setValue(math.clamp(math.round(Velocities.indicatedAirspeedKt.getValue()), FMGCInternal.v2, 399));
2020-07-08 20:14:55 +00:00
},
2019-10-14 16:48:35 +00:00
syncMach: func() {
2021-06-16 12:36:36 +00:00
Input.mach.setValue(math.clamp(math.round(Velocities.indicatedMach.getValue(), 0.001), 0.1, 0.99));
2019-10-14 16:48:35 +00:00
},
2020-07-09 01:12:02 +00:00
syncHdg: func() {
2019-10-14 16:48:35 +00:00
Input.hdg.setValue(math.round(Internal.hdgPredicted.getValue())); # Switches to track automatically
},
2020-07-09 01:12:02 +00:00
syncAlt: func() {
2019-10-14 16:48:35 +00:00
Input.alt.setValue(math.clamp(math.round(Internal.altPredicted.getValue(), 100), 0, 50000));
Internal.alt.setValue(math.clamp(math.round(Internal.altPredicted.getValue(), 100), 0, 50000));
},
2021-06-16 11:39:59 +00:00
tempVS: 0,
2020-07-09 01:12:02 +00:00
syncVs: func() {
2021-06-16 11:39:59 +00:00
me.tempVS = math.clamp(math.round(Internal.vs.getValue(), 100), -6000, 6000);
Input.vs.setValue(me.tempVS);
fmgc.Custom.Output.vsFCU.setValue(left(sprintf("%+05.0f",me.tempVS),3));
2019-10-14 16:48:35 +00:00
},
2020-07-09 01:12:02 +00:00
syncFpa: func() {
2019-10-14 16:48:35 +00:00
Input.fpa.setValue(math.clamp(math.round(Internal.fpa.getValue(), 0.1), -9.9, 9.9));
},
# Custom Stuff Below
updateFma: func() {
if (!Output.ap1.getBoolValue() and !Output.ap2.getBoolValue() and !Output.fd1.getBoolValue() and !Output.fd2.getBoolValue()) {
me.setLatMode(9);
me.setVertMode(9);
me.setLatArm(0);
Custom.Output.fmaPower.setBoolValue(0);
} else {
Custom.Output.fmaPower.setBoolValue(1);
me.revertBasicMode();
}
},
revertBasicMode: func() {
if (!Gear.wow1.getBoolValue() and !Gear.wow2.getBoolValue()) { # Don't do this on ground
if (Output.lat.getValue() == 9) {
me.setLatMode(3);
}
if (Output.vert.getValue() == 9) {
if (Custom.trkFpa.getBoolValue()) {
me.setVertMode(5);
} else {
me.setVertMode(1);
}
}
}
},
2020-12-22 18:41:07 +00:00
disarmLoc: func() {
me.updateLocArm(0);
2019-10-14 16:48:35 +00:00
},
2020-12-22 18:41:07 +00:00
disarmAppr: func() {
me.updateApprArm(0);
2019-10-14 16:48:35 +00:00
},
toggleTrkFpa: func() {
if (Custom.trkFpa.getBoolValue()) {
me.trkFpaOff();
} else {
me.trkFpaOn();
}
},
trkFpaOn: func() {
Custom.trkFpa.setBoolValue(1);
if (Output.vert.getValue() == 1) {
Input.vert.setValue(5); # This way we only do this if all conditions are true
}
Input.trk.setBoolValue(1);
Custom.ndTrkSel[0].setBoolValue(1);
Custom.ndTrkSel[1].setBoolValue(1);
2020-04-18 19:48:47 +00:00
Input.hdgCalc = Input.hdg.getValue() + math.round(Internal.driftAngle.getValue());
if (Input.hdgCalc > 360) { # It's rounded, so this is ok. Otherwise do >= 360.5
Input.hdgCalc = Input.hdgCalc - 360;
} else if (Input.hdgCalc < 1) { # It's rounded, so this is ok. Otherwise do < 0.5
Input.hdgCalc = Input.hdgCalc + 360;
2019-10-14 16:48:35 +00:00
}
2020-04-18 19:48:47 +00:00
Input.hdg.setValue(Input.hdgCalc);
2019-10-14 16:48:35 +00:00
},
trkFpaOff: func() {
Custom.trkFpa.setBoolValue(0);
if (Output.vert.getValue() == 5) {
Input.vert.setValue(1); # This way we only do this if all conditions are true
}
Input.trk.setBoolValue(0);
Custom.ndTrkSel[0].setBoolValue(0);
Custom.ndTrkSel[1].setBoolValue(0);
2020-04-18 19:48:47 +00:00
Input.hdgCalc = Input.hdg.getValue() - math.round(Internal.driftAngle.getValue());
if (Input.hdgCalc > 360) { # It's rounded, so this is ok. Otherwise do >= 360.5
Input.hdgCalc = Input.hdgCalc - 360;
} else if (Input.hdgCalc < 1) { # It's rounded, so this is ok. Otherwise do < 0.5
Input.hdgCalc = Input.hdgCalc + 360;
2019-10-14 16:48:35 +00:00
}
2020-04-18 19:48:47 +00:00
Input.hdg.setValue(Input.hdgCalc);
2019-10-14 16:48:35 +00:00
},
2020-12-22 18:41:07 +00:00
updateLatText: func(t) {
Text.lat.setValue(t);
updateFma.lat();
},
updateVertText: func(t) {
Text.vert.setValue(t);
updateFma.vert();
},
updateLnavArm: func(n) {
Output.lnavArm.setBoolValue(n);
updateFma.arm();
},
updateLocArm: func(n) {
Output.locArm.setBoolValue(n);
updateFma.arm();
},
updateApprArm: func(n) {
Output.apprArm.setBoolValue(n);
updateFma.arm();
},
2019-10-14 16:48:35 +00:00
};
2020-12-22 18:41:07 +00:00
setlistener("/it-autoflight/input/ap1", func() {
2019-10-14 16:48:35 +00:00
Input.ap1Temp = Input.ap1.getBoolValue();
if (Input.ap1Temp != Output.ap1.getBoolValue()) {
ITAF.ap1Master(Input.ap1Temp);
}
2021-06-10 15:49:09 +00:00
});
2019-10-14 16:48:35 +00:00
2020-12-22 18:41:07 +00:00
setlistener("/it-autoflight/input/ap2", func() {
2019-10-14 16:48:35 +00:00
Input.ap2Temp = Input.ap2.getBoolValue();
if (Input.ap2Temp != Output.ap2.getBoolValue()) {
ITAF.ap2Master(Input.ap2Temp);
}
2021-06-10 15:49:09 +00:00
});
2019-10-14 16:48:35 +00:00
2020-12-22 18:41:07 +00:00
setlistener("/it-autoflight/input/athr", func() {
2019-10-14 16:48:35 +00:00
Input.athrTemp = Input.athr.getBoolValue();
if (Input.athrTemp != Output.athr.getBoolValue()) {
ITAF.athrMaster(Input.athrTemp);
}
2021-06-10 15:49:09 +00:00
});
2019-10-14 16:48:35 +00:00
2020-12-22 18:41:07 +00:00
setlistener("/it-autoflight/input/fd1", func() {
2019-10-14 16:48:35 +00:00
Input.fd1Temp = Input.fd1.getBoolValue();
if (Input.fd1Temp != Output.fd1.getBoolValue()) {
ITAF.fd1Master(Input.fd1Temp);
}
2021-06-10 15:49:09 +00:00
});
2019-10-14 16:48:35 +00:00
2020-12-22 18:41:07 +00:00
setlistener("/it-autoflight/input/fd2", func() {
2019-10-14 16:48:35 +00:00
Input.fd2Temp = Input.fd2.getBoolValue();
if (Input.fd2Temp != Output.fd2.getBoolValue()) {
ITAF.fd2Master(Input.fd2Temp);
}
2021-06-10 15:49:09 +00:00
});
2019-10-14 16:48:35 +00:00
2020-12-22 18:41:07 +00:00
setlistener("/it-autoflight/input/kts-mach", func() {
if (Output.vert.getValue() == 7) { # Mach is not allowed in Mode 7, and don't sync
if (Input.ktsMach.getBoolValue()) {
Input.ktsMach.setBoolValue(0);
}
2019-10-14 16:48:35 +00:00
} else {
2020-12-22 18:41:07 +00:00
if (Input.ktsMach.getBoolValue()) {
ITAF.syncMach();
} else {
ITAF.syncKts();
}
2019-10-14 16:48:35 +00:00
}
}, 0, 0);
2020-12-22 18:41:07 +00:00
setlistener("/it-autoflight/input/toga", func() {
2019-10-14 16:48:35 +00:00
if (Input.toga.getBoolValue()) {
ITAF.takeoffGoAround();
Input.toga.setBoolValue(0);
}
2021-06-10 15:49:09 +00:00
});
2019-10-14 16:48:35 +00:00
2020-12-22 18:41:07 +00:00
setlistener("/it-autoflight/input/lat", func() {
2019-10-14 16:48:35 +00:00
Input.latTemp = Input.lat.getValue();
Output.ap1Temp = Output.ap1.getBoolValue();
Output.ap2Temp = Output.ap2.getBoolValue();
Output.fd1Temp = Output.fd1.getBoolValue();
Output.fd2Temp = Output.fd2.getBoolValue();
2021-06-10 15:38:16 +00:00
2019-10-14 16:48:35 +00:00
if (!Gear.wow1.getBoolValue() and !Gear.wow2.getBoolValue()) {
if (Output.ap1Temp or Output.ap2Temp or Output.fd1Temp or Output.fd2Temp) {
ITAF.setLatMode(Input.latTemp);
} else {
ITAF.setLatMode(9);
}
} else {
if (Output.ap1Temp or Output.ap2Temp or Output.fd1Temp or Output.fd2Temp) {
ITAF.setLatArm(Input.latTemp);
} else {
ITAF.setLatArm(0);
}
}
2021-06-10 15:49:09 +00:00
});
2019-10-14 16:48:35 +00:00
2020-12-22 18:41:07 +00:00
setlistener("/it-autoflight/input/vert", func() {
2019-10-14 16:48:35 +00:00
if (!Gear.wow1.getBoolValue() and !Gear.wow2.getBoolValue() and (Output.ap1.getBoolValue() or Output.ap2.getBoolValue() or Output.fd1.getBoolValue() or Output.fd2.getBoolValue())) {
ITAF.setVertMode(Input.vert.getValue());
} else {
ITAF.setVertMode(9);
}
2021-06-10 15:49:09 +00:00
});
2019-10-14 16:48:35 +00:00
2021-06-19 00:46:57 +00:00
# Mode Reversions
setlistener(pts.Systems.Navigation.ADR.Output.overspeed, func(v) {
if (v.getBoolValue() and !Output.ap1.getBoolValue() and !Output.ap2.getBoolValue() and Output.athr.getBoolValue() and Modes.PFD.FMA.pitchMode.getValue() == "OP CLB" and Modes.PFD.FMA.throttle.getValue() == "THR CLB") {
Input.fd1.setValue(0);
Input.fd2.setValue(0);
ecam.aural[5].setBoolValue(0);
settimer(func() {
ecam.aural[5].setBoolValue(1);
}, 0.15);
}
}, 0, 0);
2021-06-19 13:25:21 +00:00
setlistener(pts.Systems.Navigation.ADR.Output.underspeed, func(v) {
if (v.getBoolValue() and !Output.ap1.getBoolValue() and !Output.ap2.getBoolValue() and Output.athr.getBoolValue() and Modes.PFD.FMA.pitchMode.getValue() == "OP DES" and Modes.PFD.FMA.throttle.getValue() == "THR IDLE") {
Input.fd1.setValue(0);
Input.fd2.setValue(0);
ecam.aural[5].setBoolValue(0);
settimer(func() {
ecam.aural[5].setBoolValue(1);
}, 0.15);
}
}, 0, 0);
2021-06-19 00:46:57 +00:00
2019-10-14 16:48:35 +00:00
setlistener("/sim/signals/fdm-initialized", func {
ITAF.init();
2021-06-10 15:49:09 +00:00
});
2019-10-14 16:48:35 +00:00
# For Canvas Nav Display.
2020-12-22 18:41:07 +00:00
setlistener("/it-autoflight/input/hdg", func() {
2021-06-10 15:38:16 +00:00
pts.Autopilot.Settings.headingBugDeg.setValue(Input.hdg.getValue());
2020-09-27 18:00:46 +00:00
}, 0, 0);
2019-10-14 16:48:35 +00:00
2020-12-22 18:41:07 +00:00
setlistener("/it-autoflight/internal/alt", func() {
2021-06-10 15:38:16 +00:00
pts.Autopilot.Settings.targetAltitudeFt.setValue(Internal.alt.getValue());
2020-09-27 18:00:46 +00:00
}, 0, 0);
2019-10-14 16:48:35 +00:00
var loopTimer = maketimer(0.1, ITAF, ITAF.loop);
var slowLoopTimer = maketimer(1, ITAF, ITAF.slowLoop);