@hestia-earth/ui-components 0.28.0-0 → 0.28.0-1

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.
@@ -2,7 +2,7 @@ import { Component, input, viewChild, computed, signal, effect, ChangeDetectionS
2
2
  import { FormsModule } from '@angular/forms';
3
3
  import { ellipsis, toPrecision } from '@hestia-earth/utils';
4
4
  import { propertyValue } from '@hestia-earth/utils/dist/term';
5
- import { BarController, BarElement, CategoryScale, Chart, LinearScale } from 'chart.js';
5
+ import { BarController, BarElement, CategoryScale, Chart, LinearScale, Tooltip } from 'chart.js';
6
6
  import ChartDataLabels from 'chartjs-plugin-datalabels';
7
7
  import { groupNodesByTerm } from '../../common/node-utils';
8
8
  import { defaultLabel } from '../../common/utils';
@@ -33,7 +33,7 @@ export class CyclesEmissionsChartComponent {
33
33
  borderColor: this.colors()
34
34
  }
35
35
  ]);
36
- Chart.register(CategoryScale, LinearScale, BarController, BarElement, ChartDataLabels);
36
+ Chart.register(CategoryScale, LinearScale, BarController, BarElement, Tooltip);
37
37
  effect(() => {
38
38
  this.chart = new Chart(this.chartRef().nativeElement, {
39
39
  type: 'bar',
@@ -41,19 +41,20 @@ export class CyclesEmissionsChartComponent {
41
41
  labels: [],
42
42
  datasets: []
43
43
  },
44
+ plugins: [ChartDataLabels],
44
45
  options: {
45
46
  plugins: {
46
47
  legend: {
47
48
  display: false
48
49
  },
50
+ tooltip: {
51
+ enabled: true
52
+ },
49
53
  datalabels: {
50
54
  color: 'black',
51
55
  formatter: value => toPrecision(value, 3),
52
56
  anchor: 'center',
53
57
  clamp: true
54
- },
55
- tooltip: {
56
- enabled: true
57
58
  }
58
59
  },
59
60
  responsive: true,
@@ -85,7 +86,6 @@ export class CyclesEmissionsChartComponent {
85
86
  });
86
87
  }
87
88
  ngOnDestroy() {
88
- Chart.unregister(ChartDataLabels);
89
89
  return this.chart && this.chart.destroy();
90
90
  }
91
91
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.8", ngImport: i0, type: CyclesEmissionsChartComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
@@ -95,4 +95,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.8", ngImpor
95
95
  type: Component,
96
96
  args: [{ selector: 'he-cycles-emissions-chart', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [FormsModule], template: "<div class=\"is-p-3\" [class.is-hidden]=\"!terms()?.length\">\n <div class=\"field has-addons is-pt-2 is-px-3\">\n <div class=\"control\">\n <span class=\"button is-small is-static is-secondary\">Select an Emission</span>\n </div>\n <div class=\"control is-expanded\">\n <div class=\"select is-small is-fullwidth is-secondary\">\n <select\n [ngModel]=\"selectedTermName()\"\n (change)=\"selectedTermName.set($event.target.value)\"\n id=\"selectedTermName\">\n @for (term of terms(); track term) {\n <option [value]=\"term.name\">{{ term.name }} ({{ term.units }})</option>\n }\n </select>\n </div>\n </div>\n </div>\n\n <div class=\"mt-1\">\n <div class=\"chart-container\">\n <canvas #chart></canvas>\n </div>\n </div>\n</div>\n", styles: [":host{display:block;overflow:visible}.chart-container{height:400px;position:relative}\n"] }]
97
97
  }], ctorParameters: () => [] });
98
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cycles-emissions-chart.component.js","sourceRoot":"","sources":["../../../../src/cycles/cycles-emissions-chart/cycles-emissions-chart.component.ts","../../../../src/cycles/cycles-emissions-chart/cycles-emissions-chart.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ElementRef,\n  input,\n  viewChild,\n  computed,\n  signal,\n  effect,\n  ChangeDetectionStrategy,\n  OnDestroy\n} from '@angular/core';\nimport { FormsModule } from '@angular/forms';\nimport { Emission, ICycleJSONLD } from '@hestia-earth/schema';\nimport { ellipsis, toPrecision } from '@hestia-earth/utils';\nimport { propertyValue } from '@hestia-earth/utils/dist/term';\nimport { BarController, BarElement, CategoryScale, Chart, LinearScale } from 'chart.js';\nimport ChartDataLabels from 'chartjs-plugin-datalabels';\n\nimport { groupNodesByTerm, IGroupedNodesValues } from '../../common/node-utils';\nimport { defaultLabel } from '../../common/utils';\nimport { listColor } from '../../common/color';\n\nconst cycleValue = (cycle: ICycleJSONLD, values: IGroupedNodesValues<Emission>) =>\n  (values[cycle['@id']]?.nodes[0] || { value: [0] }).value;\n\nconst cycleName = (cycle: ICycleJSONLD, index: number) => ellipsis(`${index + 1}. ${defaultLabel(cycle)}`, 25);\n\n@Component({\n  selector: 'he-cycles-emissions-chart',\n  templateUrl: './cycles-emissions-chart.component.html',\n  styleUrls: ['./cycles-emissions-chart.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [FormsModule]\n})\nexport class CyclesEmissionsChartComponent implements OnDestroy {\n  private chartRef = viewChild.required<ElementRef>('chart');\n  private chart: Chart<'bar', number[], string>;\n\n  protected cycles = input.required<ICycleJSONLD[]>();\n\n  protected emissionPerCycle = computed(() => groupNodesByTerm<ICycleJSONLD, Emission>(this.cycles(), 'emissions'));\n  protected terms = computed(() =>\n    Object.values(this.emissionPerCycle())\n      .filter(({ values }) =>\n        Object.values(values).some(({ nodes: [{ value }] }) => (propertyValue(value) as number) >= 0)\n      )\n      .map(({ term }) => term)\n  );\n  protected selectedTermName = signal<string>(undefined);\n\n  private selectedTerm = computed(() => this.terms().find(term => term.name === this.selectedTermName()));\n  private selectedTermId = computed(() => this.selectedTerm()?.['@id'] ?? '');\n  private values = computed(\n    () => (this.emissionPerCycle()[this.selectedTermName()] || { values: {} as IGroupedNodesValues<Emission> }).values\n  );\n  private labels = computed(() => this.cycles().map(cycleName));\n  private colors = computed(() => this.cycles().map(listColor));\n  private datasets = computed(() => [\n    {\n      label: this.selectedTermName(),\n      data: this.cycles().map(\n        cycle => propertyValue(cycleValue(cycle, this.values()), this.selectedTermId()) as number\n      ),\n      backgroundColor: this.colors(),\n      borderColor: this.colors()\n    }\n  ]);\n\n  constructor() {\n    Chart.register(CategoryScale, LinearScale, BarController, BarElement, ChartDataLabels);\n\n    effect(() => {\n      this.chart = new Chart(this.chartRef().nativeElement, {\n        type: 'bar',\n        data: {\n          labels: [],\n          datasets: []\n        },\n        options: {\n          plugins: {\n            legend: {\n              display: false\n            },\n            datalabels: {\n              color: 'black',\n              formatter: value => toPrecision(value, 3),\n              anchor: 'center',\n              clamp: true\n            },\n            tooltip: {\n              enabled: true\n            }\n          },\n          responsive: true,\n          maintainAspectRatio: false,\n          scales: {\n            x: {\n              display: true\n            },\n            y: {\n              position: 'left',\n              beginAtZero: true\n            }\n          }\n        }\n      });\n    });\n\n    effect(\n      () => {\n        // make sure selected term exists\n        const terms = this.terms();\n        const selectedTermName = this.selectedTermName();\n        if (!selectedTermName || !this.selectedTerm()) {\n          this.selectedTermName.set(terms[0].name);\n        }\n      },\n      { allowSignalWrites: true }\n    );\n\n    effect(() => {\n      this.chart.data.labels = this.labels();\n      this.chart.data.datasets = this.datasets();\n      this.chart.update();\n    });\n  }\n\n  ngOnDestroy() {\n    Chart.unregister(ChartDataLabels);\n    return this.chart && this.chart.destroy();\n  }\n}\n","<div class=\"is-p-3\" [class.is-hidden]=\"!terms()?.length\">\n  <div class=\"field has-addons is-pt-2 is-px-3\">\n    <div class=\"control\">\n      <span class=\"button is-small is-static is-secondary\">Select an Emission</span>\n    </div>\n    <div class=\"control is-expanded\">\n      <div class=\"select is-small is-fullwidth is-secondary\">\n        <select\n          [ngModel]=\"selectedTermName()\"\n          (change)=\"selectedTermName.set($event.target.value)\"\n          id=\"selectedTermName\">\n          @for (term of terms(); track term) {\n            <option [value]=\"term.name\">{{ term.name }} ({{ term.units }})</option>\n          }\n        </select>\n      </div>\n    </div>\n  </div>\n\n  <div class=\"mt-1\">\n    <div class=\"chart-container\">\n      <canvas #chart></canvas>\n    </div>\n  </div>\n</div>\n"]}
98
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cycles-emissions-chart.component.js","sourceRoot":"","sources":["../../../../src/cycles/cycles-emissions-chart/cycles-emissions-chart.component.ts","../../../../src/cycles/cycles-emissions-chart/cycles-emissions-chart.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ElementRef,\n  input,\n  viewChild,\n  computed,\n  signal,\n  effect,\n  ChangeDetectionStrategy,\n  OnDestroy\n} from '@angular/core';\nimport { FormsModule } from '@angular/forms';\nimport { Emission, ICycleJSONLD } from '@hestia-earth/schema';\nimport { ellipsis, toPrecision } from '@hestia-earth/utils';\nimport { propertyValue } from '@hestia-earth/utils/dist/term';\nimport { BarController, BarElement, CategoryScale, Chart, LinearScale, Tooltip } from 'chart.js';\nimport ChartDataLabels from 'chartjs-plugin-datalabels';\n\nimport { groupNodesByTerm, IGroupedNodesValues } from '../../common/node-utils';\nimport { defaultLabel } from '../../common/utils';\nimport { listColor } from '../../common/color';\n\nconst cycleValue = (cycle: ICycleJSONLD, values: IGroupedNodesValues<Emission>) =>\n  (values[cycle['@id']]?.nodes[0] || { value: [0] }).value;\n\nconst cycleName = (cycle: ICycleJSONLD, index: number) => ellipsis(`${index + 1}. ${defaultLabel(cycle)}`, 25);\n\n@Component({\n  selector: 'he-cycles-emissions-chart',\n  templateUrl: './cycles-emissions-chart.component.html',\n  styleUrls: ['./cycles-emissions-chart.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [FormsModule]\n})\nexport class CyclesEmissionsChartComponent implements OnDestroy {\n  private chartRef = viewChild.required<ElementRef>('chart');\n  private chart: Chart<any, number[], string>;\n\n  protected cycles = input.required<ICycleJSONLD[]>();\n\n  protected emissionPerCycle = computed(() => groupNodesByTerm<ICycleJSONLD, Emission>(this.cycles(), 'emissions'));\n  protected terms = computed(() =>\n    Object.values(this.emissionPerCycle())\n      .filter(({ values }) =>\n        Object.values(values).some(({ nodes: [{ value }] }) => (propertyValue(value) as number) >= 0)\n      )\n      .map(({ term }) => term)\n  );\n  protected selectedTermName = signal<string>(undefined);\n\n  private selectedTerm = computed(() => this.terms().find(term => term.name === this.selectedTermName()));\n  private selectedTermId = computed(() => this.selectedTerm()?.['@id'] ?? '');\n  private values = computed(\n    () => (this.emissionPerCycle()[this.selectedTermName()] || { values: {} as IGroupedNodesValues<Emission> }).values\n  );\n  private labels = computed(() => this.cycles().map(cycleName));\n  private colors = computed(() => this.cycles().map(listColor));\n  private datasets = computed(() => [\n    {\n      label: this.selectedTermName(),\n      data: this.cycles().map(\n        cycle => propertyValue(cycleValue(cycle, this.values()), this.selectedTermId()) as number\n      ),\n      backgroundColor: this.colors(),\n      borderColor: this.colors()\n    }\n  ]);\n\n  constructor() {\n    Chart.register(CategoryScale, LinearScale, BarController, BarElement, Tooltip);\n\n    effect(() => {\n      this.chart = new Chart(this.chartRef().nativeElement, {\n        type: 'bar',\n        data: {\n          labels: [],\n          datasets: []\n        },\n        plugins: [ChartDataLabels],\n        options: {\n          plugins: {\n            legend: {\n              display: false\n            },\n            tooltip: {\n              enabled: true\n            },\n            datalabels: {\n              color: 'black',\n              formatter: value => toPrecision(value, 3),\n              anchor: 'center',\n              clamp: true\n            }\n          },\n          responsive: true,\n          maintainAspectRatio: false,\n          scales: {\n            x: {\n              display: true\n            },\n            y: {\n              position: 'left',\n              beginAtZero: true\n            }\n          }\n        }\n      });\n    });\n\n    effect(\n      () => {\n        // make sure selected term exists\n        const terms = this.terms();\n        const selectedTermName = this.selectedTermName();\n        if (!selectedTermName || !this.selectedTerm()) {\n          this.selectedTermName.set(terms[0].name);\n        }\n      },\n      { allowSignalWrites: true }\n    );\n\n    effect(() => {\n      this.chart.data.labels = this.labels();\n      this.chart.data.datasets = this.datasets();\n      this.chart.update();\n    });\n  }\n\n  ngOnDestroy() {\n    return this.chart && this.chart.destroy();\n  }\n}\n","<div class=\"is-p-3\" [class.is-hidden]=\"!terms()?.length\">\n  <div class=\"field has-addons is-pt-2 is-px-3\">\n    <div class=\"control\">\n      <span class=\"button is-small is-static is-secondary\">Select an Emission</span>\n    </div>\n    <div class=\"control is-expanded\">\n      <div class=\"select is-small is-fullwidth is-secondary\">\n        <select\n          [ngModel]=\"selectedTermName()\"\n          (change)=\"selectedTermName.set($event.target.value)\"\n          id=\"selectedTermName\">\n          @for (term of terms(); track term) {\n            <option [value]=\"term.name\">{{ term.name }} ({{ term.units }})</option>\n          }\n        </select>\n      </div>\n    </div>\n  </div>\n\n  <div class=\"mt-1\">\n    <div class=\"chart-container\">\n      <canvas #chart></canvas>\n    </div>\n  </div>\n</div>\n"]}
@@ -1,5 +1,5 @@
1
1
  import { Component, ChangeDetectionStrategy, viewChild, input, effect, computed } from '@angular/core';
2
- import { Chart, CategoryScale, LinearScale, BarController, BarElement } from 'chart.js';
2
+ import { Chart, CategoryScale, LinearScale, BarController, BarElement, Tooltip } from 'chart.js';
3
3
  import { propertyValue } from '@hestia-earth/utils/dist/term';
4
4
  import { groupNodesByTerm } from '../../common/node-utils';
5
5
  import { ellipsis, defaultLabel } from '../../common/utils';
@@ -22,7 +22,7 @@ export class CyclesResultComponent {
22
22
  data: this.cycles().map(({ '@id': id }) => (values[id] ? propertyValue(values[id].value, termId) : 0))
23
23
  };
24
24
  }));
25
- Chart.register(CategoryScale, LinearScale, BarController, BarElement);
25
+ Chart.register(CategoryScale, LinearScale, BarController, BarElement, Tooltip);
26
26
  effect(() => {
27
27
  this.chart = new Chart(this.chartRef().nativeElement, {
28
28
  type: 'bar',
@@ -36,18 +36,15 @@ export class CyclesResultComponent {
36
36
  maintainAspectRatio: false,
37
37
  plugins: {
38
38
  legend: {
39
- display: true
39
+ display: false
40
40
  },
41
41
  tooltip: {
42
42
  enabled: true,
43
- position: 'nearest'
44
- // callbacks: {
45
- // // title: (tooltipItems, data) => data.labels![tooltipItems[0].index!] as any
46
- // title() {
47
- // console.log(arguments);
48
- // return 'test';
49
- // }
50
- // }
43
+ callbacks: {
44
+ title([{ dataIndex }]) {
45
+ return this.chart.data.labels[dataIndex];
46
+ }
47
+ }
51
48
  }
52
49
  },
53
50
  scales: {
@@ -81,4 +78,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.8", ngImpor
81
78
  type: Component,
82
79
  args: [{ selector: 'he-cycles-result', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, template: "<div class=\"chart-container\">\n <canvas #chart></canvas>\n</div>\n", styles: [":host{display:block}.chart-container{height:100%;min-height:300px;position:relative}\n"] }]
83
80
  }], ctorParameters: () => [] });
84
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cycles-result.component.js","sourceRoot":"","sources":["../../../../src/cycles/cycles-result/cycles-result.component.ts","../../../../src/cycles/cycles-result/cycles-result.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ElementRef,\n  ChangeDetectionStrategy,\n  viewChild,\n  input,\n  effect,\n  computed,\n  OnDestroy\n} from '@angular/core';\nimport { ICycleJSONLD, Product } from '@hestia-earth/schema';\nimport { Chart, CategoryScale, LinearScale, BarController, BarElement } from 'chart.js';\nimport { propertyValue, propertyValueType } from '@hestia-earth/utils/dist/term';\n\nimport { groupNodesByTerm } from '../../common/node-utils';\nimport { ellipsis, defaultLabel } from '../../common/utils';\nimport { getColor } from '../../common/color';\n\nconst cycleName = (cycle: ICycleJSONLD, index: number) => ellipsis(`${index + 1}. ${defaultLabel(cycle)}`, 25);\n\n@Component({\n  selector: 'he-cycles-result',\n  templateUrl: './cycles-result.component.html',\n  styleUrls: ['./cycles-result.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true\n})\nexport class CyclesResultComponent implements OnDestroy {\n  private chartRef = viewChild.required<ElementRef>('chart');\n  private chart: Chart<'bar', propertyValueType[], string>;\n\n  protected cycles = input.required<ICycleJSONLD[]>();\n\n  private productsPerCycle = computed(() => groupNodesByTerm<ICycleJSONLD, Product>(this.cycles(), 'products'));\n\n  private labels = computed(() => this.cycles().map(cycleName));\n  private datasets = computed(() =>\n    Object.values(this.productsPerCycle()).map(({ term: { name, units, '@id': termId }, values }, index) => {\n      const color = getColor(index);\n      return {\n        label: `${name}${units ? ` (in ${units})` : ''}`,\n        backgroundColor: color,\n        borderColor: color,\n        barPercentage: 0.5,\n        data: this.cycles().map(({ '@id': id }) => (values[id] ? propertyValue(values[id].value, termId) : 0))\n      };\n    })\n  );\n\n  constructor() {\n    Chart.register(CategoryScale, LinearScale, BarController, BarElement);\n\n    effect(() => {\n      this.chart = new Chart(this.chartRef().nativeElement, {\n        type: 'bar',\n        data: {\n          labels: [],\n          datasets: []\n        },\n        options: {\n          indexAxis: 'y',\n          responsive: true,\n          maintainAspectRatio: false,\n          plugins: {\n            legend: {\n              display: true\n            },\n            tooltip: {\n              enabled: true,\n              position: 'nearest'\n              // callbacks: {\n              //   // title: (tooltipItems, data) => data.labels![tooltipItems[0].index!] as any\n              //   title() {\n              //     console.log(arguments);\n              //     return 'test';\n              //   }\n              // }\n            }\n          },\n          scales: {\n            x: {\n              beginAtZero: true\n            },\n            y: {\n              position: 'left',\n              title: {\n                display: true,\n                text: 'Cycle'\n              }\n            }\n          }\n        }\n      });\n    });\n\n    effect(() => {\n      this.chart.data.labels = this.labels();\n      this.chart.data.datasets = this.datasets();\n      this.chart.update();\n    });\n  }\n\n  ngOnDestroy() {\n    return this.chart && this.chart.destroy();\n  }\n}\n","<div class=\"chart-container\">\n  <canvas #chart></canvas>\n</div>\n"]}
81
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cycles-result.component.js","sourceRoot":"","sources":["../../../../src/cycles/cycles-result/cycles-result.component.ts","../../../../src/cycles/cycles-result/cycles-result.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ElementRef,\n  ChangeDetectionStrategy,\n  viewChild,\n  input,\n  effect,\n  computed,\n  OnDestroy\n} from '@angular/core';\nimport { ICycleJSONLD, Product } from '@hestia-earth/schema';\nimport { Chart, CategoryScale, LinearScale, BarController, BarElement, Tooltip } from 'chart.js';\nimport { propertyValue, propertyValueType } from '@hestia-earth/utils/dist/term';\n\nimport { groupNodesByTerm } from '../../common/node-utils';\nimport { ellipsis, defaultLabel } from '../../common/utils';\nimport { getColor } from '../../common/color';\n\nconst cycleName = (cycle: ICycleJSONLD, index: number) => ellipsis(`${index + 1}. ${defaultLabel(cycle)}`, 25);\n\n@Component({\n  selector: 'he-cycles-result',\n  templateUrl: './cycles-result.component.html',\n  styleUrls: ['./cycles-result.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true\n})\nexport class CyclesResultComponent implements OnDestroy {\n  private chartRef = viewChild.required<ElementRef>('chart');\n  private chart: Chart<'bar', propertyValueType[], string>;\n\n  protected cycles = input.required<ICycleJSONLD[]>();\n\n  private productsPerCycle = computed(() => groupNodesByTerm<ICycleJSONLD, Product>(this.cycles(), 'products'));\n\n  private labels = computed(() => this.cycles().map(cycleName));\n  private datasets = computed(() =>\n    Object.values(this.productsPerCycle()).map(({ term: { name, units, '@id': termId }, values }, index) => {\n      const color = getColor(index);\n      return {\n        label: `${name}${units ? ` (in ${units})` : ''}`,\n        backgroundColor: color,\n        borderColor: color,\n        barPercentage: 0.5,\n        data: this.cycles().map(({ '@id': id }) => (values[id] ? propertyValue(values[id].value, termId) : 0))\n      };\n    })\n  );\n\n  constructor() {\n    Chart.register(CategoryScale, LinearScale, BarController, BarElement, Tooltip);\n\n    effect(() => {\n      this.chart = new Chart(this.chartRef().nativeElement, {\n        type: 'bar',\n        data: {\n          labels: [],\n          datasets: []\n        },\n        options: {\n          indexAxis: 'y',\n          responsive: true,\n          maintainAspectRatio: false,\n          plugins: {\n            legend: {\n              display: false\n            },\n            tooltip: {\n              enabled: true,\n              callbacks: {\n                title([{ dataIndex }]) {\n                  return this.chart.data.labels[dataIndex] as string;\n                }\n              }\n            }\n          },\n          scales: {\n            x: {\n              beginAtZero: true\n            },\n            y: {\n              position: 'left',\n              title: {\n                display: true,\n                text: 'Cycle'\n              }\n            }\n          }\n        }\n      });\n    });\n\n    effect(() => {\n      this.chart.data.labels = this.labels();\n      this.chart.data.datasets = this.datasets();\n      this.chart.update();\n    });\n  }\n\n  ngOnDestroy() {\n    return this.chart && this.chart.destroy();\n  }\n}\n","<div class=\"chart-container\">\n  <canvas #chart></canvas>\n</div>\n"]}
@@ -99,7 +99,7 @@ export class ImpactAssessmentsIndicatorBreakdownChartComponent {
99
99
  borderColor: this.logs().map(listColor)
100
100
  }
101
101
  ]);
102
- Chart.register(CategoryScale, LinearScale, BarController, BarElement, ChartDataLabels);
102
+ Chart.register(CategoryScale, LinearScale, BarController, BarElement);
103
103
  this.impactAssessment$
104
104
  .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) => {
105
105
  const log = values[0];
@@ -132,12 +132,16 @@ export class ImpactAssessmentsIndicatorBreakdownChartComponent {
132
132
  labels: [],
133
133
  datasets: []
134
134
  },
135
+ plugins: [ChartDataLabels],
135
136
  options: {
136
137
  indexAxis: 'y',
137
138
  plugins: {
138
139
  legend: {
139
140
  display: false
140
141
  },
142
+ tooltip: {
143
+ enabled: false
144
+ },
141
145
  datalabels: {
142
146
  color: 'black',
143
147
  formatter: value => {
@@ -145,9 +149,6 @@ export class ImpactAssessmentsIndicatorBreakdownChartComponent {
145
149
  return value > 0 ? `${toPrecision(value, 3)} (${ratio}%)` : '';
146
150
  },
147
151
  align: 'end'
148
- },
149
- tooltip: {
150
- enabled: false
151
152
  }
152
153
  },
153
154
  responsive: true,
@@ -178,7 +179,6 @@ export class ImpactAssessmentsIndicatorBreakdownChartComponent {
178
179
  });
179
180
  }
180
181
  ngOnDestroy() {
181
- Chart.unregister(ChartDataLabels);
182
182
  return this.chart && this.chart.destroy();
183
183
  }
184
184
  selectTerm({ target: { value } }) {
@@ -196,4 +196,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.8", ngImpor
196
196
  type: Component,
197
197
  args: [{ selector: 'he-impact-assessments-indicator-breakdown-chart', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [FaIconComponent, FormsModule], template: "@if (loading()) {\n <div class=\"has-text-center py-3\">\n <fa-icon [icon]=\"faSpinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n </div>\n} @else {\n <div class=\"p-3\" [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 @if (terms()?.length) {\n <div class=\"control is-expanded\">\n <div class=\"select is-fullwidth is-small is-secondary\">\n <select (change)=\"selectTerm($event)\">\n @for (term of terms(); track term) {\n <option [value]=\"term['@id']\">{{ term.name }} ({{ term.units }})</option>\n }\n </select>\n </div>\n </div>\n }\n @if (methods()?.length) {\n <div class=\"control is-expanded\">\n <div class=\"select is-fullwidth is-small is-secondary\">\n <select (change)=\"selectMethod($event)\">\n <option [ngValue]=\"undefined\">Filter Model</option>\n @for (term of methods(); track term) {\n <option [value]=\"term['@id']\">{{ term.name }}</option>\n }\n </select>\n </div>\n </div>\n }\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]=\"faDownload\"></fa-icon>\n </a>\n </div>\n </div>\n @if (!selectedMethod()) {\n <p class=\"is-size-7\">\n <i>Selecting a Model is recommended to avoid duplicated entries.</i>\n </p>\n }\n </div>\n}\n\n<div class=\"is-mt-1\">\n @if (!loading() && noData()) {\n <p class=\"has-text-centered\">\n <span>No breakdown available for</span>\n @if (selectedTerm()) {\n <span class=\"is-pl-1\">{{ selectedTerm().name }}</span>\n }\n @if (selectedMethod()) {\n <span class=\"is-pl-1\">({{ selectedMethod().name }})</span>\n }\n <span>.</span>\n </p>\n }\n <div class=\"chart-container h-100\">\n <canvas #chart></canvas>\n </div>\n</div>\n", styles: [":host{display:block;overflow:visible}.chart-container{height:400px;position:relative}\n"] }]
198
198
  }], ctorParameters: () => [] });
199
- //# 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  ChangeDetectionStrategy,\n  Component,\n  ElementRef,\n  OnDestroy,\n  computed,\n  effect,\n  inject,\n  input,\n  signal,\n  viewChild\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 { BarController, BarElement, CategoryScale, Chart, LinearScale } 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 { FormsModule } from '@angular/forms';\nimport { FaIconComponent } from '@fortawesome/angular-fontawesome';\nimport { faDownload, faSpinner } from '@fortawesome/free-solid-svg-icons';\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  standalone: true,\n  imports: [FaIconComponent, FormsModule]\n})\nexport class ImpactAssessmentsIndicatorBreakdownChartComponent implements OnDestroy {\n  private domSanitizer = inject(DomSanitizer);\n  private searchService = inject(HeSearchService);\n  private nodeService = inject(HeNodeService);\n\n  private chartRef = viewChild.required<ElementRef>('chart');\n  private chart: Chart<'bar', number[], string>;\n\n  protected readonly faSpinner = faSpinner;\n  protected readonly faDownload = faDownload;\n\n  protected impactAssessment = input(undefined as IImpactAssessmentJSONLD);\n  private impactAssessment$ = toObservable(this.impactAssessment);\n\n  protected indicators = input([] as Indicator[]);\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    Chart.register(CategoryScale, LinearScale, BarController, BarElement, ChartDataLabels);\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[]) => this.allLogs.set(logs));\n\n    effect(() => {\n      this.chart = new Chart(this.chartRef().nativeElement, {\n        type: 'bar',\n        data: {\n          labels: [],\n          datasets: []\n        },\n        options: {\n          indexAxis: 'y',\n          plugins: {\n            legend: {\n              display: false\n            },\n            datalabels: {\n              color: 'black',\n              formatter: value => {\n                const ratio = toPrecision((value * 100) / this.total(), 2);\n                return value > 0 ? `${toPrecision(value, 3)} (${ratio}%)` : '';\n              },\n              align: 'end'\n            },\n            tooltip: {\n              enabled: false\n            }\n          },\n          responsive: true,\n          maintainAspectRatio: false,\n          scales: {\n            x: {\n              display: true\n            },\n            y: {\n              position: 'left'\n            }\n          }\n        }\n      });\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      },\n      { allowSignalWrites: true }\n    );\n\n    effect(() => {\n      this.chart.data.labels = this.labels();\n      this.chart.data.datasets = this.datasets();\n      this.chart.update();\n    });\n  }\n\n  ngOnDestroy() {\n    Chart.unregister(ChartDataLabels);\n    return this.chart && this.chart.destroy();\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","@if (loading()) {\n  <div class=\"has-text-center py-3\">\n    <fa-icon [icon]=\"faSpinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n  </div>\n} @else {\n  <div class=\"p-3\" [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          @if (terms()?.length) {\n            <div class=\"control is-expanded\">\n              <div class=\"select is-fullwidth is-small is-secondary\">\n                <select (change)=\"selectTerm($event)\">\n                  @for (term of terms(); track term) {\n                    <option [value]=\"term['@id']\">{{ term.name }} ({{ term.units }})</option>\n                  }\n                </select>\n              </div>\n            </div>\n          }\n          @if (methods()?.length) {\n            <div class=\"control is-expanded\">\n              <div class=\"select is-fullwidth is-small is-secondary\">\n                <select (change)=\"selectMethod($event)\">\n                  <option [ngValue]=\"undefined\">Filter Model</option>\n                  @for (term of methods(); track term) {\n                    <option [value]=\"term['@id']\">{{ term.name }}</option>\n                  }\n                </select>\n              </div>\n            </div>\n          }\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]=\"faDownload\"></fa-icon>\n        </a>\n      </div>\n    </div>\n    @if (!selectedMethod()) {\n      <p class=\"is-size-7\">\n        <i>Selecting a Model is recommended to avoid duplicated entries.</i>\n      </p>\n    }\n  </div>\n}\n\n<div class=\"is-mt-1\">\n  @if (!loading() && noData()) {\n    <p class=\"has-text-centered\">\n      <span>No breakdown available for</span>\n      @if (selectedTerm()) {\n        <span class=\"is-pl-1\">{{ selectedTerm().name }}</span>\n      }\n      @if (selectedMethod()) {\n        <span class=\"is-pl-1\">({{ selectedMethod().name }})</span>\n      }\n      <span>.</span>\n    </p>\n  }\n  <div class=\"chart-container h-100\">\n    <canvas #chart></canvas>\n  </div>\n</div>\n"]}
199
+ //# 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  ChangeDetectionStrategy,\n  Component,\n  ElementRef,\n  OnDestroy,\n  computed,\n  effect,\n  inject,\n  input,\n  signal,\n  viewChild\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 { BarController, BarElement, CategoryScale, Chart, LinearScale } 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 { FormsModule } from '@angular/forms';\nimport { FaIconComponent } from '@fortawesome/angular-fontawesome';\nimport { faDownload, faSpinner } from '@fortawesome/free-solid-svg-icons';\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  standalone: true,\n  imports: [FaIconComponent, FormsModule]\n})\nexport class ImpactAssessmentsIndicatorBreakdownChartComponent implements OnDestroy {\n  private domSanitizer = inject(DomSanitizer);\n  private searchService = inject(HeSearchService);\n  private nodeService = inject(HeNodeService);\n\n  private chartRef = viewChild.required<ElementRef>('chart');\n  private chart: Chart<any, number[], string>;\n\n  protected readonly faSpinner = faSpinner;\n  protected readonly faDownload = faDownload;\n\n  protected impactAssessment = input(undefined as IImpactAssessmentJSONLD);\n  private impactAssessment$ = toObservable(this.impactAssessment);\n\n  protected indicators = input([] as Indicator[]);\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    Chart.register(CategoryScale, LinearScale, BarController, BarElement);\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[]) => this.allLogs.set(logs));\n\n    effect(() => {\n      this.chart = new Chart(this.chartRef().nativeElement, {\n        type: 'bar',\n        data: {\n          labels: [],\n          datasets: []\n        },\n        plugins: [ChartDataLabels],\n        options: {\n          indexAxis: 'y',\n          plugins: {\n            legend: {\n              display: false\n            },\n            tooltip: {\n              enabled: false\n            },\n            datalabels: {\n              color: 'black',\n              formatter: value => {\n                const ratio = toPrecision((value * 100) / this.total(), 2);\n                return value > 0 ? `${toPrecision(value, 3)} (${ratio}%)` : '';\n              },\n              align: 'end'\n            }\n          },\n          responsive: true,\n          maintainAspectRatio: false,\n          scales: {\n            x: {\n              display: true\n            },\n            y: {\n              position: 'left'\n            }\n          }\n        }\n      });\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      },\n      { allowSignalWrites: true }\n    );\n\n    effect(() => {\n      this.chart.data.labels = this.labels();\n      this.chart.data.datasets = this.datasets();\n      this.chart.update();\n    });\n  }\n\n  ngOnDestroy() {\n    return this.chart && this.chart.destroy();\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","@if (loading()) {\n  <div class=\"has-text-center py-3\">\n    <fa-icon [icon]=\"faSpinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n  </div>\n} @else {\n  <div class=\"p-3\" [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          @if (terms()?.length) {\n            <div class=\"control is-expanded\">\n              <div class=\"select is-fullwidth is-small is-secondary\">\n                <select (change)=\"selectTerm($event)\">\n                  @for (term of terms(); track term) {\n                    <option [value]=\"term['@id']\">{{ term.name }} ({{ term.units }})</option>\n                  }\n                </select>\n              </div>\n            </div>\n          }\n          @if (methods()?.length) {\n            <div class=\"control is-expanded\">\n              <div class=\"select is-fullwidth is-small is-secondary\">\n                <select (change)=\"selectMethod($event)\">\n                  <option [ngValue]=\"undefined\">Filter Model</option>\n                  @for (term of methods(); track term) {\n                    <option [value]=\"term['@id']\">{{ term.name }}</option>\n                  }\n                </select>\n              </div>\n            </div>\n          }\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]=\"faDownload\"></fa-icon>\n        </a>\n      </div>\n    </div>\n    @if (!selectedMethod()) {\n      <p class=\"is-size-7\">\n        <i>Selecting a Model is recommended to avoid duplicated entries.</i>\n      </p>\n    }\n  </div>\n}\n\n<div class=\"is-mt-1\">\n  @if (!loading() && noData()) {\n    <p class=\"has-text-centered\">\n      <span>No breakdown available for</span>\n      @if (selectedTerm()) {\n        <span class=\"is-pl-1\">{{ selectedTerm().name }}</span>\n      }\n      @if (selectedMethod()) {\n        <span class=\"is-pl-1\">({{ selectedMethod().name }})</span>\n      }\n      <span>.</span>\n    </p>\n  }\n  <div class=\"chart-container h-100\">\n    <canvas #chart></canvas>\n  </div>\n</div>\n"]}
@@ -39,7 +39,7 @@ export class ImpactAssessmentsIndicatorsChartComponent {
39
39
  borderColor: this.colors()
40
40
  }
41
41
  ]);
42
- Chart.register(CategoryScale, LinearScale, BarController, BarElement, ChartDataLabels);
42
+ Chart.register(CategoryScale, LinearScale, BarController, BarElement);
43
43
  effect(() => {
44
44
  this.chart = new Chart(this.chartRef()?.nativeElement, {
45
45
  type: 'bar',
@@ -47,19 +47,20 @@ export class ImpactAssessmentsIndicatorsChartComponent {
47
47
  labels: [],
48
48
  datasets: []
49
49
  },
50
+ plugins: [ChartDataLabels],
50
51
  options: {
51
52
  plugins: {
52
53
  legend: {
53
54
  display: false
54
55
  },
56
+ tooltip: {
57
+ enabled: false
58
+ },
55
59
  datalabels: {
56
60
  color: 'black',
57
61
  formatter: value => toPrecision(value, 3),
58
62
  anchor: 'center',
59
63
  clamp: true
60
- },
61
- tooltip: {
62
- enabled: false
63
64
  }
64
65
  },
65
66
  responsive: true,
@@ -91,7 +92,6 @@ export class ImpactAssessmentsIndicatorsChartComponent {
91
92
  });
92
93
  }
93
94
  ngOnDestroy() {
94
- Chart.unregister(ChartDataLabels);
95
95
  return this.chart && this.chart.destroy();
96
96
  }
97
97
  selectTerm({ target: { value } }) {
@@ -105,4 +105,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.8", ngImpor
105
105
  type: Component,
106
106
  args: [{ selector: 'he-impact-assessments-indicators-chart', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [FormsModule], template: "<div class=\"p-3\" [class.is-hidden]=\"!terms()?.length\">\n <div class=\"field has-addons pt-2 px-3\">\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-small is-fullwidth is-secondary\">\n <select (change)=\"selectTerm($event)\">\n @for (term of terms(); track term) {\n <option [value]=\"term['@id']\">{{ term.name }} ({{ term.units }})</option>\n }\n </select>\n </div>\n </div>\n </div>\n\n <div class=\"mt-1\">\n <div class=\"chart-container\">\n <canvas #chart></canvas>\n </div>\n </div>\n</div>\n", styles: [":host{display:block;overflow:visible}.chart-container{height:400px;position:relative}\n"] }]
107
107
  }], ctorParameters: () => [] });
108
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"impact-assessments-indicators-chart.component.js","sourceRoot":"","sources":["../../../../src/impact-assessments/impact-assessments-indicators-chart/impact-assessments-indicators-chart.component.ts","../../../../src/impact-assessments/impact-assessments-indicators-chart/impact-assessments-indicators-chart.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ElementRef,\n  computed,\n  ChangeDetectionStrategy,\n  signal,\n  effect,\n  input,\n  inject,\n  viewChild,\n  OnDestroy\n} from '@angular/core';\nimport { FormsModule } from '@angular/forms';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { DataState } from '@hestia-earth/api';\nimport { IImpactAssessmentJSONLD, Indicator, ITermJSONLD, NodeType, Term, TermTermType } from '@hestia-earth/schema';\nimport { toPrecision } from '@hestia-earth/utils';\nimport { BarController, BarElement, CategoryScale, Chart, LinearScale } from 'chart.js';\nimport ChartDataLabels from 'chartjs-plugin-datalabels';\n\nimport { HeNodeStoreService } from '../../node/node-store.service';\nimport { groupNodesByTerm, IGroupedNodesValues } from '../../common/node-utils';\nimport { defaultLabel } from '../../common/utils';\nimport { listColor } from '../../common/color';\nimport { modelKey } from '../impact-assessments.model';\n\nconst impactValue = (impact: IImpactAssessmentJSONLD, values: IGroupedNodesValues<Indicator>) =>\n  (values[impact['@id']]?.nodes[0] || { value: 0 }).value;\n\nconst impactName = (impact: IImpactAssessmentJSONLD, index: number) =>\n  `${index + 1}. ${defaultLabel(impact)} (${impact.product?.term?.units})`;\n\nconst termAllowed = (filterTermTypes: TermTermType[] = [], term?: Term | ITermJSONLD) =>\n  !filterTermTypes?.length || (filterTermTypes || []).includes(term?.termType);\n\n@Component({\n  selector: 'he-impact-assessments-indicators-chart',\n  templateUrl: './impact-assessments-indicators-chart.component.html',\n  styleUrls: ['./impact-assessments-indicators-chart.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [FormsModule]\n})\nexport class ImpactAssessmentsIndicatorsChartComponent implements OnDestroy {\n  private nodeStoreService = inject(HeNodeStoreService);\n\n  private chartRef = viewChild.required<ElementRef>('chart');\n  private chart: Chart<'bar', number[], string>;\n\n  protected key = input('impacts' as modelKey);\n  protected filterTermTypes = input([] as TermTermType[]);\n\n  private impactAssessments = toSignal(\n    this.nodeStoreService.findByState$<IImpactAssessmentJSONLD>(NodeType.ImpactAssessment, DataState.recalculated)\n  );\n\n  private indicatorPerImpactAssessment = computed(() =>\n    groupNodesByTerm<IImpactAssessmentJSONLD, Indicator>(this.impactAssessments(), this.key())\n  );\n\n  protected terms = computed(() =>\n    Object.values(this.indicatorPerImpactAssessment())\n      .filter(\n        ({ term, values }) =>\n          termAllowed(this.filterTermTypes(), term) &&\n          Object.values(values).some(({ nodes: [{ value }] }) => value >= 0)\n      )\n      .map(({ term }: any) => term as Term)\n  );\n\n  private selectedTerm = signal(undefined as Term);\n  private labels = computed(() => this.impactAssessments().map(impactName));\n  private colors = computed(() => this.impactAssessments().map(listColor));\n  private values = computed(() => this.indicatorPerImpactAssessment()?.[this.selectedTerm()?.name]?.values || {});\n  private datasets = computed(() => [\n    {\n      label: this.selectedTerm()?.name || '',\n      data: this.impactAssessments().map(impact => impactValue(impact, this.values())),\n      backgroundColor: this.colors(),\n      borderColor: this.colors()\n    }\n  ]);\n\n  constructor() {\n    Chart.register(CategoryScale, LinearScale, BarController, BarElement, ChartDataLabels);\n\n    effect(() => {\n      this.chart = new Chart(this.chartRef()?.nativeElement, {\n        type: 'bar',\n        data: {\n          labels: [],\n          datasets: []\n        },\n        options: {\n          plugins: {\n            legend: {\n              display: false\n            },\n            datalabels: {\n              color: 'black',\n              formatter: value => toPrecision(value, 3),\n              anchor: 'center',\n              clamp: true\n            },\n            tooltip: {\n              enabled: false\n            }\n          },\n          responsive: true,\n          maintainAspectRatio: false,\n          scales: {\n            x: {\n              display: true\n            },\n            y: {\n              position: 'left',\n              beginAtZero: true\n            }\n          }\n        }\n      });\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      },\n      { allowSignalWrites: true }\n    );\n\n    effect(() => {\n      this.chart.data.labels = this.labels();\n      this.chart.data.datasets = this.datasets();\n      this.chart.update();\n    });\n  }\n\n  ngOnDestroy() {\n    Chart.unregister(ChartDataLabels);\n    return this.chart && this.chart.destroy();\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","<div class=\"p-3\" [class.is-hidden]=\"!terms()?.length\">\n  <div class=\"field has-addons pt-2 px-3\">\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-small is-fullwidth is-secondary\">\n        <select (change)=\"selectTerm($event)\">\n          @for (term of terms(); track term) {\n            <option [value]=\"term['@id']\">{{ term.name }} ({{ term.units }})</option>\n          }\n        </select>\n      </div>\n    </div>\n  </div>\n\n  <div class=\"mt-1\">\n    <div class=\"chart-container\">\n      <canvas #chart></canvas>\n    </div>\n  </div>\n</div>\n"]}
108
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"impact-assessments-indicators-chart.component.js","sourceRoot":"","sources":["../../../../src/impact-assessments/impact-assessments-indicators-chart/impact-assessments-indicators-chart.component.ts","../../../../src/impact-assessments/impact-assessments-indicators-chart/impact-assessments-indicators-chart.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ElementRef,\n  computed,\n  ChangeDetectionStrategy,\n  signal,\n  effect,\n  input,\n  inject,\n  viewChild,\n  OnDestroy\n} from '@angular/core';\nimport { FormsModule } from '@angular/forms';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { DataState } from '@hestia-earth/api';\nimport { IImpactAssessmentJSONLD, Indicator, ITermJSONLD, NodeType, Term, TermTermType } from '@hestia-earth/schema';\nimport { toPrecision } from '@hestia-earth/utils';\nimport { BarController, BarElement, CategoryScale, Chart, LinearScale } from 'chart.js';\nimport ChartDataLabels from 'chartjs-plugin-datalabels';\n\nimport { HeNodeStoreService } from '../../node/node-store.service';\nimport { groupNodesByTerm, IGroupedNodesValues } from '../../common/node-utils';\nimport { defaultLabel } from '../../common/utils';\nimport { listColor } from '../../common/color';\nimport { modelKey } from '../impact-assessments.model';\n\nconst impactValue = (impact: IImpactAssessmentJSONLD, values: IGroupedNodesValues<Indicator>) =>\n  (values[impact['@id']]?.nodes[0] || { value: 0 }).value;\n\nconst impactName = (impact: IImpactAssessmentJSONLD, index: number) =>\n  `${index + 1}. ${defaultLabel(impact)} (${impact.product?.term?.units})`;\n\nconst termAllowed = (filterTermTypes: TermTermType[] = [], term?: Term | ITermJSONLD) =>\n  !filterTermTypes?.length || (filterTermTypes || []).includes(term?.termType);\n\n@Component({\n  selector: 'he-impact-assessments-indicators-chart',\n  templateUrl: './impact-assessments-indicators-chart.component.html',\n  styleUrls: ['./impact-assessments-indicators-chart.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [FormsModule]\n})\nexport class ImpactAssessmentsIndicatorsChartComponent implements OnDestroy {\n  private nodeStoreService = inject(HeNodeStoreService);\n\n  private chartRef = viewChild.required<ElementRef>('chart');\n  private chart: Chart<any, number[], string>;\n\n  protected key = input('impacts' as modelKey);\n  protected filterTermTypes = input([] as TermTermType[]);\n\n  private impactAssessments = toSignal(\n    this.nodeStoreService.findByState$<IImpactAssessmentJSONLD>(NodeType.ImpactAssessment, DataState.recalculated)\n  );\n\n  private indicatorPerImpactAssessment = computed(() =>\n    groupNodesByTerm<IImpactAssessmentJSONLD, Indicator>(this.impactAssessments(), this.key())\n  );\n\n  protected terms = computed(() =>\n    Object.values(this.indicatorPerImpactAssessment())\n      .filter(\n        ({ term, values }) =>\n          termAllowed(this.filterTermTypes(), term) &&\n          Object.values(values).some(({ nodes: [{ value }] }) => value >= 0)\n      )\n      .map(({ term }: any) => term as Term)\n  );\n\n  private selectedTerm = signal(undefined as Term);\n  private labels = computed(() => this.impactAssessments().map(impactName));\n  private colors = computed(() => this.impactAssessments().map(listColor));\n  private values = computed(() => this.indicatorPerImpactAssessment()?.[this.selectedTerm()?.name]?.values || {});\n  private datasets = computed(() => [\n    {\n      label: this.selectedTerm()?.name || '',\n      data: this.impactAssessments().map(impact => impactValue(impact, this.values())),\n      backgroundColor: this.colors(),\n      borderColor: this.colors()\n    }\n  ]);\n\n  constructor() {\n    Chart.register(CategoryScale, LinearScale, BarController, BarElement);\n\n    effect(() => {\n      this.chart = new Chart(this.chartRef()?.nativeElement, {\n        type: 'bar',\n        data: {\n          labels: [],\n          datasets: []\n        },\n        plugins: [ChartDataLabels],\n        options: {\n          plugins: {\n            legend: {\n              display: false\n            },\n            tooltip: {\n              enabled: false\n            },\n            datalabels: {\n              color: 'black',\n              formatter: value => toPrecision(value, 3),\n              anchor: 'center',\n              clamp: true\n            }\n          },\n          responsive: true,\n          maintainAspectRatio: false,\n          scales: {\n            x: {\n              display: true\n            },\n            y: {\n              position: 'left',\n              beginAtZero: true\n            }\n          }\n        }\n      });\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      },\n      { allowSignalWrites: true }\n    );\n\n    effect(() => {\n      this.chart.data.labels = this.labels();\n      this.chart.data.datasets = this.datasets();\n      this.chart.update();\n    });\n  }\n\n  ngOnDestroy() {\n    return this.chart && this.chart.destroy();\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","<div class=\"p-3\" [class.is-hidden]=\"!terms()?.length\">\n  <div class=\"field has-addons pt-2 px-3\">\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-small is-fullwidth is-secondary\">\n        <select (change)=\"selectTerm($event)\">\n          @for (term of terms(); track term) {\n            <option [value]=\"term['@id']\">{{ term.name }} ({{ term.units }})</option>\n          }\n        </select>\n      </div>\n    </div>\n  </div>\n\n  <div class=\"mt-1\">\n    <div class=\"chart-container\">\n      <canvas #chart></canvas>\n    </div>\n  </div>\n</div>\n"]}