@hestia-earth/ui-components 0.24.2 → 0.25.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/common/utils.d.ts +1 -0
- package/cycles/cycles-nodes/cycles-nodes.component.d.ts +1 -0
- package/esm2022/common/utils.mjs +2 -1
- package/esm2022/cycles/cycles-nodes/cycles-nodes.component.mjs +8 -6
- package/esm2022/engine/aggregation-engine.service.mjs +2 -2
- package/esm2022/engine/engine.service.mjs +3 -3
- package/esm2022/files/files-error-summary.model.mjs +1 -1
- package/esm2022/files/files-error.model.mjs +5 -3
- package/esm2022/files/files-form/files-form.component.mjs +1 -1
- package/esm2022/files/files-form.model.mjs +1 -1
- package/esm2022/impact-assessments/impact-assessments-indicator-breakdown-chart/impact-assessments-indicator-breakdown-chart.component.mjs +141 -122
- package/esm2022/impact-assessments/impact-assessments-products/impact-assessments-products.component.mjs +38 -13
- package/esm2022/node/node-logs-models/node-logs-models.component.mjs +8 -4
- package/esm2022/node/node-value-details/node-value-details.component.mjs +2 -1
- package/esm2022/schema/index.mjs +2 -1
- package/esm2022/schema/schema-validation.model.mjs +2 -0
- package/esm2022/sites/index.mjs +2 -2
- package/esm2022/sites/sites-nodes/sites-nodes.component.mjs +129 -0
- package/esm2022/sites/sites.module.mjs +4 -4
- package/fesm2022/hestia-earth-ui-components.mjs +1504 -1425
- package/fesm2022/hestia-earth-ui-components.mjs.map +1 -1
- package/files/files-error-summary.model.d.ts +2 -1
- package/files/files-error.model.d.ts +2 -20
- package/files/files-form.model.d.ts +1 -1
- package/impact-assessments/impact-assessments-indicator-breakdown-chart/impact-assessments-indicator-breakdown-chart.component.d.ts +35 -17
- package/impact-assessments/impact-assessments-products/impact-assessments-products.component.d.ts +16 -6
- package/node/node-logs-models/node-logs-models.component.d.ts +3 -0
- package/package.json +1 -1
- package/schema/index.d.ts +1 -0
- package/schema/schema-validation.model.d.ts +30 -0
- package/sites/index.d.ts +1 -1
- package/sites/{sites-measurements/sites-measurements.component.d.ts → sites-nodes/sites-nodes.component.d.ts} +24 -11
- package/sites/sites.module.d.ts +2 -2
- package/esm2022/sites/sites-measurements/sites-measurements.component.mjs +0 -101
|
@@ -1,12 +1,14 @@
|
|
|
1
|
-
import { Component, Input, ViewChild } from '@angular/core';
|
|
2
|
-
import {
|
|
3
|
-
import { filter, groupBy, map, mergeAll, mergeMap, toArray } from 'rxjs/operators';
|
|
1
|
+
import { ChangeDetectionStrategy, Component, Input, ViewChild, computed, effect, signal } from '@angular/core';
|
|
2
|
+
import { toObservable, toSignal } from '@angular/core/rxjs-interop';
|
|
3
|
+
import { filter, groupBy, map, mergeAll, mergeMap, take, tap, toArray } from 'rxjs/operators';
|
|
4
4
|
import { Chart } from 'chart.js';
|
|
5
5
|
import ChartDataLabels from 'chartjs-plugin-datalabels';
|
|
6
6
|
import { DataState } from '@hestia-earth/api';
|
|
7
7
|
import { NodeType, TermTermType } from '@hestia-earth/schema';
|
|
8
8
|
import { toPrecision, unique } from '@hestia-earth/utils';
|
|
9
|
+
import { of } from 'rxjs';
|
|
9
10
|
import { matchTermType, matchType } from '../../search/search.model';
|
|
11
|
+
import { distinctUntilChangedDeep } from '../../common/rxjs-utils';
|
|
10
12
|
import { Level, parseLines, parseMessage } from '../../common/logs-utils';
|
|
11
13
|
import { listColor } from '../../common/color';
|
|
12
14
|
import * as i0 from "@angular/core";
|
|
@@ -45,39 +47,74 @@ const logToCsv = (logs, impact) => [
|
|
|
45
47
|
.map(v => [...v, impact.product?.term?.units].join(','))
|
|
46
48
|
].join('\n');
|
|
47
49
|
export class ImpactAssessmentsIndicatorBreakdownChartComponent {
|
|
50
|
+
set impactAssessment(value) {
|
|
51
|
+
this._impactAssessment.set(value);
|
|
52
|
+
}
|
|
53
|
+
set indicators(value) {
|
|
54
|
+
this._indicators.set(value);
|
|
55
|
+
}
|
|
48
56
|
constructor(ngZone, domSanitizer, searchService, nodeService) {
|
|
49
57
|
this.ngZone = ngZone;
|
|
50
58
|
this.domSanitizer = domSanitizer;
|
|
51
59
|
this.searchService = searchService;
|
|
52
60
|
this.nodeService = nodeService;
|
|
53
|
-
this.
|
|
54
|
-
this.
|
|
55
|
-
this.
|
|
56
|
-
this.
|
|
57
|
-
this.
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
61
|
+
this._impactAssessment = signal(undefined);
|
|
62
|
+
this.impactAssessment$ = toObservable(this._impactAssessment);
|
|
63
|
+
this._indicators = signal([]);
|
|
64
|
+
this.allLogs = signal([]);
|
|
65
|
+
this.logs = computed(() => this.allLogs()
|
|
66
|
+
.filter(({ impactTermId, modelId }) => impactTermId === this.selectedTerm()?.['@id'] &&
|
|
67
|
+
(!this.selectedMethod() || modelId === this.selectedMethod()?.['@id']))
|
|
68
|
+
.sort((a, b) => b.value - a.value));
|
|
69
|
+
this.emissions = toSignal(this.searchService
|
|
70
|
+
.search$({
|
|
71
|
+
fields: ['@type', '@id', 'name'],
|
|
72
|
+
limit: 1000,
|
|
73
|
+
query: {
|
|
74
|
+
bool: {
|
|
75
|
+
must: [matchType(NodeType.Term)],
|
|
76
|
+
should: filterTermTypes.map(termType => matchTermType(termType)),
|
|
77
|
+
minimum_should_match: 1
|
|
78
|
+
}
|
|
79
|
+
}
|
|
68
80
|
})
|
|
69
|
-
.pipe(
|
|
81
|
+
.pipe(map(({ results }) => results)));
|
|
82
|
+
this.loading = signal(true);
|
|
83
|
+
this.selectedTerm = signal(undefined);
|
|
84
|
+
this.terms = computed(() => unique((this._indicators() || []).map(({ term }) => term).filter(Boolean)));
|
|
85
|
+
this.selectedMethod = signal(undefined);
|
|
86
|
+
this.methods = computed(() => unique((this._indicators() || []).map(({ methodModel }) => methodModel).filter(Boolean)));
|
|
87
|
+
this.impacts = computed(() => [
|
|
88
|
+
...(this._impactAssessment()?.impacts || []),
|
|
89
|
+
...(this._impactAssessment()?.endpoints || [])
|
|
90
|
+
]);
|
|
91
|
+
this.noData = computed(() => this.logs()?.length === 0);
|
|
92
|
+
this.csvContent = computed(() => this.domSanitizer.bypassSecurityTrustResourceUrl(`data:text/html;charset=utf-8,${encodeURIComponent(logToCsv(this.logs(), this._impactAssessment()))}`));
|
|
93
|
+
this.id = computed(() => this._impactAssessment()?.['@id'] || this._impactAssessment()?.id);
|
|
94
|
+
this.downloadFilename = computed(() => `${this.id()}-logs.csv`);
|
|
95
|
+
this.total = computed(() => this.logs().reduce((prev, curr) => prev + curr.value, 0));
|
|
96
|
+
this.labels = computed(() => this.logs().map(({ blankNodeTermId }) => this.emissions()?.find(v => v['@id'] === blankNodeTermId)?.name || blankNodeTermId));
|
|
97
|
+
this.datasets = computed(() => [
|
|
98
|
+
{
|
|
99
|
+
label: this.selectedTerm()?.name ?? '',
|
|
100
|
+
data: this.logs().map(({ value }) => value),
|
|
101
|
+
backgroundColor: this.logs().map(listColor),
|
|
102
|
+
borderColor: this.logs().map(listColor)
|
|
103
|
+
}
|
|
104
|
+
]);
|
|
105
|
+
this.impactAssessment$
|
|
106
|
+
.pipe(filter(v => !!v), distinctUntilChangedDeep(), take(1), tap(() => this.loading.set(true)), mergeMap(node => (node ? this.nodeService.getLog({ ...node, dataState: DataState.recalculated }) : of(''))), map(value => (value ? parseLines(value) : [])), mergeAll(), filter(({ data }) => data.logger === 'hestia_earth.models' && data.level === Level.debug), filter(({ data: { message } }) => !!message), map(({ data: { message } }) => parseMessage(message)), filter(message => 'node' in message), map(parseLog), filter(log => !!log.impactTermId && !!log.blankNodeTermId && !isNaN(log.value) && log.value > 0), groupBy(log => [log.impactTermId, log.blankNodeTermId, log.modelId].join('/')), mergeMap(group => group.pipe(toArray())), map((values) => {
|
|
70
107
|
const log = values[0];
|
|
71
108
|
const total = values.reduce((prev, curr) => prev + curr.value, 0);
|
|
72
109
|
const blankNodes = [
|
|
73
|
-
...(this.
|
|
74
|
-
...(this.
|
|
110
|
+
...(this._impactAssessment()?.emissionsResourceUse ?? []),
|
|
111
|
+
...(this._impactAssessment()?.impacts ?? [])
|
|
75
112
|
].filter(v => v.term['@id'] === log.blankNodeTermId);
|
|
76
113
|
const inputs = blankNodes
|
|
77
114
|
.filter(v => v.inputs?.length)
|
|
78
115
|
.map(v => ({ name: v.inputs.map(i => i['@id']).join(';'), value: v.value * log.coefficient }));
|
|
79
116
|
const inputsValue = inputs.reduce((prev, curr) => prev + curr.value, 0);
|
|
80
|
-
const impact = impacts.find(v => v.term['@id'] === log.impactTermId);
|
|
117
|
+
const impact = this.impacts().find(v => v.term['@id'] === log.impactTermId);
|
|
81
118
|
// logs might exist but impact was not added => skip logs
|
|
82
119
|
return impact
|
|
83
120
|
? {
|
|
@@ -88,122 +125,104 @@ export class ImpactAssessmentsIndicatorBreakdownChartComponent {
|
|
|
88
125
|
inputsValue
|
|
89
126
|
}
|
|
90
127
|
: null;
|
|
91
|
-
}), filter(log => !!log), toArray())
|
|
92
|
-
.
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
128
|
+
}), filter(log => !!log), toArray(), tap(() => this.loading.set(false)))
|
|
129
|
+
.subscribe((logs) => {
|
|
130
|
+
this.allLogs.set(logs);
|
|
131
|
+
this.initChart();
|
|
132
|
+
});
|
|
133
|
+
effect(() => {
|
|
134
|
+
// make sure selected term exists
|
|
135
|
+
const terms = this.terms();
|
|
136
|
+
const selectedTermId = this.selectedTerm()?.['@id'];
|
|
137
|
+
if (!selectedTermId || !terms.find(term => term['@id'] === selectedTermId)) {
|
|
138
|
+
this.selectedTerm.set(terms[0]);
|
|
139
|
+
}
|
|
140
|
+
this.initChart();
|
|
141
|
+
}, { allowSignalWrites: true });
|
|
142
|
+
effect(() => {
|
|
143
|
+
if (this.datasets().length) {
|
|
144
|
+
this.initChart();
|
|
103
145
|
}
|
|
104
146
|
});
|
|
105
|
-
this.emissions = results;
|
|
106
|
-
this.loading = false;
|
|
107
|
-
this.selectedTerm = this.terms[0];
|
|
108
|
-
this.updateChart();
|
|
109
147
|
}
|
|
110
|
-
|
|
111
|
-
|
|
148
|
+
ngAfterViewInit() {
|
|
149
|
+
this.initChart();
|
|
112
150
|
}
|
|
113
151
|
initChart() {
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
(!this.selectedMethod || modelId === this.selectedMethod['@id']))
|
|
117
|
-
.sort((a, b) => b.value - a.value);
|
|
118
|
-
this.noData = logs.length === 0;
|
|
119
|
-
const total = logs.reduce((prev, curr) => prev + curr.value, 0);
|
|
120
|
-
const labels = logs.map(({ blankNodeTermId }) => this.emissions.find(v => v['@id'] === blankNodeTermId)?.name || blankNodeTermId);
|
|
121
|
-
// display the breakdown by input
|
|
122
|
-
// const inputs = logs.flatMap(v => v.inputs.map(i => i.name));
|
|
123
|
-
// const datasets = [
|
|
124
|
-
// ...inputs.map((input, index) => ({
|
|
125
|
-
// label: input,
|
|
126
|
-
// data: logs.map(log => log.inputs.find(({ name }) => name === input)?.value || 0),
|
|
127
|
-
// backgroundColor: getColor(index),
|
|
128
|
-
// borderColor: getColor(index)
|
|
129
|
-
// })),
|
|
130
|
-
// // add dataset for rest not linked to inputs
|
|
131
|
-
// {
|
|
132
|
-
// label: this.selectedTerm?.name ?? '',
|
|
133
|
-
// data: logs.map(log => log.value - log.inputsValue),
|
|
134
|
-
// backgroundColor: getColor(-1),
|
|
135
|
-
// borderColor: getColor(-1)
|
|
136
|
-
// }
|
|
137
|
-
// ];
|
|
138
|
-
const datasets = [
|
|
139
|
-
{
|
|
140
|
-
label: this.selectedTerm?.name ?? '',
|
|
141
|
-
data: logs.map(({ value }) => value),
|
|
142
|
-
backgroundColor: logs.map(listColor),
|
|
143
|
-
borderColor: logs.map(listColor)
|
|
144
|
-
}
|
|
145
|
-
];
|
|
146
|
-
if (this.chart) {
|
|
147
|
-
this.chart.destroy();
|
|
152
|
+
if (!this.chartRef) {
|
|
153
|
+
return;
|
|
148
154
|
}
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
const ratio = toPrecision((value * 100) / total, 2);
|
|
162
|
-
return value > 0 ? `${toPrecision(value, 3)} (${ratio}%)` : '';
|
|
163
|
-
},
|
|
164
|
-
align: 'end'
|
|
165
|
-
}
|
|
166
|
-
},
|
|
167
|
-
responsive: true,
|
|
168
|
-
maintainAspectRatio: false,
|
|
169
|
-
legend: {
|
|
170
|
-
display: false
|
|
171
|
-
},
|
|
172
|
-
tooltips: {
|
|
173
|
-
enabled: false,
|
|
174
|
-
callbacks: {
|
|
175
|
-
title: (tooltipItems, data) => data.datasets[tooltipItems[0].datasetIndex]?.label || '',
|
|
176
|
-
label: tooltipItem => {
|
|
177
|
-
const value = +(tooltipItem.value || '0');
|
|
178
|
-
const ratio = toPrecision((value * 100) / total, 2);
|
|
179
|
-
return value > 0 ? `${toPrecision(value, 3)} (${ratio}%)` : '';
|
|
180
|
-
}
|
|
181
|
-
}
|
|
155
|
+
const labels = this.labels();
|
|
156
|
+
const datasets = this.datasets();
|
|
157
|
+
const total = this.total();
|
|
158
|
+
this.ngZone.runOutsideAngular(() => {
|
|
159
|
+
if (this.chart) {
|
|
160
|
+
this.chart.destroy();
|
|
161
|
+
}
|
|
162
|
+
this.chart = new Chart(this.chartRef?.nativeElement, {
|
|
163
|
+
type: 'horizontalBar',
|
|
164
|
+
data: {
|
|
165
|
+
labels,
|
|
166
|
+
datasets
|
|
182
167
|
},
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
168
|
+
plugins: [ChartDataLabels],
|
|
169
|
+
options: {
|
|
170
|
+
plugins: {
|
|
171
|
+
datalabels: {
|
|
172
|
+
color: 'black',
|
|
173
|
+
formatter: value => {
|
|
174
|
+
const ratio = toPrecision((value * 100) / total, 2);
|
|
175
|
+
return value > 0 ? `${toPrecision(value, 3)} (${ratio}%)` : '';
|
|
176
|
+
},
|
|
177
|
+
align: 'end'
|
|
187
178
|
}
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
179
|
+
},
|
|
180
|
+
responsive: true,
|
|
181
|
+
maintainAspectRatio: false,
|
|
182
|
+
legend: {
|
|
183
|
+
display: false
|
|
184
|
+
},
|
|
185
|
+
tooltips: {
|
|
186
|
+
enabled: false,
|
|
187
|
+
callbacks: {
|
|
188
|
+
title: (tooltipItems, data) => data.datasets[tooltipItems[0].datasetIndex]?.label || '',
|
|
189
|
+
label: tooltipItem => {
|
|
190
|
+
const value = +(tooltipItem.value || '0');
|
|
191
|
+
const ratio = toPrecision((value * 100) / total, 2);
|
|
192
|
+
return value > 0 ? `${toPrecision(value, 3)} (${ratio}%)` : '';
|
|
193
|
+
}
|
|
192
194
|
}
|
|
193
|
-
|
|
195
|
+
},
|
|
196
|
+
scales: {
|
|
197
|
+
xAxes: [
|
|
198
|
+
{
|
|
199
|
+
display: true
|
|
200
|
+
}
|
|
201
|
+
],
|
|
202
|
+
yAxes: [
|
|
203
|
+
{
|
|
204
|
+
position: 'left'
|
|
205
|
+
}
|
|
206
|
+
]
|
|
207
|
+
}
|
|
194
208
|
}
|
|
195
|
-
}
|
|
209
|
+
});
|
|
196
210
|
});
|
|
197
211
|
}
|
|
198
|
-
|
|
199
|
-
this.
|
|
212
|
+
selectTerm({ target: { value } }) {
|
|
213
|
+
const term = this.terms().find(term => term['@id'] === value);
|
|
214
|
+
this.selectedTerm.set(term);
|
|
215
|
+
}
|
|
216
|
+
selectMethod({ target: { value } }) {
|
|
217
|
+
const term = this.methods().find(term => term['@id'] === value);
|
|
218
|
+
this.selectedMethod.set(term);
|
|
200
219
|
}
|
|
201
220
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.8", ngImport: i0, type: ImpactAssessmentsIndicatorBreakdownChartComponent, deps: [{ token: i0.NgZone }, { token: i1.DomSanitizer }, { token: i2.HeSearchService }, { token: i3.HeNodeService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
202
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.8", type: ImpactAssessmentsIndicatorBreakdownChartComponent, selector: "he-impact-assessments-indicator-breakdown-chart", inputs: { impactAssessment: "impactAssessment", indicators: "indicators" }, viewQueries: [{ propertyName: "chartRef", first: true, predicate: ["chart"], descendants: true }], ngImport: i0, template: "<div class=\"p-3\" [class.is-hidden]=\"
|
|
221
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.8", type: ImpactAssessmentsIndicatorBreakdownChartComponent, selector: "he-impact-assessments-indicator-breakdown-chart", inputs: { impactAssessment: "impactAssessment", indicators: "indicators" }, viewQueries: [{ propertyName: "chartRef", first: true, predicate: ["chart"], descendants: true }], ngImport: i0, template: "<div class=\"p-3\" *ngIf=\"!loading(); else loader\" [class.is-hidden]=\"!terms()?.length\">\n <div class=\"columns\">\n <div class=\"column\">\n <div class=\"field has-addons\">\n <div class=\"control\">\n <span class=\"button is-small is-static is-secondary\">Select an Indicator</span>\n </div>\n <div class=\"control is-expanded\" *ngIf=\"terms()?.length\">\n <div class=\"select is-fullwidth is-small is-secondary\">\n <select (change)=\"selectTerm($event)\">\n <option *ngFor=\"let term of terms()\" [value]=\"term['@id']\">{{ term.name }} ({{ term.units }})</option>\n </select>\n </div>\n </div>\n <div class=\"control is-expanded\" *ngIf=\"methods()?.length\">\n <div class=\"select is-fullwidth is-small is-secondary\">\n <select (change)=\"selectMethod($event)\">\n <option [ngValue]=\"undefined\">Filter Model</option>\n <option *ngFor=\"let term of methods()\" [value]=\"term['@id']\">{{ term.name }}</option>\n </select>\n </div>\n </div>\n </div>\n </div>\n <div class=\"column is-narrow\">\n <a\n class=\"button is-ghost is-small\"\n [href]=\"csvContent()\"\n [download]=\"downloadFilename()\"\n nbgTooltip=\"Download as CSV\"\n placement=\"bottom\">\n <fa-icon icon=\"download\"></fa-icon>\n </a>\n </div>\n </div>\n\n <p class=\"is-size-7\" *ngIf=\"!selectedMethod()\">\n <i>Selecting a Model is recommended to avoid duplicated entries.</i>\n </p>\n</div>\n\n<div class=\"is-mt-1\">\n <p *ngIf=\"!loading() && noData()\" class=\"has-text-centered\">\n <span>No breakdown available for</span>\n <span class=\"is-pl-1\" *ngIf=\"selectedTerm()\">{{ selectedTerm().name }}</span>\n <span class=\"is-pl-1\" *ngIf=\"selectedMethod()\">({{ selectedMethod().name }})</span>\n <span>.</span>\n </p>\n <div class=\"chart-container h-100\">\n <canvas #chart></canvas>\n </div>\n</div>\n\n<ng-template #loader>\n <div class=\"has-text-center py-3\">\n <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n </div>\n</ng-template>\n", styles: [":host{display:block;overflow:visible}.chart-container{height:400px;position:relative}\n"], dependencies: [{ kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i5.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i5.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "component", type: i6.FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "spin", "pulse", "mask", "styles", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "classes", "transform", "a11yRole"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
203
222
|
}
|
|
204
223
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.8", ngImport: i0, type: ImpactAssessmentsIndicatorBreakdownChartComponent, decorators: [{
|
|
205
224
|
type: Component,
|
|
206
|
-
args: [{ selector: 'he-impact-assessments-indicator-breakdown-chart', template: "<div class=\"p-3\" [class.is-hidden]=\"
|
|
225
|
+
args: [{ selector: 'he-impact-assessments-indicator-breakdown-chart', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"p-3\" *ngIf=\"!loading(); else loader\" [class.is-hidden]=\"!terms()?.length\">\n <div class=\"columns\">\n <div class=\"column\">\n <div class=\"field has-addons\">\n <div class=\"control\">\n <span class=\"button is-small is-static is-secondary\">Select an Indicator</span>\n </div>\n <div class=\"control is-expanded\" *ngIf=\"terms()?.length\">\n <div class=\"select is-fullwidth is-small is-secondary\">\n <select (change)=\"selectTerm($event)\">\n <option *ngFor=\"let term of terms()\" [value]=\"term['@id']\">{{ term.name }} ({{ term.units }})</option>\n </select>\n </div>\n </div>\n <div class=\"control is-expanded\" *ngIf=\"methods()?.length\">\n <div class=\"select is-fullwidth is-small is-secondary\">\n <select (change)=\"selectMethod($event)\">\n <option [ngValue]=\"undefined\">Filter Model</option>\n <option *ngFor=\"let term of methods()\" [value]=\"term['@id']\">{{ term.name }}</option>\n </select>\n </div>\n </div>\n </div>\n </div>\n <div class=\"column is-narrow\">\n <a\n class=\"button is-ghost is-small\"\n [href]=\"csvContent()\"\n [download]=\"downloadFilename()\"\n nbgTooltip=\"Download as CSV\"\n placement=\"bottom\">\n <fa-icon icon=\"download\"></fa-icon>\n </a>\n </div>\n </div>\n\n <p class=\"is-size-7\" *ngIf=\"!selectedMethod()\">\n <i>Selecting a Model is recommended to avoid duplicated entries.</i>\n </p>\n</div>\n\n<div class=\"is-mt-1\">\n <p *ngIf=\"!loading() && noData()\" class=\"has-text-centered\">\n <span>No breakdown available for</span>\n <span class=\"is-pl-1\" *ngIf=\"selectedTerm()\">{{ selectedTerm().name }}</span>\n <span class=\"is-pl-1\" *ngIf=\"selectedMethod()\">({{ selectedMethod().name }})</span>\n <span>.</span>\n </p>\n <div class=\"chart-container h-100\">\n <canvas #chart></canvas>\n </div>\n</div>\n\n<ng-template #loader>\n <div class=\"has-text-center py-3\">\n <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n </div>\n</ng-template>\n", styles: [":host{display:block;overflow:visible}.chart-container{height:400px;position:relative}\n"] }]
|
|
207
226
|
}], ctorParameters: function () { return [{ type: i0.NgZone }, { type: i1.DomSanitizer }, { type: i2.HeSearchService }, { type: i3.HeNodeService }]; }, propDecorators: { chartRef: [{
|
|
208
227
|
type: ViewChild,
|
|
209
228
|
args: ['chart']
|
|
@@ -212,4 +231,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.8", ngImpor
|
|
|
212
231
|
}], indicators: [{
|
|
213
232
|
type: Input
|
|
214
233
|
}] } });
|
|
215
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"impact-assessments-indicator-breakdown-chart.component.js","sourceRoot":"","sources":["../../../../src/impact-assessments/impact-assessments-indicator-breakdown-chart/impact-assessments-indicator-breakdown-chart.component.ts","../../../../src/impact-assessments/impact-assessments-indicator-breakdown-chart/impact-assessments-indicator-breakdown-chart.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, Input, NgZone, OnInit, ViewChild } from '@angular/core';\nimport { DomSanitizer, SafeResourceUrl } from '@angular/platform-browser';\nimport { of } from 'rxjs';\nimport { filter, groupBy, map, mergeAll, mergeMap, toArray } from 'rxjs/operators';\nimport { Chart } from 'chart.js';\nimport ChartDataLabels from 'chartjs-plugin-datalabels';\nimport { DataState } from '@hestia-earth/api';\nimport { IImpactAssessmentJSONLD, Indicator, ITermJSONLD, NodeType, Term, TermTermType } from '@hestia-earth/schema';\nimport { toPrecision, unique } from '@hestia-earth/utils';\n\nimport { matchTermType, matchType } from '../../search/search.model';\nimport { HeSearchService } from '../../search/search.service';\nimport { HeNodeService } from '../../node/node.service';\nimport { Level, parseLines, parseMessage } from '../../common/logs-utils';\nimport { listColor } from '../../common/color';\n\ninterface ILog {\n  modelId: string;\n  impactTermId: string;\n  impactTermUnits?: string;\n  blankNodeTermId: string;\n  coefficient: number;\n  value: number;\n  // store all indicators with inputs\n  inputs?: {\n    name: string;\n    value: number;\n  }[];\n  inputsValue?: number;\n}\n\nconst parseLog = (data: any): ILog => ({\n  modelId: data.model,\n  impactTermId: data['key/term'] || data.term || data.indicator,\n  blankNodeTermId: data.node,\n  coefficient: +data.coefficient,\n  value: +data.value * +data.coefficient\n});\n\nconst filterTermTypes = [TermTermType.emission, TermTermType.characterisedIndicator];\n\nconst csvHeaders = [\n  'Impact',\n  'Impact Unit',\n  'Method',\n  'Emission',\n  'Value',\n  'Inputs',\n  'Inputs value',\n  'Functional Unit'\n];\n\nconst logToCsv = (logs: ILog[], impact: IImpactAssessmentJSONLD) =>\n  [\n    csvHeaders.join(','),\n    ...logs\n      .sort((a, b) => a.impactTermId.localeCompare(b.impactTermId))\n      .flatMap(({ impactTermId, impactTermUnits, modelId, blankNodeTermId, value, inputs }) => [\n        [impactTermId, impactTermUnits, modelId, blankNodeTermId, value, '', ''],\n        ...inputs.map(v => [impactTermId, impactTermUnits, modelId, blankNodeTermId, '', v.name, v.value])\n      ])\n      .map(v => [...v, impact.product?.term?.units].join(','))\n  ].join('\\n');\n\n@Component({\n  selector: 'he-impact-assessments-indicator-breakdown-chart',\n  templateUrl: './impact-assessments-indicator-breakdown-chart.component.html',\n  styleUrls: ['./impact-assessments-indicator-breakdown-chart.component.scss']\n})\nexport class ImpactAssessmentsIndicatorBreakdownChartComponent implements OnInit {\n  private logs: ILog[] = [];\n  private emissions: ITermJSONLD[] = [];\n\n  @ViewChild('chart')\n  private chartRef?: ElementRef;\n  private chart: any;\n\n  @Input()\n  public impactAssessment?: IImpactAssessmentJSONLD;\n  @Input()\n  public indicators: Indicator[] = [];\n\n  public loading = true;\n  public terms: Term[] = [];\n  public selectedTerm?: Term;\n  public methods: Term[] = [];\n  public selectedMethod?: Term;\n  public noData = false;\n  public csvContent?: SafeResourceUrl;\n\n  constructor(\n    private ngZone: NgZone,\n    private domSanitizer: DomSanitizer,\n    private searchService: HeSearchService,\n    private nodeService: HeNodeService\n  ) {}\n\n  async ngOnInit() {\n    this.terms = unique((this.indicators || []).map(({ term }) => term!).filter(Boolean));\n    this.methods = unique((this.indicators || []).map(({ methodModel }) => methodModel!).filter(Boolean));\n\n    const impacts = [...(this.impactAssessment.impacts || []), ...(this.impactAssessment.endpoints || [])];\n\n    this.logs = (await of({\n      ...this.impactAssessment,\n      dataState: DataState.recalculated\n    })\n      .pipe(\n        mergeMap(node => this.nodeService.getLog(node)),\n        map(value => (value ? parseLines(value) : [])),\n        mergeAll(),\n        filter(({ data }) => data.logger === 'hestia_earth.models' && data.level === Level.debug),\n        filter(({ data: { message } }) => !!message),\n        map(({ data: { message } }) => parseMessage(message)),\n        filter(message => 'node' in message),\n        map(parseLog),\n        filter(log => !!log.impactTermId && !!log.blankNodeTermId && !isNaN(log.value) && log.value > 0),\n        groupBy(log => [log.impactTermId, log.blankNodeTermId, log.modelId].join('/')),\n        mergeMap(group => group.pipe(toArray())),\n        map((values: ILog[]) => {\n          const log = values[0];\n          const total = values.reduce((prev, curr) => prev + curr.value, 0);\n          const blankNodes = [\n            ...(this.impactAssessment.emissionsResourceUse || []),\n            ...(this.impactAssessment.impacts || [])\n          ].filter(v => v.term['@id'] === log.blankNodeTermId);\n\n          const inputs = blankNodes\n            .filter(v => v.inputs?.length)\n            .map(v => ({ name: v.inputs.map(i => i['@id']).join(';'), value: v.value * log.coefficient }));\n          const inputsValue = inputs.reduce((prev, curr) => prev + curr.value, 0);\n\n          const impact = impacts.find(v => v.term['@id'] === log.impactTermId);\n\n          // logs might exist but impact was not added => skip logs\n          return impact\n            ? {\n                ...log,\n                impactTermUnits: impact.term?.units,\n                value: total,\n                inputs,\n                inputsValue\n              }\n            : null;\n        }),\n        filter(log => !!log),\n        toArray()\n      )\n      .toPromise()) as ILog[];\n    this.csvContent = this.domSanitizer.bypassSecurityTrustResourceUrl(\n      `data:text/html;charset=utf-8,${encodeURIComponent(logToCsv(this.logs, this.impactAssessment))}`\n    );\n\n    const { results } = await this.searchService.search<ITermJSONLD, NodeType.Term>({\n      fields: ['@type', '@id', 'name'],\n      limit: 1000,\n      query: {\n        bool: {\n          must: [matchType(NodeType.Term)],\n          should: filterTermTypes.map(termType => matchTermType(termType)),\n          minimum_should_match: 1\n        }\n      }\n    });\n    this.emissions = results;\n\n    this.loading = false;\n\n    this.selectedTerm = this.terms[0];\n    this.updateChart();\n  }\n\n  public get id() {\n    return this.impactAssessment!['@id'] || (this.impactAssessment as any).id;\n  }\n\n  private initChart() {\n    const logs = this.logs\n      .filter(\n        ({ impactTermId, modelId }) =>\n          impactTermId === this.selectedTerm?.['@id'] &&\n          (!this.selectedMethod || modelId === this.selectedMethod['@id'])\n      )\n      .sort((a, b) => b.value - a.value);\n    this.noData = logs.length === 0;\n\n    const total = logs.reduce((prev, curr) => prev + curr.value, 0);\n    const labels = logs.map(\n      ({ blankNodeTermId }) => this.emissions.find(v => v['@id'] === blankNodeTermId)?.name || blankNodeTermId\n    );\n\n    // display the breakdown by input\n    // const inputs = logs.flatMap(v => v.inputs.map(i => i.name));\n    // const datasets = [\n    //   ...inputs.map((input, index) => ({\n    //     label: input,\n    //     data: logs.map(log => log.inputs.find(({ name }) => name === input)?.value || 0),\n    //     backgroundColor: getColor(index),\n    //     borderColor: getColor(index)\n    //   })),\n    //   // add dataset for rest not linked to inputs\n    //   {\n    //     label: this.selectedTerm?.name ?? '',\n    //     data: logs.map(log => log.value - log.inputsValue),\n    //     backgroundColor: getColor(-1),\n    //     borderColor: getColor(-1)\n    //   }\n    // ];\n\n    const datasets = [\n      {\n        label: this.selectedTerm?.name ?? '',\n        data: logs.map(({ value }) => value),\n        backgroundColor: logs.map(listColor),\n        borderColor: logs.map(listColor)\n      }\n    ];\n\n    if (this.chart) {\n      this.chart.destroy();\n    }\n    this.chart = new Chart(this.chartRef?.nativeElement, {\n      type: 'horizontalBar',\n      data: {\n        labels,\n        datasets\n      },\n      plugins: [ChartDataLabels],\n      options: {\n        plugins: {\n          datalabels: {\n            color: 'black',\n            formatter: value => {\n              const ratio = toPrecision((value * 100) / total, 2);\n              return value > 0 ? `${toPrecision(value, 3)} (${ratio}%)` : '';\n            },\n            align: 'end'\n          }\n        },\n        responsive: true,\n        maintainAspectRatio: false,\n        legend: {\n          display: false\n        },\n        tooltips: {\n          enabled: false,\n          callbacks: {\n            title: (tooltipItems, data) => data.datasets![tooltipItems[0].datasetIndex!]?.label || '',\n            label: tooltipItem => {\n              const value = +(tooltipItem.value || '0');\n              const ratio = toPrecision((value * 100) / total, 2);\n              return value > 0 ? `${toPrecision(value, 3)} (${ratio}%)` : '';\n            }\n          }\n        },\n        scales: {\n          xAxes: [\n            {\n              display: true\n            }\n          ],\n          yAxes: [\n            {\n              position: 'left'\n            }\n          ]\n        }\n      }\n    });\n  }\n\n  public updateChart() {\n    this.ngZone.runOutsideAngular(() => this.initChart());\n  }\n}\n","<div class=\"p-3\" [class.is-hidden]=\"loading || !terms?.length\">\n  <div class=\"columns\">\n    <div class=\"column\">\n      <div class=\"field has-addons\">\n        <div class=\"control\">\n          <span class=\"button is-small is-static is-secondary\">Select an Indicator</span>\n        </div>\n        <div class=\"control is-expanded\">\n          <div class=\"select is-fullwidth is-small is-secondary\">\n            <select (change)=\"updateChart()\" [(ngModel)]=\"selectedTerm\" id=\"selectedTerm\">\n              <option *ngFor=\"let term of terms\" [ngValue]=\"term\">{{ term.name }} ({{ term.units }})</option>\n            </select>\n          </div>\n        </div>\n        <div class=\"control is-expanded\" *ngIf=\"methods?.length\">\n          <div class=\"select is-fullwidth is-small is-secondary\">\n            <select (change)=\"updateChart()\" [(ngModel)]=\"selectedMethod\" id=\"selectedMethod\">\n              <option [ngValue]=\"undefined\">Filter Model</option>\n              <option *ngFor=\"let term of methods\" [ngValue]=\"term\">{{ term.name }}</option>\n            </select>\n          </div>\n        </div>\n      </div>\n    </div>\n    <div class=\"column is-narrow\">\n      <a\n        class=\"button is-ghost is-small\"\n        [href]=\"csvContent\"\n        [download]=\"id + '-logs.csv'\"\n        nbgTooltip=\"Download as CSV\"\n        placement=\"bottom\">\n        <fa-icon icon=\"download\"></fa-icon>\n      </a>\n    </div>\n  </div>\n\n  <p class=\"is-size-7\" *ngIf=\"!selectedMethod\">\n    <i>Selecting a Model is recommended to avoid duplicated entries.</i>\n  </p>\n\n  <div class=\"mt-1\">\n    <p *ngIf=\"noData\" class=\"has-text-centered\">\n      <span>No breakdown available for {{ selectedTerm?.name }}</span>\n      <span class=\"pl-1\" *ngIf=\"selectedMethod\">({{ selectedMethod.name }})</span>\n      <span>.</span>\n    </p>\n    <div class=\"chart-container h-100\">\n      <canvas #chart></canvas>\n    </div>\n  </div>\n</div>\n\n<ng-container *ngIf=\"loading\">\n  <div class=\"has-text-center py-3\">\n    <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n  </div>\n</ng-container>\n"]}
|
|
234
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"impact-assessments-indicator-breakdown-chart.component.js","sourceRoot":"","sources":["../../../../src/impact-assessments/impact-assessments-indicator-breakdown-chart/impact-assessments-indicator-breakdown-chart.component.ts","../../../../src/impact-assessments/impact-assessments-indicator-breakdown-chart/impact-assessments-indicator-breakdown-chart.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  Component,\n  ElementRef,\n  Input,\n  NgZone,\n  ViewChild,\n  computed,\n  effect,\n  signal\n} from '@angular/core';\nimport { DomSanitizer } from '@angular/platform-browser';\nimport { toObservable, toSignal } from '@angular/core/rxjs-interop';\nimport { filter, groupBy, map, mergeAll, mergeMap, take, tap, toArray } from 'rxjs/operators';\nimport { Chart } from 'chart.js';\nimport ChartDataLabels from 'chartjs-plugin-datalabels';\nimport { DataState } from '@hestia-earth/api';\nimport { IImpactAssessmentJSONLD, Indicator, ITermJSONLD, NodeType, Term, TermTermType } from '@hestia-earth/schema';\nimport { toPrecision, unique } from '@hestia-earth/utils';\nimport { of } from 'rxjs';\n\nimport { matchTermType, matchType } from '../../search/search.model';\nimport { HeSearchService } from '../../search/search.service';\nimport { HeNodeService } from '../../node/node.service';\nimport { distinctUntilChangedDeep } from '../../common/rxjs-utils';\nimport { Level, parseLines, parseMessage } from '../../common/logs-utils';\nimport { listColor } from '../../common/color';\n\ninterface ILog {\n  modelId: string;\n  impactTermId: string;\n  impactTermUnits?: string;\n  blankNodeTermId: string;\n  coefficient: number;\n  value: number;\n  // store all indicators with inputs\n  inputs?: {\n    name: string;\n    value: number;\n  }[];\n  inputsValue?: number;\n}\n\nconst parseLog = (data: any): ILog => ({\n  modelId: data.model,\n  impactTermId: data['key/term'] || data.term || data.indicator,\n  blankNodeTermId: data.node,\n  coefficient: +data.coefficient,\n  value: +data.value * +data.coefficient\n});\n\nconst filterTermTypes = [TermTermType.emission, TermTermType.characterisedIndicator];\n\nconst csvHeaders = [\n  'Impact',\n  'Impact Unit',\n  'Method',\n  'Emission',\n  'Value',\n  'Inputs',\n  'Inputs value',\n  'Functional Unit'\n];\n\nconst logToCsv = (logs: ILog[], impact: IImpactAssessmentJSONLD) =>\n  [\n    csvHeaders.join(','),\n    ...logs\n      .sort((a, b) => a.impactTermId.localeCompare(b.impactTermId))\n      .flatMap(({ impactTermId, impactTermUnits, modelId, blankNodeTermId, value, inputs }) => [\n        [impactTermId, impactTermUnits, modelId, blankNodeTermId, value, '', ''],\n        ...inputs.map(v => [impactTermId, impactTermUnits, modelId, blankNodeTermId, '', v.name, v.value])\n      ])\n      .map(v => [...v, impact.product?.term?.units].join(','))\n  ].join('\\n');\n\n@Component({\n  selector: 'he-impact-assessments-indicator-breakdown-chart',\n  templateUrl: './impact-assessments-indicator-breakdown-chart.component.html',\n  styleUrls: ['./impact-assessments-indicator-breakdown-chart.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class ImpactAssessmentsIndicatorBreakdownChartComponent implements AfterViewInit {\n  @ViewChild('chart')\n  private chartRef?: ElementRef;\n  private chart: any;\n\n  private _impactAssessment = signal(undefined as IImpactAssessmentJSONLD);\n  private impactAssessment$ = toObservable(this._impactAssessment);\n  @Input()\n  set impactAssessment(value: IImpactAssessmentJSONLD) {\n    this._impactAssessment.set(value);\n  }\n\n  private _indicators = signal([] as Indicator[]);\n  @Input()\n  set indicators(value: Indicator[]) {\n    this._indicators.set(value);\n  }\n\n  private allLogs = signal([] as ILog[]);\n  private logs = computed(() =>\n    this.allLogs()\n      .filter(\n        ({ impactTermId, modelId }) =>\n          impactTermId === this.selectedTerm()?.['@id'] &&\n          (!this.selectedMethod() || modelId === this.selectedMethod()?.['@id'])\n      )\n      .sort((a, b) => b.value - a.value)\n  );\n\n  private emissions = toSignal(\n    this.searchService\n      .search$<ITermJSONLD, NodeType.Term>({\n        fields: ['@type', '@id', 'name'],\n        limit: 1000,\n        query: {\n          bool: {\n            must: [matchType(NodeType.Term)],\n            should: filterTermTypes.map(termType => matchTermType(termType)),\n            minimum_should_match: 1\n          }\n        }\n      })\n      .pipe(map(({ results }) => results))\n  );\n\n  protected loading = signal(true);\n\n  protected selectedTerm = signal(undefined as Term);\n  protected terms = computed(() => unique((this._indicators() || []).map(({ term }) => term!).filter(Boolean)));\n\n  protected selectedMethod = signal(undefined as Term);\n  protected methods = computed(() =>\n    unique((this._indicators() || []).map(({ methodModel }) => methodModel!).filter(Boolean))\n  );\n\n  private impacts = computed(() => [\n    ...(this._impactAssessment()?.impacts || []),\n    ...(this._impactAssessment()?.endpoints || [])\n  ]);\n  protected noData = computed(() => this.logs()?.length === 0);\n\n  protected csvContent = computed(() =>\n    this.domSanitizer.bypassSecurityTrustResourceUrl(\n      `data:text/html;charset=utf-8,${encodeURIComponent(logToCsv(this.logs(), this._impactAssessment()))}`\n    )\n  );\n  private id = computed(() => this._impactAssessment()?.['@id'] || (this._impactAssessment() as any)?.id);\n  protected downloadFilename = computed(() => `${this.id()}-logs.csv`);\n\n  private total = computed(() => this.logs().reduce((prev, curr) => prev + curr.value, 0));\n  private labels = computed(() =>\n    this.logs().map(\n      ({ blankNodeTermId }) => this.emissions()?.find(v => v['@id'] === blankNodeTermId)?.name || blankNodeTermId\n    )\n  );\n  private datasets = computed(() => [\n    {\n      label: this.selectedTerm()?.name ?? '',\n      data: this.logs().map(({ value }) => value),\n      backgroundColor: this.logs().map(listColor),\n      borderColor: this.logs().map(listColor)\n    }\n  ]);\n\n  constructor(\n    private ngZone: NgZone,\n    private domSanitizer: DomSanitizer,\n    private searchService: HeSearchService,\n    private nodeService: HeNodeService\n  ) {\n    this.impactAssessment$\n      .pipe(\n        filter(v => !!v),\n        distinctUntilChangedDeep(),\n        take(1),\n        tap(() => this.loading.set(true)),\n        mergeMap(node => (node ? this.nodeService.getLog({ ...node, dataState: DataState.recalculated }) : of(''))),\n        map(value => (value ? parseLines(value) : [])),\n        mergeAll(),\n        filter(({ data }) => data.logger === 'hestia_earth.models' && data.level === Level.debug),\n        filter(({ data: { message } }) => !!message),\n        map(({ data: { message } }) => parseMessage(message)),\n        filter(message => 'node' in message),\n        map(parseLog),\n        filter(log => !!log.impactTermId && !!log.blankNodeTermId && !isNaN(log.value) && log.value > 0),\n        groupBy(log => [log.impactTermId, log.blankNodeTermId, log.modelId].join('/')),\n        mergeMap(group => group.pipe(toArray())),\n        map((values: ILog[]) => {\n          const log = values[0];\n          const total = values.reduce((prev, curr) => prev + curr.value, 0);\n          const blankNodes = [\n            ...(this._impactAssessment()?.emissionsResourceUse ?? []),\n            ...(this._impactAssessment()?.impacts ?? [])\n          ].filter(v => v.term['@id'] === log.blankNodeTermId);\n\n          const inputs = blankNodes\n            .filter(v => v.inputs?.length)\n            .map(v => ({ name: v.inputs.map(i => i['@id']).join(';'), value: v.value * log.coefficient }));\n          const inputsValue = inputs.reduce((prev, curr) => prev + curr.value, 0);\n\n          const impact = this.impacts().find(v => v.term['@id'] === log.impactTermId);\n\n          // logs might exist but impact was not added => skip logs\n          return impact\n            ? {\n                ...log,\n                impactTermUnits: impact.term?.units,\n                value: total,\n                inputs,\n                inputsValue\n              }\n            : null;\n        }),\n        filter(log => !!log),\n        toArray(),\n        tap(() => this.loading.set(false))\n      )\n      .subscribe((logs: ILog[]) => {\n        this.allLogs.set(logs);\n        this.initChart();\n      });\n\n    effect(\n      () => {\n        // make sure selected term exists\n        const terms = this.terms();\n        const selectedTermId = this.selectedTerm()?.['@id'];\n        if (!selectedTermId || !terms.find(term => term['@id'] === selectedTermId)) {\n          this.selectedTerm.set(terms[0]);\n        }\n        this.initChart();\n      },\n      { allowSignalWrites: true }\n    );\n\n    effect(() => {\n      if (this.datasets().length) {\n        this.initChart();\n      }\n    });\n  }\n\n  ngAfterViewInit() {\n    this.initChart();\n  }\n\n  private initChart() {\n    if (!this.chartRef) {\n      return;\n    }\n\n    const labels = this.labels();\n    const datasets = this.datasets();\n    const total = this.total();\n\n    this.ngZone.runOutsideAngular(() => {\n      if (this.chart) {\n        this.chart.destroy();\n      }\n\n      this.chart = new Chart(this.chartRef?.nativeElement, {\n        type: 'horizontalBar',\n        data: {\n          labels,\n          datasets\n        },\n        plugins: [ChartDataLabels],\n        options: {\n          plugins: {\n            datalabels: {\n              color: 'black',\n              formatter: value => {\n                const ratio = toPrecision((value * 100) / total, 2);\n                return value > 0 ? `${toPrecision(value, 3)} (${ratio}%)` : '';\n              },\n              align: 'end'\n            }\n          },\n          responsive: true,\n          maintainAspectRatio: false,\n          legend: {\n            display: false\n          },\n          tooltips: {\n            enabled: false,\n            callbacks: {\n              title: (tooltipItems, data) => data.datasets![tooltipItems[0].datasetIndex!]?.label || '',\n              label: tooltipItem => {\n                const value = +(tooltipItem.value || '0');\n                const ratio = toPrecision((value * 100) / total, 2);\n                return value > 0 ? `${toPrecision(value, 3)} (${ratio}%)` : '';\n              }\n            }\n          },\n          scales: {\n            xAxes: [\n              {\n                display: true\n              }\n            ],\n            yAxes: [\n              {\n                position: 'left'\n              }\n            ]\n          }\n        }\n      });\n    });\n  }\n\n  protected selectTerm({ target: { value } }) {\n    const term = this.terms().find(term => term['@id'] === value);\n    this.selectedTerm.set(term);\n  }\n\n  protected selectMethod({ target: { value } }) {\n    const term = this.methods().find(term => term['@id'] === value);\n    this.selectedMethod.set(term);\n  }\n}\n","<div class=\"p-3\" *ngIf=\"!loading(); else loader\" [class.is-hidden]=\"!terms()?.length\">\n  <div class=\"columns\">\n    <div class=\"column\">\n      <div class=\"field has-addons\">\n        <div class=\"control\">\n          <span class=\"button is-small is-static is-secondary\">Select an Indicator</span>\n        </div>\n        <div class=\"control is-expanded\" *ngIf=\"terms()?.length\">\n          <div class=\"select is-fullwidth is-small is-secondary\">\n            <select (change)=\"selectTerm($event)\">\n              <option *ngFor=\"let term of terms()\" [value]=\"term['@id']\">{{ term.name }} ({{ term.units }})</option>\n            </select>\n          </div>\n        </div>\n        <div class=\"control is-expanded\" *ngIf=\"methods()?.length\">\n          <div class=\"select is-fullwidth is-small is-secondary\">\n            <select (change)=\"selectMethod($event)\">\n              <option [ngValue]=\"undefined\">Filter Model</option>\n              <option *ngFor=\"let term of methods()\" [value]=\"term['@id']\">{{ term.name }}</option>\n            </select>\n          </div>\n        </div>\n      </div>\n    </div>\n    <div class=\"column is-narrow\">\n      <a\n        class=\"button is-ghost is-small\"\n        [href]=\"csvContent()\"\n        [download]=\"downloadFilename()\"\n        nbgTooltip=\"Download as CSV\"\n        placement=\"bottom\">\n        <fa-icon icon=\"download\"></fa-icon>\n      </a>\n    </div>\n  </div>\n\n  <p class=\"is-size-7\" *ngIf=\"!selectedMethod()\">\n    <i>Selecting a Model is recommended to avoid duplicated entries.</i>\n  </p>\n</div>\n\n<div class=\"is-mt-1\">\n  <p *ngIf=\"!loading() && noData()\" class=\"has-text-centered\">\n    <span>No breakdown available for</span>\n    <span class=\"is-pl-1\" *ngIf=\"selectedTerm()\">{{ selectedTerm().name }}</span>\n    <span class=\"is-pl-1\" *ngIf=\"selectedMethod()\">({{ selectedMethod().name }})</span>\n    <span>.</span>\n  </p>\n  <div class=\"chart-container h-100\">\n    <canvas #chart></canvas>\n  </div>\n</div>\n\n<ng-template #loader>\n  <div class=\"has-text-center py-3\">\n    <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n  </div>\n</ng-template>\n"]}
|