Newer
Older
import { CalculatorResults } from "./calculator-results";
import { CalculatedParamResults } from "./param-calc-results";
import { NgParameter } from "../formulaire/ngparam";
import { ResultElement, ParamFamily, capitalize } from "jalhyd";
import { ServiceFactory } from "../services/service-factory";
import { ChartType } from "./chart-type";
export class VarResults extends CalculatedParamResults implements PlottableData {
private _variatedParams: NgParameter[];
* titre des colonnes des résultats variés
private _variableParamHeaders: string[];
mathias.chouet
committed
* clés des résultats
mathias.chouet
committed
public resultKeys: string[];
mathias.chouet
committed
* entête des colonnes des résultats
mathias.chouet
committed
private _resultHeaders: string[];
mathias.chouet
committed
* type de graphe
protected _graphType: ChartType = ChartType.Scatter;
mathias.chouet
committed
* variated parameter or result displayed as chart's X-axis
mathias.chouet
committed
public chartX: string;
/**
* variated parameter or result displayed as chart's Y-axis
*/
public chartY: string;
/** size of the longest variated parameter */
public size: number;
mathias.chouet
committed
/** index of the longest variated parameter */
public longest: number;
/**
* tableau des ordonnées du graphe des résultats variés
*/
private _yValues: number[] = [];
constructor() {
super();
this.reset();
}
public reset() {
super.reset();
this._variableParamHeaders = [];
mathias.chouet
committed
this._resultHeaders = [];
this.resultKeys = [];
this._yValues = [];
mathias.chouet
committed
this.longest = 0;
public get variatedParameters(): NgParameter[] {
return this._variatedParams;
public set variatedParameters(p: NgParameter[]) {
this._variatedParams = p;
public get variableParamHeaders() {
return this._variableParamHeaders;
}
public get yValues() {
return this._yValues;
}
public get resultElements(): ResultElement[] {
mathias.chouet
committed
public get resultHeaders() {
return this._resultHeaders;
public get graphType(): ChartType {
public set graphType(gt: ChartType) {
this._graphType = gt;
this.resetDefaultAxisIfNeeded();
}
public getChartAxisLabel(symbol: string): string {
// 1. calculated param ?
if (this.calculatedParameter && this.calculatedParameter.symbol === symbol) {
return this.calculatedParameterHeader;
for (let i = 0; i < this.variatedParameters.length; i++) {
if (this._variatedParams[i].symbol === symbol) {
return this.variableParamHeaders[i];
mathias.chouet
committed
// 3. Result element / child result
return this.expandLabelFromSymbol(symbol);
}
/**
* Returns the translated name of the given symbol (usually a result or child result) with
* its unit, but without the symbol itself
*/
public expandLabelFromSymbol(symbol: string): string {
mathias.chouet
committed
let ret = "";
// calculator type for translation
const sn = this.result.sourceNub;
let ct = sn.calcType;
if (sn.parent) {
ct = sn.parent.calcType;
}
mathias.chouet
committed
// detect children results
const match = /^([0-9]+)_(.+)$/.exec(symbol);
if (match !== null) {
const pos = +match[1];
ct = sn.getChildren()[pos].calcType;
symbol = match[2];
const cn = capitalize(ServiceFactory.instance.i18nService.childName(sn));
ret += sprintf(ServiceFactory.instance.i18nService.localizeText("INFO_STUFF_N"), cn) + (pos + 1) + " : ";
mathias.chouet
committed
}
ret += ServiceFactory.instance.formulaireService.expandVariableNameAndUnit(ct, symbol);
return ret;
mathias.chouet
committed
/**
* Returns the series of values for the required variated parameter / result element
mathias.chouet
committed
* @param symbol parameter / result symbol (ex: "Q", "0_Q"...)
mathias.chouet
committed
*/
public getValuesSeries(symbol: string): number[] {
let found = false;
const series: number[] = [];
// detect children results
const isChildResult = /^([0-9]+)_(.+)$/.exec(symbol);
mathias.chouet
committed
// 1. variated param ?
for (let i = 0; i < this.variatedParameters.length; i++) {
const vp = this._variatedParams[i];
let isTheGoodChild = false;
// are we looking for a child variated param ?
if (isChildResult !== null) {
const children = this.result.sourceNub.getChildren();
const parameterNub = vp.paramDefinition.parentNub;
if (children.includes(parameterNub)) { // current var param is a child param !
const pos = parameterNub.findPositionInParent();
isTheGoodChild = (pos === +isChildResult[1] && vp.symbol === isChildResult[2]);
}
}
// in any case
if (isTheGoodChild || vp.symbol === symbol) {
for (const v of iter) {
mathias.chouet
committed
}
mathias.chouet
committed
// 2. Result element ?
if (! found) {
for (const r of this.result.resultElements) { // re:ResultElement
for (const k in r.values) {
if (k === symbol) {
found = true;
series.push(r.getValue(k));
}
mathias.chouet
committed
}
}
}
mathias.chouet
committed
// 3. Child result element ?
const child = sn.getChildren()[pos];
for (const r of child.result.resultElements) {
series.push(r.getValue(symbol));
}
mathias.chouet
committed
}
}
mathias.chouet
committed
return series;
}
/**
* Returns a list of plottable parameters / result elements, that can be defined
mathias.chouet
committed
*/
let res: string[] = [];
res = res.concat(this.getVariatingParametersSymbols());
mathias.chouet
committed
for (const erk of this.resultKeys) {
if (erk.indexOf("ENUM_") === -1) { // ENUM variables are not plottable
res.push(erk);
}
}
// children results
const sn = this.result.sourceNub;
for (const c of sn.getChildren()) {
if (c.result) {
// using latest ResultElement; results count / types are supposed to be the same on every iteration
for (const k of c.result.resultElement.keys) {
if (k.indexOf("ENUM_") === -1) { // ENUM variables are not plottable
res.push(c.findPositionInParent() + "_" + k);
}
mathias.chouet
committed
}
}
mathias.chouet
committed
}
return res;
}
* Same as X axis, plus results families if chart type is Scatter
* (for multi-series comparison)
*/
public getAvailableYAxis(): string[] {
const res: string[] = this.getAvailableXAxis();
if (this._graphType === ChartType.Scatter) {
// add families having more than 1 variable as plottable ordinates
const families = this.extractFamilies();
for (const f in families) {
if (families[f].length > 1) {
res.push(f);
}
}
}
return res;
}
/**
* Browses all parameters and results to produce a map of families => list of
* symbols in this family
*/
// exclude pseudo-family "ANY"
const fam = v.paramDefinition.family;
if (fam !== undefined && fam !== ParamFamily.ANY) {
const f = ParamFamily[fam];
if (! (f in families)) {
families[f] = [];
}
families[f].push(v.symbol);
}
}
mathias.chouet
committed
for (const erk of this.resultKeys) {
const fam = this.result.sourceNub.getFamily(erk);
// exclude pseudo-family "ANY"
if (fam !== undefined && fam !== ParamFamily.ANY) {
const f = ParamFamily[fam];
if (! (f in families)) {
families[f] = [];
}
families[f].push(erk);
}
}
// children results
const sn = this.result.sourceNub;
for (const c of sn.getChildren()) {
if (c.result) {
for (const k of c.result.resultElement.keys) {
const fam = this.result.sourceNub.getFamily(k);
// exclude pseudo-family "ANY"
if (fam !== undefined && fam !== ParamFamily.ANY) {
const f = ParamFamily[fam];
if (! (f in families)) {
families[f] = [];
}
const pos = c.findPositionInParent();
families[f].push(pos + "_" + k);
}
}
}
}
/**
* Returns the list of variating parameters
* (used by tooltip functions)
*/
public getVariatingParametersSymbols(): string[] {
return this._variatedParams.map((vp) => {
// detect if variated param is a children param
const parameterNub = vp.paramDefinition.parentNub;
const children = sn.getChildren();
let symb = vp.symbol;
if (children.includes(parameterNub)) {
symb = parameterNub.findPositionInParent() + "_" + symb;
}
return symb;
});
}
// refresh param headers
this._variableParamHeaders = this._variatedParams.map((v) => {
let h = CalculatorResults.paramLabel(v, true, this.result.sourceNub);
h += this.getHelpLink(v.symbol);
return h;
// liste la plus longue
this.size = 0;
mathias.chouet
committed
let i = 0;
for (const v of this._variatedParams) {
const s = v.valuesIterator.count();
if (s > this.size) {
this.size = s;
mathias.chouet
committed
this.longest = i;
mathias.chouet
committed
i++;
mathias.chouet
committed
// result keys (extra or not) - some lines might miss some results, in case of an error;
// use those keys to ensure all columns are filled
if (this.resultKeys.length === 0) {
for (const re of this.result.resultElements) { // re:ResultElement
mathias.chouet
committed
for (const erk in re.values) {
if (!this.resultKeys.includes(erk)) {
this.resultKeys.push(erk);
// set axis selectors values the first time
let defaultY = this.chartY;
mathias.chouet
committed
if (this.resultKeys.length > 0) {
defaultY = this.resultKeys[0];
}
mathias.chouet
committed
this.chartX = this.chartX || this.variatedParameters[this.longest].symbol;
this.chartY = defaultY;
// calculator type for translation
const sn = this.result.sourceNub;
let ct = sn.calcType;
if (sn.parent) {
ct = sn.parent.calcType;
}
mathias.chouet
committed
// entêtes des résultats
this._resultHeaders = [];
for (const k of this.resultKeys) {
let unit;
// is k the calculated parameter ? If so, extract its unit
try {
const p = sn.getParameter(k);
if (p) {
unit = p.unit;
}
} catch (e) { /* silent fail */ }
let rh = ServiceFactory.instance.formulaireService.expandVariableNameAndUnit(ct, k, unit);
rh += this.getHelpLink(k);
this._resultHeaders.push(rh);
mathias.chouet
committed
// entêtes des résultats des enfants
for (const c of sn.getChildren()) {
const cn = capitalize(ServiceFactory.instance.i18nService.childName(sn));
// using latest ResultElement; results count / types are supposed to be the same on every iteration
for (const k of c.result.resultElement.keys) {
let rh = sprintf(ServiceFactory.instance.i18nService.localizeText("INFO_STUFF_N"), cn)
+ (c.findPositionInParent() + 1) + " : "
+ ServiceFactory.instance.formulaireService.expandVariableNameAndUnit(ct, k);
rh += this.getHelpLink(k);
this._resultHeaders.push(rh);
mathias.chouet
committed
}
}
mathias.chouet
committed
* When variable parameter or chart type changes, ensure the X / Y current values are still available
*/
public resetDefaultAxisIfNeeded() {
if (! this.getAvailableXAxis().includes(this.chartX)) {
this.chartX = this.variatedParameters[0].symbol;
mathias.chouet
committed
}
this.chartY = this.variatedParameters[0].symbol;
mathias.chouet
committed
}