1 contributor
var displayClass = {
new: func(node, role) {
var m = { parents: [ displayClass ] };
m.display = canvas.new({
"name" : role,
"size" : [1024, 768],
"view" : [1024, 768],
"mipmapping": 1
});
m.display.addPlacement({
"node": "Screen",
"parent": role
});
m.display.setColorBackground(0,0,0);
m.role = role;
m.screenElements = {};
return m;
},
loadsvg : func () {
me.screen = me.display.createGroup();
me.screen.hide();
canvas.parsesvg(me.screen, "Aircraft/Instruments-3d/zkv1000/Systems/screen.svg");
},
_showInitProgress : func (p,t) {
p.setText(t);
if (zkv.getNode(me.role ~ 'init').getValue() != 0) {
if (size(t) >= 10) t = '';
settimer(func { me._showInitProgress(p, t ~ '.'); }, 0.1);
}
else {
me.progress.hide();
me.screen.show();
var groups = {
show : [
'SoftKeysTexts',
'COMM',
'NAV',
],
hide : [ ],
text: [
'nav1-standby-freq', 'nav1-selected-freq', 'nav1-id',
'nav2-standby-freq', 'nav2-selected-freq', 'nav2-id',
'nav-freq-switch',
],
clip: [ ],
};
if (me.role == 'PFD') {
append(groups.show,
'XPDR-TIME',
'FlightInstruments',
'Horizon',
'bankPointer',
'VSI',
'Rose',
);
append(groups.hide,
'CDI',
'NAV1-pointer',
'NAV2-pointer',
'GPS-pointer',
'Bearing1',
'Bearing2',
);
append(groups.clip,
'SpeedLint1',
'SpeedTape',
'LintAlt',
'AltLint00011'
);
append(groups.text,
'VSIText',
'Speed110',
'Alt11100',
'HDG-text',
'AltBigC', 'AltSmallC'
);
for (var place = 1; place <= 6; place +=1) {
append(groups.text,
'AltBigU' ~ place,
'AltSmallU' ~ place,
'AltBigD' ~ place,
'AltSmallD' ~ place
);
}
}
else
append(groups.show, 'Header');
me.loadGroup(groups);
if (me.role == 'PFD') {
me.updateAI(getprop('/orientation/roll-deg'),getprop('orientation/pitch-deg'));
me.updateVSI(getprop('/instrumentation/vertical-speed-indicator/indicated-speed-fpm'));
me.updateIAS(getprop('/velocities/airspeed-kt'));
me.updateALT(getprop('instrumentation/altimeter/indicated-altitude-ft'));
me.updateHSI(getprop('orientation/heading-deg'));
}
me.updateNAV({refresh:1, auto:1});
me.progress.removeAllChildren();
me.progress = nil;
me.showInitProgress = nil;
me._showInitProgress = nil;
zkv.removeChild(me.role ~ 'init');
}
},
showInitProgress : func (role) {
me.progress = me.display.createGroup();
me.progress.show();
me.progress.createChild("text", role ~ " title")
.setTranslation(512, 384)
.setAlignment("center-center")
.setFont("LiberationFonts/LiberationSans-Italic.ttf")
.setFontSize(64, 1)
.setColor(1,1,1)
.setText("ZKV1000 " ~ role ~ " init");
zkv.getNode(role ~ 'init',1).setIntValue(1);
me._showInitProgress(me.progress.createChild("text", role ~ "progress")
.setTranslation(512, 484)
.setAlignment("center-center")
.setFont("LiberationFonts/LiberationSans-Bold.ttf")
.setFontSize(128, 1)
.setColor(1,0,0), '.');
},
loadGroup : func (h) {
if (typeof(h) != 'hash') {
msg_dbg(sprintf("%s need a hash, but get a %s from %s",
caller(0)[0],
typeof(h),
caller(1)[0]));
return;
}
var setMethod = func (e, t) {
if (t == 'hide')
me.screenElements[e].hide();
elsif (t == 'show')
me.screenElements[e].show();
elsif (t == 'rot' or t == 'trans') {
if (! contains(me.screenElements[e], t))
me.screenElements[e][t] = me.screenElements[e].createTransform();
}
elsif (t == 'clip') {
if (contains(me.clips, e))
me.screenElements[e].set("clip", me.clips[e]);
else
print('no defined clip for ' ~ e);
}
elsif (t == 'text') {
if (contains(me.texts, e)) {
if (contains(me.texts[e], 'alignment'))
me.screenElements[e].setAlignment(me.texts[e].alignment);
if (contains(me.texts[e], 'default'))
me.screenElements[e].setText(me.texts[e].default);
if (contains(me.texts[e], 'color'))
me.screenElements[e].setColor(me.texts[e].color);
}
# else
# print('no text format for ' ~ e);
}
else
print('unknown method ' ~ t);
};
foreach (var todo; keys(h)) {
if (typeof(h[todo]) != 'vector') h[todo] = [ h[todo] ];
foreach (var id; h[todo]) {
if (! contains(me.screenElements, id)) {
me.screenElements[id] = me.screen.getElementById(id);
if (me.screenElements[id] != nil)
setMethod(id, todo);
else
print('SVG ID ' ~ id ~ ' not found');
}
else
setMethod(id, todo);
}
}
},
clips : {
PitchScale : "rect(70,664,370,256)",
SpeedLint1 : "rect(252,226,318,204)",
SpeedTape : "rect(115,239,455,156)",
LintAlt : "rect(115,808,455,704)",
AltLint00011 : "rect(252,804,318,771)",
},
texts : {
VSIText : {
alignment: "right-bottom", default : num('0'),
},
Speed110 : {
alignment : 'left-bottom'
},
Alt11100 : {
alignment:'left-bottom'
},
"HDG-text" : {
default: '---°'
},
'nav1-standby-freq' : {
color: [0, 1, 1],
},
'nav1-id' : {
default: ''
},
'nav2-id' : {
default: ''
},
},
updateAI: func(roll,pitch){
if (pitch > 80)
pitch = 80;
elsif (pitch < -80)
pitch = -80;
me.screenElements.Horizon
.setRotation(-roll * D2R)
.setTranslation(0, pitch * 6.8571428);
me.screenElements.bankPointer
.setRotation(-roll * D2R);
settimer(func me.updateAI(getprop('/orientation/roll-deg'),getprop('orientation/pitch-deg')), 0.1);
},
updateVSI: func (vsi) {
me.screenElements.VSIText
.setText(num(math.round(vsi, 10)));
if (vsi > 4500)
vsi = 4500;
elsif (vsi < -4500)
vsi = -4500;
me.screenElements.VSI
.setTranslation(0, vsi * -0.03465);
settimer(func me.updateVSI(getprop('/instrumentation/vertical-speed-indicator/indicated-speed-fpm')), 0.1);
},
updateIAS: func (ias) {
if (ias >= 10)
me.screenElements.Speed110
.setText(sprintf("% 2u",num(math.floor(ias/10))));
else
me.screenElements.Speed110
.setText('');
me.screenElements.SpeedLint1
.setTranslation(0,(math.mod(ias,10) + (ias >= 10)*10) * 36);
me.screenElements.SpeedTape
.setTranslation(0,ias * 5.711);
settimer(func me.updateIAS(getprop('/velocities/airspeed-kt')), 0.1);
},
updateALT: func (alt) {
if (alt < 0)
me.screenElements.Alt11100
.setText(sprintf("% 3i",math.ceil(alt/100)));
elsif (alt < 100)
me.screenElements.Alt11100
.setText('');
else
me.screenElements.Alt11100
.setText(sprintf("% 3i",math.floor(alt/100)));
me.screenElements.AltLint00011
.setTranslation(0,math.fmod(alt,100) * 1.24);
# From Farmin/G1000 http://wiki.flightgear.org/Project_Farmin/FG1000
if (alt> -1000 and alt< 1000000) {
var Offset10 = 0;
var Offset100 = 0;
var Offset1000 = 0;
if (alt< 0) {
var Ne = 1;
var alt= -alt;
}
else
var Ne = 0;
var Alt10 = math.mod(alt,100);
var Alt100 = int(math.mod(alt/100,10));
var Alt1000 = int(math.mod(alt/1000,10));
var Alt10000 = int(math.mod(alt/10000,10));
var Alt20 = math.mod(Alt10,20)/20;
if (Alt10 >= 80)
var Alt100 += Alt20;
if (Alt10 >= 80 and Alt100 >= 9)
var Alt1000 += Alt20;
if (Alt10 >= 80 and Alt100 >= 9 and Alt1000 >= 9)
var Alt10000 += Alt20;
if (alt> 100)
var Offset10 = 100;
if (alt> 1000)
var Offset100 = 10;
if (alt> 10000)
var Offset1000 = 10;
if (!Ne) {
me.screenElements.LintAlt.setTranslation(0,(math.mod(alt,100))*0.57375);
var altCentral = (int(alt/100)*100);
}
elsif (Ne) {
me.screenElements.LintAlt.setTranslation(0,(math.mod(alt,100))*-0.57375);
var altCentral = -(int(alt/100)*100);
}
me.screenElements["AltBigC"].setText("");
me.screenElements["AltSmallC"].setText("");
for (var place = 1; place <= 6; place += 1) {
var altUP = altCentral + (place*100);
var offset = -30.078;
if (altUP < 0) {
var altUP = -altUP;
var prefix = "-";
var offset += 15.039;
}
else
var prefix = "";
if (altUP == 0) {
var AltBigUP = "";
var AltSmallUP = "0";
}
elsif (math.mod(altUP,500) == 0 and altUP != 0) {
var AltBigUP = sprintf(prefix~"%1d", altUP);
var AltSmallUP = "";
}
elsif (altUP < 1000 and (math.mod(altUP,500))) {
var AltBigUP = "";
var AltSmallUP = sprintf(prefix~"%1d", int(math.mod(altUP,1000)));
var offset = -30.078;
}
elsif ((altUP < 10000) and (altUP >= 1000) and (math.mod(altUP,500))) {
var AltBigUP = sprintf(prefix~"%1d", int(altUP/1000));
var AltSmallUP = sprintf("%1d", int(math.mod(altUP,1000)));
var offset += 15.039;
}
else {
var AltBigUP = sprintf(prefix~"%1d", int(altUP/1000));
var mod = int(math.mod(altUP,1000));
var AltSmallUP = sprintf("%1d", mod);
var offset += 30.078;
}
me.screenElements["AltBigU"~place].setText(AltBigUP);
me.screenElements["AltSmallU"~place].setText(AltSmallUP);
me.screenElements["AltSmallU"~place].setTranslation(offset,0);
var altDOWN = altCentral - (place*100);
var offset = -30.078;
if (altDOWN < 0) {
var altDOWN = -altDOWN;
var prefix = "-";
var offset += 15.039;
}
else
var prefix = "";
if (altDOWN == 0) {
var AltBigDOWN = "";
var AltSmallDOWN = "0";
}
elsif (math.mod(altDOWN,500) == 0 and altDOWN != 0) {
var AltBigDOWN = sprintf(prefix~"%1d", altDOWN);
var AltSmallDOWN = "";
}
elsif (altDOWN < 1000 and (math.mod(altDOWN,500))) {
var AltBigDOWN = "";
var AltSmallDOWN = sprintf(prefix~"%1d", int(math.mod(altDOWN,1000)));
var offset = -30.078;
}
elsif ((altDOWN < 10000) and (altDOWN >= 1000) and (math.mod(altDOWN,500))) {
var AltBigDOWN = sprintf(prefix~"%1d", int(altDOWN/1000));
var AltSmallDOWN = sprintf("%1d", int(math.mod(altDOWN,1000)));
var offset += 15.039;
}
else {
var AltBigDOWN = sprintf(prefix~"%1d", int(altDOWN/1000));
var mod = int(math.mod(altDOWN,1000));
var AltSmallDOWN = sprintf("%1d", mod);
var offset += 30.078;
}
me.screenElements["AltBigD"~place].setText(AltBigDOWN);
me.screenElements["AltSmallD"~place].setText(AltSmallDOWN);
me.screenElements["AltSmallD"~place].setTranslation(offset,0);
}
}
settimer(func me.updateALT(getprop('instrumentation/altimeter/indicated-altitude-ft')), 0.2);
},
updateHSI : func (hdg) {
me.screenElements.Rose
.setRotation(-hdg * D2R);
me.screenElements['HDG-text']
.setText(sprintf("%03u°", hdg));
settimer(func me.updateHSI(getprop('orientation/heading-deg')), 0.1);
},
updateNAV : func {
# made active via menu
if (contains(arg[0], "active")) {
if (arg[0]['active'] == 'none') {
me.screenElements['nav1-id']
.setColor(1,1,1);
me.screenElements['nav1-selected-freq']
.setColor(1,1,1);
me.screenElements['nav2-id']
.setColor(1,1,1);
me.screenElements['nav2-selected-freq']
.setColor(1,1,1);
me.screenElements['NAV1-pointer']
.hide();
me.screenElements['NAV2-pointer']
.hide();
}
else {
var inactive = (arg[0]['active'] == 1) + 1;
me.screenElements['nav' ~ arg[0]['active'] ~ '-id']
.setColor(0,1,0);
me.screenElements['nav' ~ arg[0]['active'] ~ '-selected-freq']
.setColor(0,1,0);
me.screenElements['NAV' ~ arg[0]['active'] ~ '-pointer']
.show();
# me.screenElements['HDI']
# .setRotation();
me.screenElements['nav' ~ inactive ~ '-id']
.setColor(1,1,1);
me.screenElements['nav' ~ inactive ~ '-selected-freq']
.setColor(1,1,1);
# me.screenElements['NAV' ~ inactive ~ '-pointer']
# .hide();
# foreach (var e; [ 'FROM', 'TO', 'CDI' ])
# me.screenElements['NAV' ~ inactive ~ '-' ~ e]
# .hide();
}
}
if (contains(arg[0], 'tune')) {
var n = getprop('/instrumentation/zkv1000/radios/nav-tune');
# n = 0 -> NAV1
# n = 1 -> NAV2
me.screenElements['nav-freq-switch']
.setTranslation(0, n * 25);
me.screenElements['nav' ~ (n + 1) ~ '-standby-freq']
.setColor(0,1,1);
me.screenElements['nav' ~ ((n == 0) + 1) ~ '-standby-freq']
.setColor(1,1,1);
}
if (contains(arg[0], 'nav-id')) {
# TODO: récupérer la valeur via les paramètres transmis du listener
var navid = getprop('/instrumentation/nav[' ~ (arg[0]['nav-id'] - 1) ~ ']/nav-id');
if (navid == nil)
navid = '';
me.screenElements["nav" ~ arg[0]['nav-id'] ~ "-id"]
.setText(navid); # veut pas exister au début...
}
if (contains(arg[0], 'refresh')) {
# rafraichi une seule ligne NAV1 ou NAV2
me.screenElements['nav' ~ arg[0].refresh ~ '-selected-freq']
.setText(getprop('/instrumentation/nav[' ~ (arg[0].refresh - 1) ~ ']/frequencies/selected-mhz-fmt'));
me.screenElements['nav' ~ arg[0].refresh ~ '-standby-freq']
.setText(getprop('/instrumentation/nav[' ~ (arg[0].refresh - 1) ~ ']/frequencies/standby-mhz-fmt'));
}
if (contains(arg[0], 'set')) {
# positionne la valeur modifiée, les listeners "trigguent" en permanence ces propriétés, donc exit
var n = getprop('/instrumentation/zkv1000/radios/nav-tune');
me.screenElements['nav' ~ (n + 1) ~ '-standby-freq']
.setText(getprop('/instrumentation/nav[' ~ n ~ ']/frequencies/standby-mhz-fmt'));
}
if (contains(arg[0], 'auto')) {
# pour rafraichir automagiquement, toutes les deux secondes un refresh pour un NAV
me.updateNAV({refresh: 1});
settimer(func me.updateNAV({refresh: 2}), 1);
settimer(func me.updateNAV({auto:1}), 2);
}
},
};