@hestia-earth/ui-components 0.24.0-3 → 0.24.0-5

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.
@@ -1,3 +1,4 @@
1
+ /// <reference types="google.maps" />
1
2
  import { HttpClient } from '@angular/common/http';
2
3
  import { InjectionToken } from '@angular/core';
3
4
  import { Observable } from 'rxjs';
@@ -55,7 +56,12 @@ export declare const markerPie: (configs: IMarkerPieConfig[], size?: number) =>
55
56
  labelOrigin: google.maps.Point;
56
57
  };
57
58
  export declare const markerIcon: (scale?: number) => google.maps.Symbol;
58
- export declare const createMarker: (position?: google.maps.LatLng | google.maps.LatLngLiteral, text?: string, color?: string, scale?: number) => google.maps.Marker;
59
+ export interface IMarker {
60
+ position: google.maps.LatLng | google.maps.LatLngLiteral;
61
+ label: google.maps.MarkerLabel;
62
+ icon: google.maps.Symbol;
63
+ }
64
+ export declare const createMarker: (position?: google.maps.LatLng | google.maps.LatLngLiteral, text?: string, color?: string, scale?: number) => IMarker;
59
65
  interface IPolygon {
60
66
  type: 'Polygon';
61
67
  coordinates: number[][][];
@@ -78,11 +84,16 @@ export interface IGeometryCollection {
78
84
  type: 'GeometryCollection';
79
85
  geometries: geometry[];
80
86
  }
81
- type feature = IFeature | IFeatureCollection | IGeometryCollection | IPolygon | IMultiPolygon;
87
+ export type feature = IFeature | IFeatureCollection | IGeometryCollection | IPolygon | IMultiPolygon;
82
88
  export declare const defaultFeature: () => IFeature;
83
89
  export declare const pointToCoordinates: (point: google.maps.LatLng) => number[];
84
90
  export declare const coordinatesToPoint: (values: (number | number[])[]) => any;
85
91
  export declare const polygonToCoordinates: (polygon: google.maps.Polygon) => number[][];
92
+ export interface IPolygonMap {
93
+ paths: google.maps.MVCArray<google.maps.MVCArray<google.maps.LatLng>>;
94
+ options: google.maps.PolygonOptions;
95
+ }
96
+ export declare const polygonToMap: (polygon: google.maps.Polygon, color?: string) => IPolygonMap;
86
97
  export declare const addPolygonToFeature: (feat: feature, polygon: google.maps.Polygon) => any;
87
98
  export declare const polygonsFromFeature: (feat: feature, color?: string) => google.maps.Polygon[];
88
99
  export declare const polygonBounds: (polygon?: google.maps.Polygon | google.maps.Polygon[]) => google.maps.LatLngBounds;
@@ -1,6 +1,6 @@
1
1
  import { InjectionToken } from '@angular/core';
2
2
  import { of } from 'rxjs';
3
- import { catchError, map, mergeMap, shareReplay } from 'rxjs/operators';
3
+ import { catchError, delay, map, mergeMap, shareReplay } from 'rxjs/operators';
4
4
  const loadMapUrl = 'https://maps.googleapis.com/maps/api/js';
5
5
  const buildMapsUrl = (params) => Object.entries(params)
6
6
  .filter(([_key, values]) => values.length)
@@ -23,7 +23,7 @@ export const HE_MAP_LOADED = new InjectionToken('Google Maps API Loaded');
23
23
  * @param libraries
24
24
  * @returns
25
25
  */
26
- export const loadMapApi = (apiKey, libraries = ['visualization', 'drawing']) => (httpClient) => of(`${loadMapUrl}?${buildMapsUrl({ key: [apiKey], libraries })}`).pipe(mergeMap(url => httpClient.jsonp(url, 'callback')), shareReplay({ bufferSize: 1, refCount: true }), map(() => true), catchError(e => {
26
+ export const loadMapApi = (apiKey, libraries = ['visualization', 'drawing']) => (httpClient) => of(`${loadMapUrl}?${buildMapsUrl({ key: [apiKey], libraries })}`).pipe(mergeMap(url => httpClient.jsonp(url, 'callback')), shareReplay({ bufferSize: 1, refCount: true }), delay(100), map(() => true), catchError(e => {
27
27
  console.error(e);
28
28
  return of(false);
29
29
  }));
@@ -88,13 +88,13 @@ export const markerIcon = (scale = 7) => ({
88
88
  scale,
89
89
  labelOrigin: new google.maps.Point(0, -2)
90
90
  });
91
- export const createMarker = (position, text = '', color = fillColor, scale) => (position && position.lat && position.lng
92
- ? new google.maps.Marker({
91
+ export const createMarker = (position, text = '', color = fillColor, scale) => position?.lat && position?.lng
92
+ ? {
93
93
  position,
94
94
  label: { color, text },
95
95
  icon: markerIcon(scale)
96
- })
97
- : undefined);
96
+ }
97
+ : undefined;
98
98
  export const defaultFeature = () => ({
99
99
  type: 'Feature',
100
100
  properties: {},
@@ -123,6 +123,13 @@ export const polygonToCoordinates = (polygon) => polygon
123
123
  const points = path.getArray();
124
124
  return [...points.map(pointToCoordinates), pointToCoordinates(points[0])];
125
125
  })[0];
126
+ export const polygonToMap = (polygon, color = strokeColor) => ({
127
+ paths: polygon.getPaths(),
128
+ options: {
129
+ ...strokeStyle,
130
+ strokeColor: color
131
+ }
132
+ });
126
133
  const coordinatesToPolygon = (coordinates, color = strokeColor) => new google.maps.Polygon({
127
134
  paths: coordinates.map(coordinatesToPoint).filter(Boolean),
128
135
  ...strokeStyle,
@@ -151,4 +158,4 @@ export const polygonBounds = (polygon) => {
151
158
  : [];
152
159
  return paths.reduce((prev, curr) => prev.extend(curr), new google.maps.LatLngBounds());
153
160
  };
154
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"maps-utils.js","sourceRoot":"","sources":["../../../src/common/maps-utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpClient } from '@angular/common/http';\nimport { InjectionToken } from '@angular/core';\nimport { Observable, of } from 'rxjs';\nimport { catchError, map, mergeMap, shareReplay } from 'rxjs/operators';\n\nconst loadMapUrl = 'https://maps.googleapis.com/maps/api/js';\nconst buildMapsUrl = (params: Record<string, string[]>) =>\n  Object.entries(params)\n    .filter(([_key, values]) => values.length)\n    .map(([key, values]) => `${key}=${values.join(',')}`)\n    .join('&');\n\nexport const HE_MAP_LOADED = new InjectionToken<Observable<boolean>>('Google Maps API Loaded');\n\n/**\n * Usage:\n * ```\n * // import HttpClientJsonpModule\n * imports: [HttpClientJsonpModule]\n *\n * // Inject in module\n * providers: [{ provide: HE_MAP_LOADED, useFactory: loadMapApi(apiKey), deps: [HttpClient]} ]\n *\n * // Inject in component\n * constructor(@Inject(HE_MAP_LOADED) public mapLoaded$: Observable<boolean>) { }\n * ```\n * @param apiKey\n * @param libraries\n * @returns\n */\nexport const loadMapApi =\n  (apiKey: string, libraries = ['visualization', 'drawing']) =>\n  (httpClient: HttpClient) =>\n    of(`${loadMapUrl}?${buildMapsUrl({ key: [apiKey], libraries })}`).pipe(\n      mergeMap(url => httpClient.jsonp(url, 'callback')),\n      shareReplay({ bufferSize: 1, refCount: true }),\n      map(() => true),\n      catchError(e => {\n        console.error(e);\n        return of(false);\n      })\n    );\n\nexport const locationQuery = (bounds: google.maps.LatLngBounds, searchField = 'location') => {\n  const topRight = bounds.getNorthEast();\n  const bottomLeft = bounds.getSouthWest();\n  return {\n    geo_bounding_box: {\n      [searchField]: {\n        top_left: { lat: topRight.lat(), lon: bottomLeft.lng() },\n        bottom_right: { lat: bottomLeft.lat(), lon: topRight.lng() }\n      }\n    }\n  };\n};\n\nexport const fillColor = '#193957';\nexport const strokeColor = '#0C1B29';\n\nexport const fillStyle = {\n  fillColor,\n  fillOpacity: 0.6\n};\n\nexport const strokeStyle = {\n  strokeColor,\n  strokeOpacity: 0.6,\n  strokeWeight: 1\n};\n\nexport const clustererImage = 'https://developers.google.com/maps/documentation/javascript/examples/markerclusterer/m';\n\nconst getCoordinatesForPercent = (percent: number) => {\n  const x = Math.cos(2 * Math.PI * percent);\n  const y = Math.sin(2 * Math.PI * percent);\n  return [x, y];\n};\n\ninterface IMarkerPieConfig {\n  percent: number;\n  color: string;\n}\n\nconst pieUrl = (configs: IMarkerPieConfig[], size: number) => {\n  let cumulativePercent = 0;\n\n  return (\n    'data:image/svg+xml;utf-8,' +\n    encodeURIComponent(\n      `\n    <svg width=\"${size}\" height=\"${size}\" viewBox=\"-1 -1 2 2\" xmlns=\"http://www.w3.org/2000/svg\">${configs\n      .map(({ percent, color }) => {\n        const [startX, startY] = getCoordinatesForPercent(cumulativePercent);\n        cumulativePercent += percent;\n        const [endX, endY] = getCoordinatesForPercent(cumulativePercent);\n        const largeArcFlag = percent > 0.5 ? 1 : 0;\n        const pathData = [\n          `M ${startX} ${startY}`, // Move\n          `A 1 1 0 ${largeArcFlag} 1 ${endX} ${endY}`, // Arc\n          `L 0 0` // Line\n        ].join(' ');\n        return `<path fill=\"${color}\" d=\"${pathData}\"></path>`;\n      })\n      .join('')}</svg>\n  `.trim()\n    )\n  );\n};\n\nexport const markerPie = (configs: IMarkerPieConfig[], size = 7) => ({\n  anchor: new google.maps.Point(size / 2, size / 2),\n  url: pieUrl(configs, size),\n  labelOrigin: new google.maps.Point(0, -2)\n});\n\nexport const markerIcon = (scale = 7): google.maps.Symbol => ({\n  path: google.maps.SymbolPath.CIRCLE,\n  ...fillStyle,\n  ...strokeStyle,\n  scale,\n  labelOrigin: new google.maps.Point(0, -2)\n});\n\nexport const createMarker = (\n  position?: google.maps.LatLng | google.maps.LatLngLiteral,\n  text = '',\n  color = fillColor,\n  scale?: number\n) =>\n  (position && position.lat && position.lng\n    ? new google.maps.Marker({\n        position,\n        label: { color, text },\n        icon: markerIcon(scale)\n      })\n    : undefined) as google.maps.Marker;\n\ninterface IPolygon {\n  type: 'Polygon';\n  coordinates: number[][][];\n}\n\ninterface IMultiPolygon {\n  type: 'MultiPolygon';\n  coordinates: number[][][];\n}\n\ntype geometry = IPolygon | IMultiPolygon;\n\nexport interface IFeature {\n  type: 'Feature';\n  properties: any;\n  geometry: geometry;\n}\n\nexport interface IFeatureCollection {\n  type: 'FeatureCollection';\n  features: IFeature[];\n}\n\nexport interface IGeometryCollection {\n  type: 'GeometryCollection';\n  geometries: geometry[];\n}\n\ntype feature = IFeature | IFeatureCollection | IGeometryCollection | IPolygon | IMultiPolygon;\n\nexport const defaultFeature = (): IFeature => ({\n  type: 'Feature',\n  properties: {},\n  geometry: {\n    type: 'Polygon',\n    coordinates: []\n  } as geometry\n});\n\nexport const pointToCoordinates = (point: google.maps.LatLng) => [point.lng(), point.lat()];\n\nexport const coordinatesToPoint = (values: (number | number[])[]) => {\n  try {\n    return Array.isArray(values[0])\n      ? values.map((v: any) => coordinatesToPoint(v))\n      : (values as number[])?.some(isNaN)\n      ? null\n      : new google.maps.LatLng((values as number[])[1], (values as number[])[0]);\n  } catch (err) {\n    return null;\n  }\n};\n\nexport const polygonToCoordinates = (polygon: google.maps.Polygon) =>\n  polygon\n    .getPaths()\n    .getArray()\n    .map(path => {\n      const points = path.getArray();\n      return [...points.map(pointToCoordinates), pointToCoordinates(points[0])];\n    })[0];\n\nconst coordinatesToPolygon = (coordinates: number[][], color = strokeColor) =>\n  new google.maps.Polygon({\n    paths: coordinates.map(coordinatesToPoint).filter(Boolean),\n    ...strokeStyle,\n    strokeColor: color\n  });\n\nconst getCoordinates = (feat: feature) =>\n  feat.type === 'Polygon' || feat.type === 'MultiPolygon'\n    ? feat.coordinates\n    : feat.type === 'Feature'\n    ? feat.geometry?.coordinates\n    : [];\n\nexport const addPolygonToFeature = (feat: feature, polygon: google.maps.Polygon) =>\n  feat.type === 'FeatureCollection'\n    ? feat.features.push(addPolygonToFeature(defaultFeature(), polygon))\n    : feat.type === 'GeometryCollection'\n    ? feat.geometries.map(geometry => (geometry?.coordinates || []).push(polygonToCoordinates(polygon)))\n    : (getCoordinates(feat) || []).push(polygonToCoordinates(polygon));\n\nexport const polygonsFromFeature = (feat: feature, color?: string): google.maps.Polygon[] =>\n  feat.type === 'FeatureCollection'\n    ? feat.features.flatMap(values => polygonsFromFeature(values, color))\n    : feat.type === 'GeometryCollection'\n    ? feat.geometries?.flatMap(geometry =>\n        (geometry.coordinates || []).map(values => coordinatesToPolygon(values, color))\n      )\n    : (getCoordinates(feat) || []).map(values => coordinatesToPolygon(values, color));\n\nexport const polygonBounds = (polygon?: google.maps.Polygon | google.maps.Polygon[]) => {\n  const paths = polygon\n    ? Array.isArray(polygon)\n      ? polygon.flatMap(p => p.getPath()?.getArray() || [])\n      : polygon.getPath()?.getArray() || []\n    : [];\n  return paths.reduce((prev, curr) => prev.extend(curr), new google.maps.LatLngBounds());\n};\n"]}
161
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"maps-utils.js","sourceRoot":"","sources":["../../../src/common/maps-utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpClient } from '@angular/common/http';\nimport { InjectionToken } from '@angular/core';\nimport { Observable, of } from 'rxjs';\nimport { catchError, delay, map, mergeMap, shareReplay } from 'rxjs/operators';\n\nconst loadMapUrl = 'https://maps.googleapis.com/maps/api/js';\nconst buildMapsUrl = (params: Record<string, string[]>) =>\n  Object.entries(params)\n    .filter(([_key, values]) => values.length)\n    .map(([key, values]) => `${key}=${values.join(',')}`)\n    .join('&');\n\nexport const HE_MAP_LOADED = new InjectionToken<Observable<boolean>>('Google Maps API Loaded');\n\n/**\n * Usage:\n * ```\n * // import HttpClientJsonpModule\n * imports: [HttpClientJsonpModule]\n *\n * // Inject in module\n * providers: [{ provide: HE_MAP_LOADED, useFactory: loadMapApi(apiKey), deps: [HttpClient]} ]\n *\n * // Inject in component\n * constructor(@Inject(HE_MAP_LOADED) public mapLoaded$: Observable<boolean>) { }\n * ```\n * @param apiKey\n * @param libraries\n * @returns\n */\nexport const loadMapApi =\n  (apiKey: string, libraries = ['visualization', 'drawing']) =>\n  (httpClient: HttpClient) =>\n    of(`${loadMapUrl}?${buildMapsUrl({ key: [apiKey], libraries })}`).pipe(\n      mergeMap(url => httpClient.jsonp(url, 'callback')),\n      shareReplay({ bufferSize: 1, refCount: true }),\n      delay(100),\n      map(() => true),\n      catchError(e => {\n        console.error(e);\n        return of(false);\n      })\n    );\n\nexport const locationQuery = (bounds: google.maps.LatLngBounds, searchField = 'location') => {\n  const topRight = bounds.getNorthEast();\n  const bottomLeft = bounds.getSouthWest();\n  return {\n    geo_bounding_box: {\n      [searchField]: {\n        top_left: { lat: topRight.lat(), lon: bottomLeft.lng() },\n        bottom_right: { lat: bottomLeft.lat(), lon: topRight.lng() }\n      }\n    }\n  };\n};\n\nexport const fillColor = '#193957';\nexport const strokeColor = '#0C1B29';\n\nexport const fillStyle = {\n  fillColor,\n  fillOpacity: 0.6\n};\n\nexport const strokeStyle = {\n  strokeColor,\n  strokeOpacity: 0.6,\n  strokeWeight: 1\n};\n\nexport const clustererImage = 'https://developers.google.com/maps/documentation/javascript/examples/markerclusterer/m';\n\nconst getCoordinatesForPercent = (percent: number) => {\n  const x = Math.cos(2 * Math.PI * percent);\n  const y = Math.sin(2 * Math.PI * percent);\n  return [x, y];\n};\n\ninterface IMarkerPieConfig {\n  percent: number;\n  color: string;\n}\n\nconst pieUrl = (configs: IMarkerPieConfig[], size: number) => {\n  let cumulativePercent = 0;\n\n  return (\n    'data:image/svg+xml;utf-8,' +\n    encodeURIComponent(\n      `\n    <svg width=\"${size}\" height=\"${size}\" viewBox=\"-1 -1 2 2\" xmlns=\"http://www.w3.org/2000/svg\">${configs\n      .map(({ percent, color }) => {\n        const [startX, startY] = getCoordinatesForPercent(cumulativePercent);\n        cumulativePercent += percent;\n        const [endX, endY] = getCoordinatesForPercent(cumulativePercent);\n        const largeArcFlag = percent > 0.5 ? 1 : 0;\n        const pathData = [\n          `M ${startX} ${startY}`, // Move\n          `A 1 1 0 ${largeArcFlag} 1 ${endX} ${endY}`, // Arc\n          `L 0 0` // Line\n        ].join(' ');\n        return `<path fill=\"${color}\" d=\"${pathData}\"></path>`;\n      })\n      .join('')}</svg>\n  `.trim()\n    )\n  );\n};\n\nexport const markerPie = (configs: IMarkerPieConfig[], size = 7) => ({\n  anchor: new google.maps.Point(size / 2, size / 2),\n  url: pieUrl(configs, size),\n  labelOrigin: new google.maps.Point(0, -2)\n});\n\nexport const markerIcon = (scale = 7): google.maps.Symbol => ({\n  path: google.maps.SymbolPath.CIRCLE,\n  ...fillStyle,\n  ...strokeStyle,\n  scale,\n  labelOrigin: new google.maps.Point(0, -2)\n});\n\nexport interface IMarker {\n  position: google.maps.LatLng | google.maps.LatLngLiteral;\n  label: google.maps.MarkerLabel;\n  icon: google.maps.Symbol;\n}\n\nexport const createMarker = (\n  position?: google.maps.LatLng | google.maps.LatLngLiteral,\n  text = '',\n  color = fillColor,\n  scale?: number\n): IMarker =>\n  position?.lat && position?.lng\n    ? {\n        position,\n        label: { color, text },\n        icon: markerIcon(scale)\n      }\n    : undefined;\n\ninterface IPolygon {\n  type: 'Polygon';\n  coordinates: number[][][];\n}\n\ninterface IMultiPolygon {\n  type: 'MultiPolygon';\n  coordinates: number[][][];\n}\n\ntype geometry = IPolygon | IMultiPolygon;\n\nexport interface IFeature {\n  type: 'Feature';\n  properties: any;\n  geometry: geometry;\n}\n\nexport interface IFeatureCollection {\n  type: 'FeatureCollection';\n  features: IFeature[];\n}\n\nexport interface IGeometryCollection {\n  type: 'GeometryCollection';\n  geometries: geometry[];\n}\n\nexport type feature = IFeature | IFeatureCollection | IGeometryCollection | IPolygon | IMultiPolygon;\n\nexport const defaultFeature = (): IFeature => ({\n  type: 'Feature',\n  properties: {},\n  geometry: {\n    type: 'Polygon',\n    coordinates: []\n  } as geometry\n});\n\nexport const pointToCoordinates = (point: google.maps.LatLng) => [point.lng(), point.lat()];\n\nexport const coordinatesToPoint = (values: (number | number[])[]) => {\n  try {\n    return Array.isArray(values[0])\n      ? values.map((v: any) => coordinatesToPoint(v))\n      : (values as number[])?.some(isNaN)\n      ? null\n      : new google.maps.LatLng((values as number[])[1], (values as number[])[0]);\n  } catch (err) {\n    return null;\n  }\n};\n\nexport const polygonToCoordinates = (polygon: google.maps.Polygon) =>\n  polygon\n    .getPaths()\n    .getArray()\n    .map(path => {\n      const points = path.getArray();\n      return [...points.map(pointToCoordinates), pointToCoordinates(points[0])];\n    })[0];\n\nexport interface IPolygonMap {\n  paths: google.maps.MVCArray<google.maps.MVCArray<google.maps.LatLng>>;\n  options: google.maps.PolygonOptions;\n}\n\nexport const polygonToMap = (polygon: google.maps.Polygon, color = strokeColor): IPolygonMap => ({\n  paths: polygon.getPaths(),\n  options: {\n    ...strokeStyle,\n    strokeColor: color\n  }\n});\n\nconst coordinatesToPolygon = (coordinates: number[][], color = strokeColor) =>\n  new google.maps.Polygon({\n    paths: coordinates.map(coordinatesToPoint).filter(Boolean),\n    ...strokeStyle,\n    strokeColor: color\n  });\n\nconst getCoordinates = (feat: feature) =>\n  feat.type === 'Polygon' || feat.type === 'MultiPolygon'\n    ? feat.coordinates\n    : feat.type === 'Feature'\n    ? feat.geometry?.coordinates\n    : [];\n\nexport const addPolygonToFeature = (feat: feature, polygon: google.maps.Polygon) =>\n  feat.type === 'FeatureCollection'\n    ? feat.features.push(addPolygonToFeature(defaultFeature(), polygon))\n    : feat.type === 'GeometryCollection'\n    ? feat.geometries.map(geometry => (geometry?.coordinates || []).push(polygonToCoordinates(polygon)))\n    : (getCoordinates(feat) || []).push(polygonToCoordinates(polygon));\n\nexport const polygonsFromFeature = (feat: feature, color?: string): google.maps.Polygon[] =>\n  feat.type === 'FeatureCollection'\n    ? feat.features.flatMap(values => polygonsFromFeature(values, color))\n    : feat.type === 'GeometryCollection'\n    ? feat.geometries?.flatMap(geometry =>\n        (geometry.coordinates || []).map(values => coordinatesToPolygon(values, color))\n      )\n    : (getCoordinates(feat) || []).map(values => coordinatesToPolygon(values, color));\n\nexport const polygonBounds = (polygon?: google.maps.Polygon | google.maps.Polygon[]) => {\n  const paths = polygon\n    ? Array.isArray(polygon)\n      ? polygon.flatMap(p => p.getPath()?.getArray() || [])\n      : polygon.getPath()?.getArray() || []\n    : [];\n  return paths.reduce((prev, curr) => prev.extend(curr), new google.maps.LatLngBounds());\n};\n"]}
@@ -129,11 +129,11 @@ export class ImpactAssessmentsProductsComponent {
129
129
  this.selectedIndex.set(+value);
130
130
  }
131
131
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.8", ngImport: i0, type: ImpactAssessmentsProductsComponent, deps: [{ token: i1.HeNodeStoreService }], target: i0.ɵɵFactoryTarget.Component }); }
132
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.8", type: ImpactAssessmentsProductsComponent, selector: "he-impact-assessments-products", inputs: { dataState: "dataState", nodeKey: "nodeKey", filterTermTypes: "filterTermTypes", enableFilterMethodModel: "enableFilterMethodModel" }, ngImport: i0, template: "<div class=\"columns is-variable is-align-items-center is-2 m-0\">\n <div class=\"column\">\n <ng-container *ngIf=\"selectedView === View.table && indicators().length\">\n <button\n class=\"button is-small is-ghost\"\n (click)=\"showDownload = true\"\n ngbTooltip=\"Download as CSV\"\n placement=\"bottom\">\n <fa-icon icon=\"download\" size=\"lg\"></fa-icon>\n </button>\n </ng-container>\n </div>\n <div class=\"column is-narrow\" *ngIf=\"selectedView === View.table\">\n <he-search-extend\n class=\"is-secondary is-small\"\n placeholder=\"Filter indicators by name\"\n (search)=\"filterByTerm($event)\"></he-search-extend>\n </div>\n <div class=\"column is-narrow\" *ngIf=\"impactAssessments().length > 1 || enableBreakdown() || !isOriginal()\">\n <div class=\"field has-addons\">\n <div class=\"control\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView === View.table\"\n (click)=\"selectedView = View.table\">\n <span class=\"icon is-small\">\n <fa-icon icon=\"list\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span class=\"is-hidden-mobile\">Table view</span>\n </button>\n </div>\n <div class=\"control\" *ngIf=\"impactAssessments().length > 1\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView === View.chart\"\n (click)=\"selectedView = View.chart\">\n <span class=\"icon is-small\">\n <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span class=\"is-hidden-mobile\">Chart view</span>\n </button>\n </div>\n <div class=\"control\" *ngIf=\"enableBreakdown()\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView === View.breakdown\"\n (click)=\"selectedView = View.breakdown\">\n <span class=\"icon is-small\">\n <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span class=\"is-hidden-mobile\">Breakdown view</span>\n </button>\n </div>\n <div class=\"control\" *ngIf=\"!isOriginal()\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView === View.logs\"\n (click)=\"selectedView = View.logs\">\n <span class=\"icon is-small\">\n <fa-icon icon=\"calculator\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span class=\"is-hidden-mobile\">Recalculations logs</span>\n </button>\n </div>\n </div>\n </div>\n</div>\n\n<div class=\"px-3 pb-3\" [class.is-hidden]=\"selectedView !== View.table\">\n <ng-container *ngIf=\"hasData(); else emptyTable\">\n <he-data-table class=\"mb-1 is-small\" [small]=\"true\" maxHeight=\"320\">\n <table class=\"table is-fullwidth is-narrow is-striped\">\n <thead>\n <tr class=\"has-text-weight-semibold\">\n <th class=\"width-auto py-0\">\n <div class=\"select is-small\" *ngIf=\"enableFilterMethodModel\">\n <select (change)=\"updateSelectedMethodModel($event)\">\n <option [ngValue]=\"undefined\">Filter Model</option>\n <option *ngFor=\"let term of methodModels()\" [ngValue]=\"term\">{{ term.name }}</option>\n </select>\n </div>\n </th>\n <th class=\"has-border-right\"></th>\n <th class=\"has-border-right\"></th>\n <th *ngFor=\"let indicator of indicators()\" [attr.title]=\"indicator.value.term.name\">\n <he-node-link [node]=\"indicator.value.term\">\n <span [innerHtml]=\"indicator.value.term.name | ellipsis: 30 | compound\"></span>\n </he-node-link>\n </th>\n </tr>\n <tr class=\"is-italic has-text-weight-semibold\">\n <th class=\"width-auto has-border-right\"></th>\n <th class=\"has-border-right\">\n <a [href]=\"baseUrl + '/schema/ImpactAssessment#functionalUnit'\" target=\"_blank\">Functional unit</a>\n </th>\n <th class=\"has-border-right\">\n <a [href]=\"baseUrl + '/schema/ImpactAssessment#product'\" target=\"_blank\">Product</a>\n </th>\n <th *ngFor=\"let indicator of indicators()\" [attr.title]=\"indicator.value.term.units\">\n <span [innerHtml]=\"indicator.value.term.units | compound\"></span>\n <he-terms-units-description\n class=\"is-inline-block is-ml-2\"\n [term]=\"indicator.value.term\"></he-terms-units-description>\n </th>\n </tr>\n </thead>\n <tbody>\n <ng-container *ngFor=\"let impactAssessment of impactAssessments(); trackBy: trackById; let i = index\">\n <tr>\n <td class=\"width-auto has-border-right\" [attr.title]=\"impactName(impactAssessment)\">\n <he-node-link [node]=\"impactAssessment\">\n <span>{{ i + 1 }}. {{ impactName(impactAssessment) }}</span>\n </he-node-link>\n </td>\n <td class=\"has-border-right\" [attr.title]=\"impactAssessment.product?.term?.units\">\n <span>1 {{ impactAssessment.product?.term?.units }}</span>\n </td>\n <td class=\"has-border-right\" [attr.title]=\"impactAssessment.product?.term?.name\">\n <he-node-link *ngIf=\"impactAssessment.product?.term\" [node]=\"impactAssessment.product.term\">\n <span>{{ impactAssessment.product.term.name | ellipsis: 30 }}</span>\n </he-node-link>\n </td>\n <td class=\"is-nowrap\" *ngFor=\"let indicator of indicators()\">\n <span\n *ngIf=\"indicator.value.values[impactAssessment['@id']]; else emptyValue\"\n class=\"trigger-popover\"\n [ngbPopover]=\"details\"\n [autoClose]=\"'outside'\"\n popoverClass=\"is-narrow\"\n triggers=\"manual\"\n #p=\"ngbPopover\"\n placement=\"left\"\n container=\"body\"\n (click)=\"\n togglePopover(p, {\n data: indicator.value.values[impactAssessment['@id']],\n impactAssessment: impactAssessment\n })\n \">\n <span pointer>\n {{\n propertyValue(indicator.value.values[impactAssessment['@id']].value, indicator.value.term['@id'])\n | precision: 3\n | default: '-'\n }}\n </span>\n <he-blank-node-state\n class=\"ml-1\"\n [dataState]=\"_dataState()\"\n [node]=\"indicator.value.values[impactAssessment['@id']].node\"\n key=\"value\"\n [state]=\"impactAssessment.aggregated ? 'aggregated' : undefined\"></he-blank-node-state>\n </span>\n </td>\n </tr>\n </ng-container>\n </tbody>\n </table>\n </he-data-table>\n\n <he-blank-node-state-notice [dataState]=\"_dataState()\"></he-blank-node-state-notice>\n </ng-container>\n</div>\n\n<he-impact-assessments-indicator-breakdown-chart\n *ngIf=\"selectedView === View.breakdown\"\n [impactAssessment]=\"impactAssessments()[0]\"\n [indicators]=\"impactAssessments()[0][_nodeKey()]\"></he-impact-assessments-indicator-breakdown-chart>\n\n<he-impact-assessments-indicators-chart\n *ngIf=\"selectedView === View.chart && impactAssessments().length > 1\"\n [key]=\"_nodeKey()\"\n [filterTermTypes]=\"_filterTermTypes()\"></he-impact-assessments-indicators-chart>\n\n<ng-container *ngIf=\"selectedView === View.logs && !isOriginal()\">\n <div class=\"field has-addons pt-2 px-3\" *ngIf=\"impactAssessments().length > 1\">\n <div class=\"control\">\n <span class=\"button is-small is-static is-secondary\">Select an Impact Assessment</span>\n </div>\n <div class=\"control is-expanded\">\n <div class=\"select is-small is-fullwidth is-secondary\">\n <select (change)=\"selectIndex($event)\">\n <option *ngFor=\"let value of impactAssessments(); let i = index\" [value]=\"i\">\n {{ i + 1 }}. {{ impactName(value) }}\n </option>\n </select>\n </div>\n </div>\n </div>\n\n <he-node-logs-models\n *ngIf=\"selectedNode()\"\n [node]=\"selectedNode()\"\n [nodeKey]=\"_nodeKey()\"\n [filterTermTypes]=\"_filterTermTypes()\"\n [originalValues]=\"selectedOriginalValues()\"\n [recalculatedValues]=\"selectedRecalculatedValues()\"\n [noDataMessage]=\"noDataMessage()\"></he-node-logs-models>\n</ng-container>\n\n<he-node-csv-export-confirm\n *ngIf=\"showDownload\"\n [nodes]=\"impactAssessments()\"\n [filename]=\"'impact-' + _nodeKey() + '.csv'\"\n [isUpload]=\"false\"\n [headerKeys]=\"['impactAssessment.id', 'impactAssessment.@id', 'impactAssessment.' + _nodeKey() + '.']\"\n (closed)=\"showDownload = false\"></he-node-csv-export-confirm>\n\n<ng-template #emptyTable>\n <div class=\"has-text-centered\">\n <span>No data available</span>\n <span class=\"is-pl-1\" [class.is-hidden]=\"!filterTerm()\">matching your search criteria</span>\n <span>.</span>\n <span [class.is-hidden]=\"!isOriginal()\">\n Switch to\n <code>recalculated</code>\n version.\n </span>\n </div>\n</ng-template>\n\n<ng-template #emptyValue>\n <span>-</span>\n</ng-template>\n\n<ng-template #details let-node=\"impactAssessment\" let-data=\"data\">\n <p *bindOnce=\"node\">\n <b>\n <span *ngIf=\"data.cycle\">{{ cycleLabel(node.cycle) }}</span>\n <span *ngIf=\"!data.cycle\">{{ data.name }}</span>\n </b>\n </p>\n <he-node-value-details\n [data]=\"data\"\n [dataState]=\"_dataState()\"\n [nodeType]=\"node['@type']\"\n [dataKey]=\"_nodeKey()\"></he-node-value-details>\n</ng-template>\n\n<ng-template #suggestion let-impact=\"result\" let-t=\"term\">\n <div class=\"is-block\">\n <ngb-highlight [result]=\"impact.name || impact.cycle.name\" [term]=\"t\"></ngb-highlight>\n </div>\n <div class=\"columns is-flex\">\n <div class=\"column\" *ngIf=\"impact.country\">\n <span class=\"pr-1 has-text-underline\">Country:</span>\n <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.country.name\" [term]=\"t\"></ngb-highlight></span>\n </div>\n <div class=\"column\" *ngIf=\"impact.product\">\n <span class=\"pr-1 has-text-underline\">Product:</span>\n <span class=\"is-inline-flex\">\n <ngb-highlight [result]=\"impact.product.term?.name || impact.product.name\" [term]=\"t\"></ngb-highlight>\n </span>\n </div>\n <div class=\"column\" *ngIf=\"impact.endDate\">\n <span class=\"pr-1 has-text-underline\">Date:</span>\n <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.endDate\" [term]=\"t\"></ngb-highlight></span>\n </div>\n </div>\n</ng-template>\n", styles: ["fa-icon{display:inline-block;width:10px}he-data-table ::ng-deep .table thead tr th:nth-child(2),he-data-table ::ng-deep .table tbody tr td:nth-child(2){max-width:102px;width:102px}he-data-table ::ng-deep .table thead tr th:nth-child(3),he-data-table ::ng-deep .table tbody tr td:nth-child(3){min-width:110px}\n"], dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i3.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "component", type: i4.NgbHighlight, selector: "ngb-highlight", inputs: ["highlightClass", "result", "term", "accentSensitive"] }, { kind: "directive", type: i4.NgbTooltip, selector: "[ngbTooltip]", inputs: ["animation", "autoClose", "placement", "popperOptions", "triggers", "positionTarget", "container", "disableTooltip", "tooltipClass", "tooltipContext", "openDelay", "closeDelay", "ngbTooltip"], outputs: ["shown", "hidden"], exportAs: ["ngbTooltip"] }, { kind: "directive", type: i4.NgbPopover, selector: "[ngbPopover]", inputs: ["animation", "autoClose", "ngbPopover", "popoverTitle", "placement", "popperOptions", "triggers", "positionTarget", "container", "disablePopover", "popoverClass", "popoverContext", "openDelay", "closeDelay"], outputs: ["shown", "hidden"], exportAs: ["ngbPopover"] }, { kind: "component", type: i5.FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "spin", "pulse", "mask", "styles", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "classes", "transform", "a11yRole"] }, { kind: "directive", type: i6.BindOnceDirective, selector: "[bindOnce]", inputs: ["bindOnce"] }, { kind: "component", type: i7.BlankNodeStateComponent, selector: "he-blank-node-state", inputs: ["dataState", "nodeType", "dataKey", "key", "node", "state", "linkClass"] }, { kind: "component", type: i8.BlankNodeStateNoticeComponent, selector: "he-blank-node-state-notice", inputs: ["dataState", "showAggregated", "showDeleted"] }, { kind: "component", type: i9.DataTableComponent, selector: "he-data-table", inputs: ["minHeight", "maxHeight", "small"] }, { kind: "component", type: i10.SearchExtendComponent, selector: "he-search-extend", inputs: ["value", "disabled", "placeholder", "class"], outputs: ["search"] }, { kind: "component", type: i11.NodeLinkComponent, selector: "he-node-link", inputs: ["node", "showExternalLink", "linkClass"] }, { kind: "component", type: i12.NodeCsvExportConfirmComponent, selector: "he-node-csv-export-confirm", inputs: ["nodes", "filename", "headerKeys", "extension", "isUpload"], outputs: ["closed"] }, { kind: "component", type: i13.NodeLogsModelsComponent, selector: "he-node-logs-models", inputs: ["node", "nodeKey", "originalValues", "recalculatedValues", "terms", "filterTermTypes", "filterTermTypesLabel", "logsKey", "noDataMessage"] }, { kind: "component", type: i14.NodeValueDetailsComponent, selector: "he-node-value-details", inputs: ["data", "nodeType", "dataKey", "dataState"] }, { kind: "component", type: i15.TermsUnitsDescriptionComponent, selector: "he-terms-units-description", inputs: ["term", "iconTemplate"] }, { kind: "component", type: i16.ImpactAssessmentsIndicatorsChartComponent, selector: "he-impact-assessments-indicators-chart", inputs: ["key", "filterTermTypes"] }, { kind: "component", type: i17.ImpactAssessmentsIndicatorBreakdownChartComponent, selector: "he-impact-assessments-indicator-breakdown-chart", inputs: ["impactAssessment", "indicators"] }, { kind: "pipe", type: i18.CompoundPipe, name: "compound" }, { kind: "pipe", type: i19.DefaultPipe, name: "default" }, { kind: "pipe", type: i20.EllipsisPipe, name: "ellipsis" }, { kind: "pipe", type: i21.PrecisionPipe, name: "precision" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
132
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.8", type: ImpactAssessmentsProductsComponent, selector: "he-impact-assessments-products", inputs: { dataState: "dataState", nodeKey: "nodeKey", filterTermTypes: "filterTermTypes", enableFilterMethodModel: "enableFilterMethodModel" }, ngImport: i0, template: "<div class=\"columns is-variable is-align-items-center is-2 is-m-0\" *ngIf=\"indicators().length\">\n <div class=\"column\">\n <ng-container *ngIf=\"selectedView === View.table\">\n <button\n class=\"button is-small is-ghost\"\n (click)=\"showDownload = true\"\n ngbTooltip=\"Download as CSV\"\n placement=\"bottom\">\n <fa-icon icon=\"download\" size=\"lg\"></fa-icon>\n </button>\n </ng-container>\n </div>\n <div class=\"column is-narrow\" *ngIf=\"selectedView === View.table\">\n <he-search-extend\n class=\"is-secondary is-small\"\n placeholder=\"Filter indicators by name\"\n (search)=\"filterByTerm($event)\"></he-search-extend>\n </div>\n <div class=\"column is-narrow\" *ngIf=\"impactAssessments().length > 1 || enableBreakdown() || !isOriginal()\">\n <div class=\"field has-addons\">\n <div class=\"control\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView === View.table\"\n (click)=\"selectedView = View.table\">\n <span class=\"icon is-small\">\n <fa-icon icon=\"list\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span class=\"is-hidden-mobile\">Table view</span>\n </button>\n </div>\n <div class=\"control\" *ngIf=\"impactAssessments().length > 1\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView === View.chart\"\n (click)=\"selectedView = View.chart\">\n <span class=\"icon is-small\">\n <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span class=\"is-hidden-mobile\">Chart view</span>\n </button>\n </div>\n <div class=\"control\" *ngIf=\"enableBreakdown()\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView === View.breakdown\"\n (click)=\"selectedView = View.breakdown\">\n <span class=\"icon is-small\">\n <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span class=\"is-hidden-mobile\">Breakdown view</span>\n </button>\n </div>\n <div class=\"control\" *ngIf=\"!isOriginal()\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView === View.logs\"\n (click)=\"selectedView = View.logs\">\n <span class=\"icon is-small\">\n <fa-icon icon=\"calculator\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span class=\"is-hidden-mobile\">Recalculations logs</span>\n </button>\n </div>\n </div>\n </div>\n</div>\n\n<div class=\"px-3 pb-3\" [class.is-hidden]=\"selectedView !== View.table\">\n <ng-container *ngIf=\"hasData(); else emptyTable\">\n <he-data-table class=\"mb-1 is-small\" [small]=\"true\" maxHeight=\"320\">\n <table class=\"table is-fullwidth is-narrow is-striped\">\n <thead>\n <tr class=\"has-text-weight-semibold\">\n <th class=\"width-auto py-0\">\n <div class=\"select is-small\" *ngIf=\"enableFilterMethodModel\">\n <select (change)=\"updateSelectedMethodModel($event)\">\n <option [ngValue]=\"undefined\">Filter Model</option>\n <option *ngFor=\"let term of methodModels()\" [ngValue]=\"term\">{{ term.name }}</option>\n </select>\n </div>\n </th>\n <th class=\"has-border-right\"></th>\n <th class=\"has-border-right\"></th>\n <th *ngFor=\"let indicator of indicators()\" [attr.title]=\"indicator.value.term.name\">\n <he-node-link [node]=\"indicator.value.term\">\n <span [innerHtml]=\"indicator.value.term.name | ellipsis: 30 | compound\"></span>\n </he-node-link>\n </th>\n </tr>\n <tr class=\"is-italic has-text-weight-semibold\">\n <th class=\"width-auto has-border-right\"></th>\n <th class=\"has-border-right\">\n <a [href]=\"baseUrl + '/schema/ImpactAssessment#functionalUnit'\" target=\"_blank\">Functional unit</a>\n </th>\n <th class=\"has-border-right\">\n <a [href]=\"baseUrl + '/schema/ImpactAssessment#product'\" target=\"_blank\">Product</a>\n </th>\n <th *ngFor=\"let indicator of indicators()\" [attr.title]=\"indicator.value.term.units\">\n <span [innerHtml]=\"indicator.value.term.units | compound\"></span>\n <he-terms-units-description\n class=\"is-inline-block is-ml-2\"\n [term]=\"indicator.value.term\"></he-terms-units-description>\n </th>\n </tr>\n </thead>\n <tbody>\n <ng-container *ngFor=\"let impactAssessment of impactAssessments(); trackBy: trackById; let i = index\">\n <tr>\n <td class=\"width-auto has-border-right\" [attr.title]=\"impactName(impactAssessment)\">\n <he-node-link [node]=\"impactAssessment\">\n <span>{{ i + 1 }}. {{ impactName(impactAssessment) }}</span>\n </he-node-link>\n </td>\n <td class=\"has-border-right\" [attr.title]=\"impactAssessment.product?.term?.units\">\n <span>1 {{ impactAssessment.product?.term?.units }}</span>\n </td>\n <td class=\"has-border-right\" [attr.title]=\"impactAssessment.product?.term?.name\">\n <he-node-link *ngIf=\"impactAssessment.product?.term\" [node]=\"impactAssessment.product.term\">\n <span>{{ impactAssessment.product.term.name | ellipsis: 30 }}</span>\n </he-node-link>\n </td>\n <td class=\"is-nowrap\" *ngFor=\"let indicator of indicators()\">\n <span\n *ngIf=\"indicator.value.values[impactAssessment['@id']]; else emptyValue\"\n class=\"trigger-popover\"\n [ngbPopover]=\"details\"\n [autoClose]=\"'outside'\"\n popoverClass=\"is-narrow\"\n triggers=\"manual\"\n #p=\"ngbPopover\"\n placement=\"left\"\n container=\"body\"\n (click)=\"\n togglePopover(p, {\n data: indicator.value.values[impactAssessment['@id']],\n impactAssessment: impactAssessment\n })\n \">\n <span pointer>\n {{\n propertyValue(indicator.value.values[impactAssessment['@id']].value, indicator.value.term['@id'])\n | precision: 3\n | default: '-'\n }}\n </span>\n <he-blank-node-state\n class=\"ml-1\"\n [dataState]=\"_dataState()\"\n [node]=\"indicator.value.values[impactAssessment['@id']].node\"\n key=\"value\"\n [state]=\"impactAssessment.aggregated ? 'aggregated' : undefined\"></he-blank-node-state>\n </span>\n </td>\n </tr>\n </ng-container>\n </tbody>\n </table>\n </he-data-table>\n\n <he-blank-node-state-notice [dataState]=\"_dataState()\"></he-blank-node-state-notice>\n </ng-container>\n</div>\n\n<he-impact-assessments-indicator-breakdown-chart\n *ngIf=\"selectedView === View.breakdown\"\n [impactAssessment]=\"impactAssessments()[0]\"\n [indicators]=\"impactAssessments()[0][_nodeKey()]\"></he-impact-assessments-indicator-breakdown-chart>\n\n<he-impact-assessments-indicators-chart\n *ngIf=\"selectedView === View.chart && impactAssessments().length > 1\"\n [key]=\"_nodeKey()\"\n [filterTermTypes]=\"_filterTermTypes()\"></he-impact-assessments-indicators-chart>\n\n<ng-container *ngIf=\"selectedView === View.logs && !isOriginal()\">\n <div class=\"field has-addons pt-2 px-3\" *ngIf=\"impactAssessments().length > 1\">\n <div class=\"control\">\n <span class=\"button is-small is-static is-secondary\">Select an Impact Assessment</span>\n </div>\n <div class=\"control is-expanded\">\n <div class=\"select is-small is-fullwidth is-secondary\">\n <select (change)=\"selectIndex($event)\">\n <option *ngFor=\"let value of impactAssessments(); let i = index\" [value]=\"i\">\n {{ i + 1 }}. {{ impactName(value) }}\n </option>\n </select>\n </div>\n </div>\n </div>\n\n <he-node-logs-models\n *ngIf=\"selectedNode()\"\n [node]=\"selectedNode()\"\n [nodeKey]=\"_nodeKey()\"\n [filterTermTypes]=\"_filterTermTypes()\"\n [originalValues]=\"selectedOriginalValues()\"\n [recalculatedValues]=\"selectedRecalculatedValues()\"\n [noDataMessage]=\"noDataMessage()\"></he-node-logs-models>\n</ng-container>\n\n<he-node-csv-export-confirm\n *ngIf=\"showDownload\"\n [nodes]=\"impactAssessments()\"\n [filename]=\"'impact-' + _nodeKey() + '.csv'\"\n [isUpload]=\"false\"\n [headerKeys]=\"['impactAssessment.id', 'impactAssessment.@id', 'impactAssessment.' + _nodeKey() + '.']\"\n (closed)=\"showDownload = false\"></he-node-csv-export-confirm>\n\n<ng-template #emptyTable>\n <div class=\"has-text-centered\">\n <span>No data available</span>\n <span class=\"is-pl-1\" [class.is-hidden]=\"!filterTerm()\">matching your search criteria</span>\n <span>.</span>\n <span [class.is-hidden]=\"!isOriginal()\">\n Switch to\n <code>recalculated</code>\n version.\n </span>\n </div>\n</ng-template>\n\n<ng-template #emptyValue>\n <span>-</span>\n</ng-template>\n\n<ng-template #details let-node=\"impactAssessment\" let-data=\"data\">\n <p *bindOnce=\"node\">\n <b>\n <span *ngIf=\"data.cycle\">{{ cycleLabel(node.cycle) }}</span>\n <span *ngIf=\"!data.cycle\">{{ data.name }}</span>\n </b>\n </p>\n <he-node-value-details\n [data]=\"data\"\n [dataState]=\"_dataState()\"\n [nodeType]=\"node['@type']\"\n [dataKey]=\"_nodeKey()\"></he-node-value-details>\n</ng-template>\n\n<ng-template #suggestion let-impact=\"result\" let-t=\"term\">\n <div class=\"is-block\">\n <ngb-highlight [result]=\"impact.name || impact.cycle.name\" [term]=\"t\"></ngb-highlight>\n </div>\n <div class=\"columns is-flex\">\n <div class=\"column\" *ngIf=\"impact.country\">\n <span class=\"pr-1 has-text-underline\">Country:</span>\n <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.country.name\" [term]=\"t\"></ngb-highlight></span>\n </div>\n <div class=\"column\" *ngIf=\"impact.product\">\n <span class=\"pr-1 has-text-underline\">Product:</span>\n <span class=\"is-inline-flex\">\n <ngb-highlight [result]=\"impact.product.term?.name || impact.product.name\" [term]=\"t\"></ngb-highlight>\n </span>\n </div>\n <div class=\"column\" *ngIf=\"impact.endDate\">\n <span class=\"pr-1 has-text-underline\">Date:</span>\n <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.endDate\" [term]=\"t\"></ngb-highlight></span>\n </div>\n </div>\n</ng-template>\n", styles: ["fa-icon{display:inline-block;width:10px}he-data-table ::ng-deep .table thead tr th:nth-child(2),he-data-table ::ng-deep .table tbody tr td:nth-child(2){max-width:102px;width:102px}he-data-table ::ng-deep .table thead tr th:nth-child(3),he-data-table ::ng-deep .table tbody tr td:nth-child(3){min-width:110px}\n"], dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i3.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "component", type: i4.NgbHighlight, selector: "ngb-highlight", inputs: ["highlightClass", "result", "term", "accentSensitive"] }, { kind: "directive", type: i4.NgbTooltip, selector: "[ngbTooltip]", inputs: ["animation", "autoClose", "placement", "popperOptions", "triggers", "positionTarget", "container", "disableTooltip", "tooltipClass", "tooltipContext", "openDelay", "closeDelay", "ngbTooltip"], outputs: ["shown", "hidden"], exportAs: ["ngbTooltip"] }, { kind: "directive", type: i4.NgbPopover, selector: "[ngbPopover]", inputs: ["animation", "autoClose", "ngbPopover", "popoverTitle", "placement", "popperOptions", "triggers", "positionTarget", "container", "disablePopover", "popoverClass", "popoverContext", "openDelay", "closeDelay"], outputs: ["shown", "hidden"], exportAs: ["ngbPopover"] }, { kind: "component", type: i5.FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "spin", "pulse", "mask", "styles", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "classes", "transform", "a11yRole"] }, { kind: "directive", type: i6.BindOnceDirective, selector: "[bindOnce]", inputs: ["bindOnce"] }, { kind: "component", type: i7.BlankNodeStateComponent, selector: "he-blank-node-state", inputs: ["dataState", "nodeType", "dataKey", "key", "node", "state", "linkClass"] }, { kind: "component", type: i8.BlankNodeStateNoticeComponent, selector: "he-blank-node-state-notice", inputs: ["dataState", "showAggregated", "showDeleted"] }, { kind: "component", type: i9.DataTableComponent, selector: "he-data-table", inputs: ["minHeight", "maxHeight", "small"] }, { kind: "component", type: i10.SearchExtendComponent, selector: "he-search-extend", inputs: ["value", "disabled", "placeholder", "class"], outputs: ["search"] }, { kind: "component", type: i11.NodeLinkComponent, selector: "he-node-link", inputs: ["node", "showExternalLink", "linkClass"] }, { kind: "component", type: i12.NodeCsvExportConfirmComponent, selector: "he-node-csv-export-confirm", inputs: ["nodes", "filename", "headerKeys", "extension", "isUpload"], outputs: ["closed"] }, { kind: "component", type: i13.NodeLogsModelsComponent, selector: "he-node-logs-models", inputs: ["node", "nodeKey", "originalValues", "recalculatedValues", "terms", "filterTermTypes", "filterTermTypesLabel", "logsKey", "noDataMessage"] }, { kind: "component", type: i14.NodeValueDetailsComponent, selector: "he-node-value-details", inputs: ["data", "nodeType", "dataKey", "dataState"] }, { kind: "component", type: i15.TermsUnitsDescriptionComponent, selector: "he-terms-units-description", inputs: ["term", "iconTemplate"] }, { kind: "component", type: i16.ImpactAssessmentsIndicatorsChartComponent, selector: "he-impact-assessments-indicators-chart", inputs: ["key", "filterTermTypes"] }, { kind: "component", type: i17.ImpactAssessmentsIndicatorBreakdownChartComponent, selector: "he-impact-assessments-indicator-breakdown-chart", inputs: ["impactAssessment", "indicators"] }, { kind: "pipe", type: i18.CompoundPipe, name: "compound" }, { kind: "pipe", type: i19.DefaultPipe, name: "default" }, { kind: "pipe", type: i20.EllipsisPipe, name: "ellipsis" }, { kind: "pipe", type: i21.PrecisionPipe, name: "precision" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
133
133
  }
134
134
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.8", ngImport: i0, type: ImpactAssessmentsProductsComponent, decorators: [{
135
135
  type: Component,
136
- args: [{ selector: 'he-impact-assessments-products', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"columns is-variable is-align-items-center is-2 m-0\">\n <div class=\"column\">\n <ng-container *ngIf=\"selectedView === View.table && indicators().length\">\n <button\n class=\"button is-small is-ghost\"\n (click)=\"showDownload = true\"\n ngbTooltip=\"Download as CSV\"\n placement=\"bottom\">\n <fa-icon icon=\"download\" size=\"lg\"></fa-icon>\n </button>\n </ng-container>\n </div>\n <div class=\"column is-narrow\" *ngIf=\"selectedView === View.table\">\n <he-search-extend\n class=\"is-secondary is-small\"\n placeholder=\"Filter indicators by name\"\n (search)=\"filterByTerm($event)\"></he-search-extend>\n </div>\n <div class=\"column is-narrow\" *ngIf=\"impactAssessments().length > 1 || enableBreakdown() || !isOriginal()\">\n <div class=\"field has-addons\">\n <div class=\"control\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView === View.table\"\n (click)=\"selectedView = View.table\">\n <span class=\"icon is-small\">\n <fa-icon icon=\"list\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span class=\"is-hidden-mobile\">Table view</span>\n </button>\n </div>\n <div class=\"control\" *ngIf=\"impactAssessments().length > 1\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView === View.chart\"\n (click)=\"selectedView = View.chart\">\n <span class=\"icon is-small\">\n <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span class=\"is-hidden-mobile\">Chart view</span>\n </button>\n </div>\n <div class=\"control\" *ngIf=\"enableBreakdown()\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView === View.breakdown\"\n (click)=\"selectedView = View.breakdown\">\n <span class=\"icon is-small\">\n <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span class=\"is-hidden-mobile\">Breakdown view</span>\n </button>\n </div>\n <div class=\"control\" *ngIf=\"!isOriginal()\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView === View.logs\"\n (click)=\"selectedView = View.logs\">\n <span class=\"icon is-small\">\n <fa-icon icon=\"calculator\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span class=\"is-hidden-mobile\">Recalculations logs</span>\n </button>\n </div>\n </div>\n </div>\n</div>\n\n<div class=\"px-3 pb-3\" [class.is-hidden]=\"selectedView !== View.table\">\n <ng-container *ngIf=\"hasData(); else emptyTable\">\n <he-data-table class=\"mb-1 is-small\" [small]=\"true\" maxHeight=\"320\">\n <table class=\"table is-fullwidth is-narrow is-striped\">\n <thead>\n <tr class=\"has-text-weight-semibold\">\n <th class=\"width-auto py-0\">\n <div class=\"select is-small\" *ngIf=\"enableFilterMethodModel\">\n <select (change)=\"updateSelectedMethodModel($event)\">\n <option [ngValue]=\"undefined\">Filter Model</option>\n <option *ngFor=\"let term of methodModels()\" [ngValue]=\"term\">{{ term.name }}</option>\n </select>\n </div>\n </th>\n <th class=\"has-border-right\"></th>\n <th class=\"has-border-right\"></th>\n <th *ngFor=\"let indicator of indicators()\" [attr.title]=\"indicator.value.term.name\">\n <he-node-link [node]=\"indicator.value.term\">\n <span [innerHtml]=\"indicator.value.term.name | ellipsis: 30 | compound\"></span>\n </he-node-link>\n </th>\n </tr>\n <tr class=\"is-italic has-text-weight-semibold\">\n <th class=\"width-auto has-border-right\"></th>\n <th class=\"has-border-right\">\n <a [href]=\"baseUrl + '/schema/ImpactAssessment#functionalUnit'\" target=\"_blank\">Functional unit</a>\n </th>\n <th class=\"has-border-right\">\n <a [href]=\"baseUrl + '/schema/ImpactAssessment#product'\" target=\"_blank\">Product</a>\n </th>\n <th *ngFor=\"let indicator of indicators()\" [attr.title]=\"indicator.value.term.units\">\n <span [innerHtml]=\"indicator.value.term.units | compound\"></span>\n <he-terms-units-description\n class=\"is-inline-block is-ml-2\"\n [term]=\"indicator.value.term\"></he-terms-units-description>\n </th>\n </tr>\n </thead>\n <tbody>\n <ng-container *ngFor=\"let impactAssessment of impactAssessments(); trackBy: trackById; let i = index\">\n <tr>\n <td class=\"width-auto has-border-right\" [attr.title]=\"impactName(impactAssessment)\">\n <he-node-link [node]=\"impactAssessment\">\n <span>{{ i + 1 }}. {{ impactName(impactAssessment) }}</span>\n </he-node-link>\n </td>\n <td class=\"has-border-right\" [attr.title]=\"impactAssessment.product?.term?.units\">\n <span>1 {{ impactAssessment.product?.term?.units }}</span>\n </td>\n <td class=\"has-border-right\" [attr.title]=\"impactAssessment.product?.term?.name\">\n <he-node-link *ngIf=\"impactAssessment.product?.term\" [node]=\"impactAssessment.product.term\">\n <span>{{ impactAssessment.product.term.name | ellipsis: 30 }}</span>\n </he-node-link>\n </td>\n <td class=\"is-nowrap\" *ngFor=\"let indicator of indicators()\">\n <span\n *ngIf=\"indicator.value.values[impactAssessment['@id']]; else emptyValue\"\n class=\"trigger-popover\"\n [ngbPopover]=\"details\"\n [autoClose]=\"'outside'\"\n popoverClass=\"is-narrow\"\n triggers=\"manual\"\n #p=\"ngbPopover\"\n placement=\"left\"\n container=\"body\"\n (click)=\"\n togglePopover(p, {\n data: indicator.value.values[impactAssessment['@id']],\n impactAssessment: impactAssessment\n })\n \">\n <span pointer>\n {{\n propertyValue(indicator.value.values[impactAssessment['@id']].value, indicator.value.term['@id'])\n | precision: 3\n | default: '-'\n }}\n </span>\n <he-blank-node-state\n class=\"ml-1\"\n [dataState]=\"_dataState()\"\n [node]=\"indicator.value.values[impactAssessment['@id']].node\"\n key=\"value\"\n [state]=\"impactAssessment.aggregated ? 'aggregated' : undefined\"></he-blank-node-state>\n </span>\n </td>\n </tr>\n </ng-container>\n </tbody>\n </table>\n </he-data-table>\n\n <he-blank-node-state-notice [dataState]=\"_dataState()\"></he-blank-node-state-notice>\n </ng-container>\n</div>\n\n<he-impact-assessments-indicator-breakdown-chart\n *ngIf=\"selectedView === View.breakdown\"\n [impactAssessment]=\"impactAssessments()[0]\"\n [indicators]=\"impactAssessments()[0][_nodeKey()]\"></he-impact-assessments-indicator-breakdown-chart>\n\n<he-impact-assessments-indicators-chart\n *ngIf=\"selectedView === View.chart && impactAssessments().length > 1\"\n [key]=\"_nodeKey()\"\n [filterTermTypes]=\"_filterTermTypes()\"></he-impact-assessments-indicators-chart>\n\n<ng-container *ngIf=\"selectedView === View.logs && !isOriginal()\">\n <div class=\"field has-addons pt-2 px-3\" *ngIf=\"impactAssessments().length > 1\">\n <div class=\"control\">\n <span class=\"button is-small is-static is-secondary\">Select an Impact Assessment</span>\n </div>\n <div class=\"control is-expanded\">\n <div class=\"select is-small is-fullwidth is-secondary\">\n <select (change)=\"selectIndex($event)\">\n <option *ngFor=\"let value of impactAssessments(); let i = index\" [value]=\"i\">\n {{ i + 1 }}. {{ impactName(value) }}\n </option>\n </select>\n </div>\n </div>\n </div>\n\n <he-node-logs-models\n *ngIf=\"selectedNode()\"\n [node]=\"selectedNode()\"\n [nodeKey]=\"_nodeKey()\"\n [filterTermTypes]=\"_filterTermTypes()\"\n [originalValues]=\"selectedOriginalValues()\"\n [recalculatedValues]=\"selectedRecalculatedValues()\"\n [noDataMessage]=\"noDataMessage()\"></he-node-logs-models>\n</ng-container>\n\n<he-node-csv-export-confirm\n *ngIf=\"showDownload\"\n [nodes]=\"impactAssessments()\"\n [filename]=\"'impact-' + _nodeKey() + '.csv'\"\n [isUpload]=\"false\"\n [headerKeys]=\"['impactAssessment.id', 'impactAssessment.@id', 'impactAssessment.' + _nodeKey() + '.']\"\n (closed)=\"showDownload = false\"></he-node-csv-export-confirm>\n\n<ng-template #emptyTable>\n <div class=\"has-text-centered\">\n <span>No data available</span>\n <span class=\"is-pl-1\" [class.is-hidden]=\"!filterTerm()\">matching your search criteria</span>\n <span>.</span>\n <span [class.is-hidden]=\"!isOriginal()\">\n Switch to\n <code>recalculated</code>\n version.\n </span>\n </div>\n</ng-template>\n\n<ng-template #emptyValue>\n <span>-</span>\n</ng-template>\n\n<ng-template #details let-node=\"impactAssessment\" let-data=\"data\">\n <p *bindOnce=\"node\">\n <b>\n <span *ngIf=\"data.cycle\">{{ cycleLabel(node.cycle) }}</span>\n <span *ngIf=\"!data.cycle\">{{ data.name }}</span>\n </b>\n </p>\n <he-node-value-details\n [data]=\"data\"\n [dataState]=\"_dataState()\"\n [nodeType]=\"node['@type']\"\n [dataKey]=\"_nodeKey()\"></he-node-value-details>\n</ng-template>\n\n<ng-template #suggestion let-impact=\"result\" let-t=\"term\">\n <div class=\"is-block\">\n <ngb-highlight [result]=\"impact.name || impact.cycle.name\" [term]=\"t\"></ngb-highlight>\n </div>\n <div class=\"columns is-flex\">\n <div class=\"column\" *ngIf=\"impact.country\">\n <span class=\"pr-1 has-text-underline\">Country:</span>\n <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.country.name\" [term]=\"t\"></ngb-highlight></span>\n </div>\n <div class=\"column\" *ngIf=\"impact.product\">\n <span class=\"pr-1 has-text-underline\">Product:</span>\n <span class=\"is-inline-flex\">\n <ngb-highlight [result]=\"impact.product.term?.name || impact.product.name\" [term]=\"t\"></ngb-highlight>\n </span>\n </div>\n <div class=\"column\" *ngIf=\"impact.endDate\">\n <span class=\"pr-1 has-text-underline\">Date:</span>\n <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.endDate\" [term]=\"t\"></ngb-highlight></span>\n </div>\n </div>\n</ng-template>\n", styles: ["fa-icon{display:inline-block;width:10px}he-data-table ::ng-deep .table thead tr th:nth-child(2),he-data-table ::ng-deep .table tbody tr td:nth-child(2){max-width:102px;width:102px}he-data-table ::ng-deep .table thead tr th:nth-child(3),he-data-table ::ng-deep .table tbody tr td:nth-child(3){min-width:110px}\n"] }]
136
+ args: [{ selector: 'he-impact-assessments-products', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"columns is-variable is-align-items-center is-2 is-m-0\" *ngIf=\"indicators().length\">\n <div class=\"column\">\n <ng-container *ngIf=\"selectedView === View.table\">\n <button\n class=\"button is-small is-ghost\"\n (click)=\"showDownload = true\"\n ngbTooltip=\"Download as CSV\"\n placement=\"bottom\">\n <fa-icon icon=\"download\" size=\"lg\"></fa-icon>\n </button>\n </ng-container>\n </div>\n <div class=\"column is-narrow\" *ngIf=\"selectedView === View.table\">\n <he-search-extend\n class=\"is-secondary is-small\"\n placeholder=\"Filter indicators by name\"\n (search)=\"filterByTerm($event)\"></he-search-extend>\n </div>\n <div class=\"column is-narrow\" *ngIf=\"impactAssessments().length > 1 || enableBreakdown() || !isOriginal()\">\n <div class=\"field has-addons\">\n <div class=\"control\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView === View.table\"\n (click)=\"selectedView = View.table\">\n <span class=\"icon is-small\">\n <fa-icon icon=\"list\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span class=\"is-hidden-mobile\">Table view</span>\n </button>\n </div>\n <div class=\"control\" *ngIf=\"impactAssessments().length > 1\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView === View.chart\"\n (click)=\"selectedView = View.chart\">\n <span class=\"icon is-small\">\n <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span class=\"is-hidden-mobile\">Chart view</span>\n </button>\n </div>\n <div class=\"control\" *ngIf=\"enableBreakdown()\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView === View.breakdown\"\n (click)=\"selectedView = View.breakdown\">\n <span class=\"icon is-small\">\n <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span class=\"is-hidden-mobile\">Breakdown view</span>\n </button>\n </div>\n <div class=\"control\" *ngIf=\"!isOriginal()\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView === View.logs\"\n (click)=\"selectedView = View.logs\">\n <span class=\"icon is-small\">\n <fa-icon icon=\"calculator\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span class=\"is-hidden-mobile\">Recalculations logs</span>\n </button>\n </div>\n </div>\n </div>\n</div>\n\n<div class=\"px-3 pb-3\" [class.is-hidden]=\"selectedView !== View.table\">\n <ng-container *ngIf=\"hasData(); else emptyTable\">\n <he-data-table class=\"mb-1 is-small\" [small]=\"true\" maxHeight=\"320\">\n <table class=\"table is-fullwidth is-narrow is-striped\">\n <thead>\n <tr class=\"has-text-weight-semibold\">\n <th class=\"width-auto py-0\">\n <div class=\"select is-small\" *ngIf=\"enableFilterMethodModel\">\n <select (change)=\"updateSelectedMethodModel($event)\">\n <option [ngValue]=\"undefined\">Filter Model</option>\n <option *ngFor=\"let term of methodModels()\" [ngValue]=\"term\">{{ term.name }}</option>\n </select>\n </div>\n </th>\n <th class=\"has-border-right\"></th>\n <th class=\"has-border-right\"></th>\n <th *ngFor=\"let indicator of indicators()\" [attr.title]=\"indicator.value.term.name\">\n <he-node-link [node]=\"indicator.value.term\">\n <span [innerHtml]=\"indicator.value.term.name | ellipsis: 30 | compound\"></span>\n </he-node-link>\n </th>\n </tr>\n <tr class=\"is-italic has-text-weight-semibold\">\n <th class=\"width-auto has-border-right\"></th>\n <th class=\"has-border-right\">\n <a [href]=\"baseUrl + '/schema/ImpactAssessment#functionalUnit'\" target=\"_blank\">Functional unit</a>\n </th>\n <th class=\"has-border-right\">\n <a [href]=\"baseUrl + '/schema/ImpactAssessment#product'\" target=\"_blank\">Product</a>\n </th>\n <th *ngFor=\"let indicator of indicators()\" [attr.title]=\"indicator.value.term.units\">\n <span [innerHtml]=\"indicator.value.term.units | compound\"></span>\n <he-terms-units-description\n class=\"is-inline-block is-ml-2\"\n [term]=\"indicator.value.term\"></he-terms-units-description>\n </th>\n </tr>\n </thead>\n <tbody>\n <ng-container *ngFor=\"let impactAssessment of impactAssessments(); trackBy: trackById; let i = index\">\n <tr>\n <td class=\"width-auto has-border-right\" [attr.title]=\"impactName(impactAssessment)\">\n <he-node-link [node]=\"impactAssessment\">\n <span>{{ i + 1 }}. {{ impactName(impactAssessment) }}</span>\n </he-node-link>\n </td>\n <td class=\"has-border-right\" [attr.title]=\"impactAssessment.product?.term?.units\">\n <span>1 {{ impactAssessment.product?.term?.units }}</span>\n </td>\n <td class=\"has-border-right\" [attr.title]=\"impactAssessment.product?.term?.name\">\n <he-node-link *ngIf=\"impactAssessment.product?.term\" [node]=\"impactAssessment.product.term\">\n <span>{{ impactAssessment.product.term.name | ellipsis: 30 }}</span>\n </he-node-link>\n </td>\n <td class=\"is-nowrap\" *ngFor=\"let indicator of indicators()\">\n <span\n *ngIf=\"indicator.value.values[impactAssessment['@id']]; else emptyValue\"\n class=\"trigger-popover\"\n [ngbPopover]=\"details\"\n [autoClose]=\"'outside'\"\n popoverClass=\"is-narrow\"\n triggers=\"manual\"\n #p=\"ngbPopover\"\n placement=\"left\"\n container=\"body\"\n (click)=\"\n togglePopover(p, {\n data: indicator.value.values[impactAssessment['@id']],\n impactAssessment: impactAssessment\n })\n \">\n <span pointer>\n {{\n propertyValue(indicator.value.values[impactAssessment['@id']].value, indicator.value.term['@id'])\n | precision: 3\n | default: '-'\n }}\n </span>\n <he-blank-node-state\n class=\"ml-1\"\n [dataState]=\"_dataState()\"\n [node]=\"indicator.value.values[impactAssessment['@id']].node\"\n key=\"value\"\n [state]=\"impactAssessment.aggregated ? 'aggregated' : undefined\"></he-blank-node-state>\n </span>\n </td>\n </tr>\n </ng-container>\n </tbody>\n </table>\n </he-data-table>\n\n <he-blank-node-state-notice [dataState]=\"_dataState()\"></he-blank-node-state-notice>\n </ng-container>\n</div>\n\n<he-impact-assessments-indicator-breakdown-chart\n *ngIf=\"selectedView === View.breakdown\"\n [impactAssessment]=\"impactAssessments()[0]\"\n [indicators]=\"impactAssessments()[0][_nodeKey()]\"></he-impact-assessments-indicator-breakdown-chart>\n\n<he-impact-assessments-indicators-chart\n *ngIf=\"selectedView === View.chart && impactAssessments().length > 1\"\n [key]=\"_nodeKey()\"\n [filterTermTypes]=\"_filterTermTypes()\"></he-impact-assessments-indicators-chart>\n\n<ng-container *ngIf=\"selectedView === View.logs && !isOriginal()\">\n <div class=\"field has-addons pt-2 px-3\" *ngIf=\"impactAssessments().length > 1\">\n <div class=\"control\">\n <span class=\"button is-small is-static is-secondary\">Select an Impact Assessment</span>\n </div>\n <div class=\"control is-expanded\">\n <div class=\"select is-small is-fullwidth is-secondary\">\n <select (change)=\"selectIndex($event)\">\n <option *ngFor=\"let value of impactAssessments(); let i = index\" [value]=\"i\">\n {{ i + 1 }}. {{ impactName(value) }}\n </option>\n </select>\n </div>\n </div>\n </div>\n\n <he-node-logs-models\n *ngIf=\"selectedNode()\"\n [node]=\"selectedNode()\"\n [nodeKey]=\"_nodeKey()\"\n [filterTermTypes]=\"_filterTermTypes()\"\n [originalValues]=\"selectedOriginalValues()\"\n [recalculatedValues]=\"selectedRecalculatedValues()\"\n [noDataMessage]=\"noDataMessage()\"></he-node-logs-models>\n</ng-container>\n\n<he-node-csv-export-confirm\n *ngIf=\"showDownload\"\n [nodes]=\"impactAssessments()\"\n [filename]=\"'impact-' + _nodeKey() + '.csv'\"\n [isUpload]=\"false\"\n [headerKeys]=\"['impactAssessment.id', 'impactAssessment.@id', 'impactAssessment.' + _nodeKey() + '.']\"\n (closed)=\"showDownload = false\"></he-node-csv-export-confirm>\n\n<ng-template #emptyTable>\n <div class=\"has-text-centered\">\n <span>No data available</span>\n <span class=\"is-pl-1\" [class.is-hidden]=\"!filterTerm()\">matching your search criteria</span>\n <span>.</span>\n <span [class.is-hidden]=\"!isOriginal()\">\n Switch to\n <code>recalculated</code>\n version.\n </span>\n </div>\n</ng-template>\n\n<ng-template #emptyValue>\n <span>-</span>\n</ng-template>\n\n<ng-template #details let-node=\"impactAssessment\" let-data=\"data\">\n <p *bindOnce=\"node\">\n <b>\n <span *ngIf=\"data.cycle\">{{ cycleLabel(node.cycle) }}</span>\n <span *ngIf=\"!data.cycle\">{{ data.name }}</span>\n </b>\n </p>\n <he-node-value-details\n [data]=\"data\"\n [dataState]=\"_dataState()\"\n [nodeType]=\"node['@type']\"\n [dataKey]=\"_nodeKey()\"></he-node-value-details>\n</ng-template>\n\n<ng-template #suggestion let-impact=\"result\" let-t=\"term\">\n <div class=\"is-block\">\n <ngb-highlight [result]=\"impact.name || impact.cycle.name\" [term]=\"t\"></ngb-highlight>\n </div>\n <div class=\"columns is-flex\">\n <div class=\"column\" *ngIf=\"impact.country\">\n <span class=\"pr-1 has-text-underline\">Country:</span>\n <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.country.name\" [term]=\"t\"></ngb-highlight></span>\n </div>\n <div class=\"column\" *ngIf=\"impact.product\">\n <span class=\"pr-1 has-text-underline\">Product:</span>\n <span class=\"is-inline-flex\">\n <ngb-highlight [result]=\"impact.product.term?.name || impact.product.name\" [term]=\"t\"></ngb-highlight>\n </span>\n </div>\n <div class=\"column\" *ngIf=\"impact.endDate\">\n <span class=\"pr-1 has-text-underline\">Date:</span>\n <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.endDate\" [term]=\"t\"></ngb-highlight></span>\n </div>\n </div>\n</ng-template>\n", styles: ["fa-icon{display:inline-block;width:10px}he-data-table ::ng-deep .table thead tr th:nth-child(2),he-data-table ::ng-deep .table tbody tr td:nth-child(2){max-width:102px;width:102px}he-data-table ::ng-deep .table thead tr th:nth-child(3),he-data-table ::ng-deep .table tbody tr td:nth-child(3){min-width:110px}\n"] }]
137
137
  }], ctorParameters: function () { return [{ type: i1.HeNodeStoreService }]; }, propDecorators: { dataState: [{
138
138
  type: Input
139
139
  }], nodeKey: [{
@@ -143,4 +143,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.8", ngImpor
143
143
  }], enableFilterMethodModel: [{
144
144
  type: Input
145
145
  }] } });
146
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"impact-assessments-products.component.js","sourceRoot":"","sources":["../../../../src/impact-assessments/impact-assessments-products/impact-assessments-products.component.ts","../../../../src/impact-assessments/impact-assessments-products/impact-assessments-products.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, Input, computed, effect, signal } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { DataState } from '@hestia-earth/api';\nimport { ICycleJSONLD, IImpactAssessmentJSONLD, Indicator, TermTermType, Term, NodeType } from '@hestia-earth/schema';\nimport { propertyValue } from '@hestia-earth/utils/dist/term';\nimport orderBy from 'lodash.orderby';\nimport uniqBy from 'lodash.uniqby';\n\nimport {\n  BlankNodesKey,\n  groupNodesByTerm,\n  IGroupedKeys,\n  grouppedKeys,\n  isMethodModelAllowed\n} from '../../common/node-utils';\nimport { HeNodeStoreService } from '../../node/node-store.service';\nimport { baseUrl } from '../../common/utils';\n\nenum View {\n  table = 'table',\n  chart = 'chart',\n  breakdown = 'breakdown',\n  logs = 'logs'\n}\n\nconst termAllowed = (term?: Term, filterTermTypes: TermTermType[] = []) =>\n  !filterTermTypes?.length || (filterTermTypes || []).includes(term?.termType);\n\nconst filterNode = (filterByMethodModel: boolean, filterMethodModel?: Term, filterTerm?: string) => (node: Indicator) =>\n  [\n    !filterByMethodModel || isMethodModelAllowed(filterMethodModel)(node),\n    !filterTerm || node?.term?.name?.toLowerCase()?.includes(filterTerm.toLowerCase())\n  ].every(Boolean);\n\n@Component({\n  selector: 'he-impact-assessments-products',\n  templateUrl: './impact-assessments-products.component.html',\n  styleUrls: ['./impact-assessments-products.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class ImpactAssessmentsProductsComponent {\n  protected _dataState = signal(DataState.original);\n\n  @Input()\n  set dataState(value: DataState) {\n    this._dataState.set(value);\n  }\n\n  protected _nodeKey = signal(BlankNodesKey.impacts);\n  @Input()\n  set nodeKey(value: BlankNodesKey) {\n    this._nodeKey.set(value);\n  }\n\n  protected _filterTermTypes = signal([] as TermTermType[]);\n\n  @Input()\n  set filterTermTypes(value: TermTermType[]) {\n    this._filterTermTypes.set(value);\n  }\n\n  private _enableFilterMethodModel = signal(false);\n\n  @Input()\n  set enableFilterMethodModel(value: boolean) {\n    this._enableFilterMethodModel.set(value);\n  }\n\n  protected propertyValue = propertyValue;\n  protected baseUrl = baseUrl();\n  protected showDownload = false;\n  protected View = View;\n  protected selectedView = View.table;\n\n  private cycles = toSignal(this.nodeStoreService.findByState$<ICycleJSONLD>(NodeType.Cycle, DataState.original));\n\n  private originalImpactAssessments = toSignal(\n    this.nodeStoreService.findByState$<IImpactAssessmentJSONLD>(NodeType.ImpactAssessment, DataState.original)\n  );\n  private _allImpactAssessments = toSignal(\n    this.nodeStoreService.find$<IImpactAssessmentJSONLD>(NodeType.ImpactAssessment)\n  );\n  protected impactAssessments = computed(\n    () => this._allImpactAssessments()?.map(data => data[this._dataState()]) || []\n  );\n\n  private selectedIndex = signal(0);\n  private ogirinalSelectedImpactAssessment = computed(() => this.originalImpactAssessments()?.[this.selectedIndex()]);\n  private selectedImpactAssessment = computed(() => this.impactAssessments()?.[this.selectedIndex()]);\n  protected selectedOriginalValues = computed(() => this.ogirinalSelectedImpactAssessment()?.[this._nodeKey()] || []);\n  protected selectedRecalculatedValues = computed(() => this.selectedImpactAssessment()?.[this._nodeKey()] || []);\n  protected selectedNode = computed(() =>\n    this.selectedImpactAssessment()\n      ? {\n          ...this.selectedImpactAssessment(),\n          dataState: DataState.recalculated\n        }\n      : null\n  );\n\n  protected isOriginal = computed(() => this._dataState() === DataState.original);\n  private originalValues = computed(() => (this.isOriginal() ? null : this.originalImpactAssessments()) || []);\n\n  protected enableBreakdown = computed(\n    () => !this.isOriginal() && this.impactAssessments().length === 1 && this._nodeKey() !== 'emissionsResourceUse'\n  );\n\n  private shouldFilterByMethodModel = computed(() => this._enableFilterMethodModel() && !this.isOriginal());\n  protected filterTerm = signal('');\n\n  // as we filter by methodModel, we might have an empty table for the default models but not for other models\n  protected hasData = computed(() =>\n    this._enableFilterMethodModel() ? this.methodModels().length : this.indicators().length\n  );\n\n  protected noDataMessage = computed(() =>\n    this._nodeKey() === BlankNodesKey.emissionsResourceUse\n      ? 'We cannot calculate these data, most likely because we are missing the quantity produced or because we are missing data to allocate environmental impact data between co-products.'\n      : undefined\n  );\n\n  protected methodModels = computed(\n    () =>\n      uniqBy(\n        this.impactAssessments()\n          .flatMap(impact => (impact[this._nodeKey()] || []).map(v => v.methodModel))\n          .filter(Boolean),\n        '@id'\n      ) as Term[]\n  );\n\n  private _selectedMethodModel = signal(undefined as Term);\n\n  protected indicators = computed(() => {\n    // TODO: when there are no indicators for the default model, the table appears empty\n    // should default back to the first model in this case\n    const indicatorPerImpactAssessment = groupNodesByTerm<IImpactAssessmentJSONLD, Indicator>(\n      this.impactAssessments(),\n      this._nodeKey(),\n      this.originalValues(),\n      filterNode(this.shouldFilterByMethodModel(), this._selectedMethodModel(), this.filterTerm())\n    );\n    return orderBy(\n      grouppedKeys(indicatorPerImpactAssessment) as any[],\n      ['value.methodTierOrder', 'key'],\n      ['asc', 'asc']\n    ).filter(({ value }) => termAllowed(value?.term, this._filterTermTypes())) as IGroupedKeys<Indicator>[];\n  });\n\n  constructor(private nodeStoreService: HeNodeStoreService) {\n    effect(() => {\n      // make sure logs does not remain displayed when switching back to original view\n      if (this.isOriginal() && this.selectedView === View.logs) {\n        this.selectedView = View.table;\n      }\n    });\n  }\n\n  protected trackById(_index: number, item: IImpactAssessmentJSONLD) {\n    return item['@id'];\n  }\n\n  protected updateSelectedMethodModel({ target: { value } }) {\n    this._selectedMethodModel.set(value);\n  }\n\n  protected togglePopover(popover, context) {\n    return popover.isOpen() ? popover.close() : popover.open(context);\n  }\n\n  protected impactName(impact: IImpactAssessmentJSONLD) {\n    return impact.name || (impact.cycle ? this.cycleLabel(impact.cycle) : impact['@id']) || (impact as any).id;\n  }\n\n  protected cycleLabel({ '@id': id, name }: any) {\n    return name || this.cycles().find(v => v['@id'] === id)?.name;\n  }\n\n  protected filterByTerm(term: string) {\n    this.filterTerm.set(term);\n  }\n\n  protected selectIndex({ target: { value } }) {\n    this.selectedIndex.set(+value);\n  }\n}\n","<div class=\"columns is-variable is-align-items-center is-2 m-0\">\n  <div class=\"column\">\n    <ng-container *ngIf=\"selectedView === View.table && indicators().length\">\n      <button\n        class=\"button is-small is-ghost\"\n        (click)=\"showDownload = true\"\n        ngbTooltip=\"Download as CSV\"\n        placement=\"bottom\">\n        <fa-icon icon=\"download\" size=\"lg\"></fa-icon>\n      </button>\n    </ng-container>\n  </div>\n  <div class=\"column is-narrow\" *ngIf=\"selectedView === View.table\">\n    <he-search-extend\n      class=\"is-secondary is-small\"\n      placeholder=\"Filter indicators by name\"\n      (search)=\"filterByTerm($event)\"></he-search-extend>\n  </div>\n  <div class=\"column is-narrow\" *ngIf=\"impactAssessments().length > 1 || enableBreakdown() || !isOriginal()\">\n    <div class=\"field has-addons\">\n      <div class=\"control\">\n        <button\n          class=\"button is-small\"\n          [class.is-selected]=\"selectedView === View.table\"\n          (click)=\"selectedView = View.table\">\n          <span class=\"icon is-small\">\n            <fa-icon icon=\"list\" aria-hidden=\"true\"></fa-icon>\n          </span>\n          <span class=\"is-hidden-mobile\">Table view</span>\n        </button>\n      </div>\n      <div class=\"control\" *ngIf=\"impactAssessments().length > 1\">\n        <button\n          class=\"button is-small\"\n          [class.is-selected]=\"selectedView === View.chart\"\n          (click)=\"selectedView = View.chart\">\n          <span class=\"icon is-small\">\n            <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n          </span>\n          <span class=\"is-hidden-mobile\">Chart view</span>\n        </button>\n      </div>\n      <div class=\"control\" *ngIf=\"enableBreakdown()\">\n        <button\n          class=\"button is-small\"\n          [class.is-selected]=\"selectedView === View.breakdown\"\n          (click)=\"selectedView = View.breakdown\">\n          <span class=\"icon is-small\">\n            <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n          </span>\n          <span class=\"is-hidden-mobile\">Breakdown view</span>\n        </button>\n      </div>\n      <div class=\"control\" *ngIf=\"!isOriginal()\">\n        <button\n          class=\"button is-small\"\n          [class.is-selected]=\"selectedView === View.logs\"\n          (click)=\"selectedView = View.logs\">\n          <span class=\"icon is-small\">\n            <fa-icon icon=\"calculator\" aria-hidden=\"true\"></fa-icon>\n          </span>\n          <span class=\"is-hidden-mobile\">Recalculations logs</span>\n        </button>\n      </div>\n    </div>\n  </div>\n</div>\n\n<div class=\"px-3 pb-3\" [class.is-hidden]=\"selectedView !== View.table\">\n  <ng-container *ngIf=\"hasData(); else emptyTable\">\n    <he-data-table class=\"mb-1 is-small\" [small]=\"true\" maxHeight=\"320\">\n      <table class=\"table is-fullwidth is-narrow is-striped\">\n        <thead>\n          <tr class=\"has-text-weight-semibold\">\n            <th class=\"width-auto py-0\">\n              <div class=\"select is-small\" *ngIf=\"enableFilterMethodModel\">\n                <select (change)=\"updateSelectedMethodModel($event)\">\n                  <option [ngValue]=\"undefined\">Filter Model</option>\n                  <option *ngFor=\"let term of methodModels()\" [ngValue]=\"term\">{{ term.name }}</option>\n                </select>\n              </div>\n            </th>\n            <th class=\"has-border-right\"></th>\n            <th class=\"has-border-right\"></th>\n            <th *ngFor=\"let indicator of indicators()\" [attr.title]=\"indicator.value.term.name\">\n              <he-node-link [node]=\"indicator.value.term\">\n                <span [innerHtml]=\"indicator.value.term.name | ellipsis: 30 | compound\"></span>\n              </he-node-link>\n            </th>\n          </tr>\n          <tr class=\"is-italic has-text-weight-semibold\">\n            <th class=\"width-auto has-border-right\"></th>\n            <th class=\"has-border-right\">\n              <a [href]=\"baseUrl + '/schema/ImpactAssessment#functionalUnit'\" target=\"_blank\">Functional unit</a>\n            </th>\n            <th class=\"has-border-right\">\n              <a [href]=\"baseUrl + '/schema/ImpactAssessment#product'\" target=\"_blank\">Product</a>\n            </th>\n            <th *ngFor=\"let indicator of indicators()\" [attr.title]=\"indicator.value.term.units\">\n              <span [innerHtml]=\"indicator.value.term.units | compound\"></span>\n              <he-terms-units-description\n                class=\"is-inline-block is-ml-2\"\n                [term]=\"indicator.value.term\"></he-terms-units-description>\n            </th>\n          </tr>\n        </thead>\n        <tbody>\n          <ng-container *ngFor=\"let impactAssessment of impactAssessments(); trackBy: trackById; let i = index\">\n            <tr>\n              <td class=\"width-auto has-border-right\" [attr.title]=\"impactName(impactAssessment)\">\n                <he-node-link [node]=\"impactAssessment\">\n                  <span>{{ i + 1 }}. {{ impactName(impactAssessment) }}</span>\n                </he-node-link>\n              </td>\n              <td class=\"has-border-right\" [attr.title]=\"impactAssessment.product?.term?.units\">\n                <span>1 {{ impactAssessment.product?.term?.units }}</span>\n              </td>\n              <td class=\"has-border-right\" [attr.title]=\"impactAssessment.product?.term?.name\">\n                <he-node-link *ngIf=\"impactAssessment.product?.term\" [node]=\"impactAssessment.product.term\">\n                  <span>{{ impactAssessment.product.term.name | ellipsis: 30 }}</span>\n                </he-node-link>\n              </td>\n              <td class=\"is-nowrap\" *ngFor=\"let indicator of indicators()\">\n                <span\n                  *ngIf=\"indicator.value.values[impactAssessment['@id']]; else emptyValue\"\n                  class=\"trigger-popover\"\n                  [ngbPopover]=\"details\"\n                  [autoClose]=\"'outside'\"\n                  popoverClass=\"is-narrow\"\n                  triggers=\"manual\"\n                  #p=\"ngbPopover\"\n                  placement=\"left\"\n                  container=\"body\"\n                  (click)=\"\n                    togglePopover(p, {\n                      data: indicator.value.values[impactAssessment['@id']],\n                      impactAssessment: impactAssessment\n                    })\n                  \">\n                  <span pointer>\n                    {{\n                      propertyValue(indicator.value.values[impactAssessment['@id']].value, indicator.value.term['@id'])\n                        | precision: 3\n                        | default: '-'\n                    }}\n                  </span>\n                  <he-blank-node-state\n                    class=\"ml-1\"\n                    [dataState]=\"_dataState()\"\n                    [node]=\"indicator.value.values[impactAssessment['@id']].node\"\n                    key=\"value\"\n                    [state]=\"impactAssessment.aggregated ? 'aggregated' : undefined\"></he-blank-node-state>\n                </span>\n              </td>\n            </tr>\n          </ng-container>\n        </tbody>\n      </table>\n    </he-data-table>\n\n    <he-blank-node-state-notice [dataState]=\"_dataState()\"></he-blank-node-state-notice>\n  </ng-container>\n</div>\n\n<he-impact-assessments-indicator-breakdown-chart\n  *ngIf=\"selectedView === View.breakdown\"\n  [impactAssessment]=\"impactAssessments()[0]\"\n  [indicators]=\"impactAssessments()[0][_nodeKey()]\"></he-impact-assessments-indicator-breakdown-chart>\n\n<he-impact-assessments-indicators-chart\n  *ngIf=\"selectedView === View.chart && impactAssessments().length > 1\"\n  [key]=\"_nodeKey()\"\n  [filterTermTypes]=\"_filterTermTypes()\"></he-impact-assessments-indicators-chart>\n\n<ng-container *ngIf=\"selectedView === View.logs && !isOriginal()\">\n  <div class=\"field has-addons pt-2 px-3\" *ngIf=\"impactAssessments().length > 1\">\n    <div class=\"control\">\n      <span class=\"button is-small is-static is-secondary\">Select an Impact Assessment</span>\n    </div>\n    <div class=\"control is-expanded\">\n      <div class=\"select is-small is-fullwidth is-secondary\">\n        <select (change)=\"selectIndex($event)\">\n          <option *ngFor=\"let value of impactAssessments(); let i = index\" [value]=\"i\">\n            {{ i + 1 }}. {{ impactName(value) }}\n          </option>\n        </select>\n      </div>\n    </div>\n  </div>\n\n  <he-node-logs-models\n    *ngIf=\"selectedNode()\"\n    [node]=\"selectedNode()\"\n    [nodeKey]=\"_nodeKey()\"\n    [filterTermTypes]=\"_filterTermTypes()\"\n    [originalValues]=\"selectedOriginalValues()\"\n    [recalculatedValues]=\"selectedRecalculatedValues()\"\n    [noDataMessage]=\"noDataMessage()\"></he-node-logs-models>\n</ng-container>\n\n<he-node-csv-export-confirm\n  *ngIf=\"showDownload\"\n  [nodes]=\"impactAssessments()\"\n  [filename]=\"'impact-' + _nodeKey() + '.csv'\"\n  [isUpload]=\"false\"\n  [headerKeys]=\"['impactAssessment.id', 'impactAssessment.@id', 'impactAssessment.' + _nodeKey() + '.']\"\n  (closed)=\"showDownload = false\"></he-node-csv-export-confirm>\n\n<ng-template #emptyTable>\n  <div class=\"has-text-centered\">\n    <span>No data available</span>\n    <span class=\"is-pl-1\" [class.is-hidden]=\"!filterTerm()\">matching your search criteria</span>\n    <span>.</span>\n    <span [class.is-hidden]=\"!isOriginal()\">\n      Switch to\n      <code>recalculated</code>\n      version.\n    </span>\n  </div>\n</ng-template>\n\n<ng-template #emptyValue>\n  <span>-</span>\n</ng-template>\n\n<ng-template #details let-node=\"impactAssessment\" let-data=\"data\">\n  <p *bindOnce=\"node\">\n    <b>\n      <span *ngIf=\"data.cycle\">{{ cycleLabel(node.cycle) }}</span>\n      <span *ngIf=\"!data.cycle\">{{ data.name }}</span>\n    </b>\n  </p>\n  <he-node-value-details\n    [data]=\"data\"\n    [dataState]=\"_dataState()\"\n    [nodeType]=\"node['@type']\"\n    [dataKey]=\"_nodeKey()\"></he-node-value-details>\n</ng-template>\n\n<ng-template #suggestion let-impact=\"result\" let-t=\"term\">\n  <div class=\"is-block\">\n    <ngb-highlight [result]=\"impact.name || impact.cycle.name\" [term]=\"t\"></ngb-highlight>\n  </div>\n  <div class=\"columns is-flex\">\n    <div class=\"column\" *ngIf=\"impact.country\">\n      <span class=\"pr-1 has-text-underline\">Country:</span>\n      <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.country.name\" [term]=\"t\"></ngb-highlight></span>\n    </div>\n    <div class=\"column\" *ngIf=\"impact.product\">\n      <span class=\"pr-1 has-text-underline\">Product:</span>\n      <span class=\"is-inline-flex\">\n        <ngb-highlight [result]=\"impact.product.term?.name || impact.product.name\" [term]=\"t\"></ngb-highlight>\n      </span>\n    </div>\n    <div class=\"column\" *ngIf=\"impact.endDate\">\n      <span class=\"pr-1 has-text-underline\">Date:</span>\n      <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.endDate\" [term]=\"t\"></ngb-highlight></span>\n    </div>\n  </div>\n</ng-template>\n"]}
146
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"impact-assessments-products.component.js","sourceRoot":"","sources":["../../../../src/impact-assessments/impact-assessments-products/impact-assessments-products.component.ts","../../../../src/impact-assessments/impact-assessments-products/impact-assessments-products.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, Input, computed, effect, signal } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { DataState } from '@hestia-earth/api';\nimport { ICycleJSONLD, IImpactAssessmentJSONLD, Indicator, TermTermType, Term, NodeType } from '@hestia-earth/schema';\nimport { propertyValue } from '@hestia-earth/utils/dist/term';\nimport orderBy from 'lodash.orderby';\nimport uniqBy from 'lodash.uniqby';\n\nimport {\n  BlankNodesKey,\n  groupNodesByTerm,\n  IGroupedKeys,\n  grouppedKeys,\n  isMethodModelAllowed\n} from '../../common/node-utils';\nimport { HeNodeStoreService } from '../../node/node-store.service';\nimport { baseUrl } from '../../common/utils';\n\nenum View {\n  table = 'table',\n  chart = 'chart',\n  breakdown = 'breakdown',\n  logs = 'logs'\n}\n\nconst termAllowed = (term?: Term, filterTermTypes: TermTermType[] = []) =>\n  !filterTermTypes?.length || (filterTermTypes || []).includes(term?.termType);\n\nconst filterNode = (filterByMethodModel: boolean, filterMethodModel?: Term, filterTerm?: string) => (node: Indicator) =>\n  [\n    !filterByMethodModel || isMethodModelAllowed(filterMethodModel)(node),\n    !filterTerm || node?.term?.name?.toLowerCase()?.includes(filterTerm.toLowerCase())\n  ].every(Boolean);\n\n@Component({\n  selector: 'he-impact-assessments-products',\n  templateUrl: './impact-assessments-products.component.html',\n  styleUrls: ['./impact-assessments-products.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class ImpactAssessmentsProductsComponent {\n  protected _dataState = signal(DataState.original);\n\n  @Input()\n  set dataState(value: DataState) {\n    this._dataState.set(value);\n  }\n\n  protected _nodeKey = signal(BlankNodesKey.impacts);\n  @Input()\n  set nodeKey(value: BlankNodesKey) {\n    this._nodeKey.set(value);\n  }\n\n  protected _filterTermTypes = signal([] as TermTermType[]);\n\n  @Input()\n  set filterTermTypes(value: TermTermType[]) {\n    this._filterTermTypes.set(value);\n  }\n\n  private _enableFilterMethodModel = signal(false);\n\n  @Input()\n  set enableFilterMethodModel(value: boolean) {\n    this._enableFilterMethodModel.set(value);\n  }\n\n  protected propertyValue = propertyValue;\n  protected baseUrl = baseUrl();\n  protected showDownload = false;\n  protected View = View;\n  protected selectedView = View.table;\n\n  private cycles = toSignal(this.nodeStoreService.findByState$<ICycleJSONLD>(NodeType.Cycle, DataState.original));\n\n  private originalImpactAssessments = toSignal(\n    this.nodeStoreService.findByState$<IImpactAssessmentJSONLD>(NodeType.ImpactAssessment, DataState.original)\n  );\n  private _allImpactAssessments = toSignal(\n    this.nodeStoreService.find$<IImpactAssessmentJSONLD>(NodeType.ImpactAssessment)\n  );\n  protected impactAssessments = computed(\n    () => this._allImpactAssessments()?.map(data => data[this._dataState()]) || []\n  );\n\n  private selectedIndex = signal(0);\n  private ogirinalSelectedImpactAssessment = computed(() => this.originalImpactAssessments()?.[this.selectedIndex()]);\n  private selectedImpactAssessment = computed(() => this.impactAssessments()?.[this.selectedIndex()]);\n  protected selectedOriginalValues = computed(() => this.ogirinalSelectedImpactAssessment()?.[this._nodeKey()] || []);\n  protected selectedRecalculatedValues = computed(() => this.selectedImpactAssessment()?.[this._nodeKey()] || []);\n  protected selectedNode = computed(() =>\n    this.selectedImpactAssessment()\n      ? {\n          ...this.selectedImpactAssessment(),\n          dataState: DataState.recalculated\n        }\n      : null\n  );\n\n  protected isOriginal = computed(() => this._dataState() === DataState.original);\n  private originalValues = computed(() => (this.isOriginal() ? null : this.originalImpactAssessments()) || []);\n\n  protected enableBreakdown = computed(\n    () => !this.isOriginal() && this.impactAssessments().length === 1 && this._nodeKey() !== 'emissionsResourceUse'\n  );\n\n  private shouldFilterByMethodModel = computed(() => this._enableFilterMethodModel() && !this.isOriginal());\n  protected filterTerm = signal('');\n\n  // as we filter by methodModel, we might have an empty table for the default models but not for other models\n  protected hasData = computed(() =>\n    this._enableFilterMethodModel() ? this.methodModels().length : this.indicators().length\n  );\n\n  protected noDataMessage = computed(() =>\n    this._nodeKey() === BlankNodesKey.emissionsResourceUse\n      ? 'We cannot calculate these data, most likely because we are missing the quantity produced or because we are missing data to allocate environmental impact data between co-products.'\n      : undefined\n  );\n\n  protected methodModels = computed(\n    () =>\n      uniqBy(\n        this.impactAssessments()\n          .flatMap(impact => (impact[this._nodeKey()] || []).map(v => v.methodModel))\n          .filter(Boolean),\n        '@id'\n      ) as Term[]\n  );\n\n  private _selectedMethodModel = signal(undefined as Term);\n\n  protected indicators = computed(() => {\n    // TODO: when there are no indicators for the default model, the table appears empty\n    // should default back to the first model in this case\n    const indicatorPerImpactAssessment = groupNodesByTerm<IImpactAssessmentJSONLD, Indicator>(\n      this.impactAssessments(),\n      this._nodeKey(),\n      this.originalValues(),\n      filterNode(this.shouldFilterByMethodModel(), this._selectedMethodModel(), this.filterTerm())\n    );\n    return orderBy(\n      grouppedKeys(indicatorPerImpactAssessment) as any[],\n      ['value.methodTierOrder', 'key'],\n      ['asc', 'asc']\n    ).filter(({ value }) => termAllowed(value?.term, this._filterTermTypes())) as IGroupedKeys<Indicator>[];\n  });\n\n  constructor(private nodeStoreService: HeNodeStoreService) {\n    effect(() => {\n      // make sure logs does not remain displayed when switching back to original view\n      if (this.isOriginal() && this.selectedView === View.logs) {\n        this.selectedView = View.table;\n      }\n    });\n  }\n\n  protected trackById(_index: number, item: IImpactAssessmentJSONLD) {\n    return item['@id'];\n  }\n\n  protected updateSelectedMethodModel({ target: { value } }) {\n    this._selectedMethodModel.set(value);\n  }\n\n  protected togglePopover(popover, context) {\n    return popover.isOpen() ? popover.close() : popover.open(context);\n  }\n\n  protected impactName(impact: IImpactAssessmentJSONLD) {\n    return impact.name || (impact.cycle ? this.cycleLabel(impact.cycle) : impact['@id']) || (impact as any).id;\n  }\n\n  protected cycleLabel({ '@id': id, name }: any) {\n    return name || this.cycles().find(v => v['@id'] === id)?.name;\n  }\n\n  protected filterByTerm(term: string) {\n    this.filterTerm.set(term);\n  }\n\n  protected selectIndex({ target: { value } }) {\n    this.selectedIndex.set(+value);\n  }\n}\n","<div class=\"columns is-variable is-align-items-center is-2 is-m-0\" *ngIf=\"indicators().length\">\n  <div class=\"column\">\n    <ng-container *ngIf=\"selectedView === View.table\">\n      <button\n        class=\"button is-small is-ghost\"\n        (click)=\"showDownload = true\"\n        ngbTooltip=\"Download as CSV\"\n        placement=\"bottom\">\n        <fa-icon icon=\"download\" size=\"lg\"></fa-icon>\n      </button>\n    </ng-container>\n  </div>\n  <div class=\"column is-narrow\" *ngIf=\"selectedView === View.table\">\n    <he-search-extend\n      class=\"is-secondary is-small\"\n      placeholder=\"Filter indicators by name\"\n      (search)=\"filterByTerm($event)\"></he-search-extend>\n  </div>\n  <div class=\"column is-narrow\" *ngIf=\"impactAssessments().length > 1 || enableBreakdown() || !isOriginal()\">\n    <div class=\"field has-addons\">\n      <div class=\"control\">\n        <button\n          class=\"button is-small\"\n          [class.is-selected]=\"selectedView === View.table\"\n          (click)=\"selectedView = View.table\">\n          <span class=\"icon is-small\">\n            <fa-icon icon=\"list\" aria-hidden=\"true\"></fa-icon>\n          </span>\n          <span class=\"is-hidden-mobile\">Table view</span>\n        </button>\n      </div>\n      <div class=\"control\" *ngIf=\"impactAssessments().length > 1\">\n        <button\n          class=\"button is-small\"\n          [class.is-selected]=\"selectedView === View.chart\"\n          (click)=\"selectedView = View.chart\">\n          <span class=\"icon is-small\">\n            <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n          </span>\n          <span class=\"is-hidden-mobile\">Chart view</span>\n        </button>\n      </div>\n      <div class=\"control\" *ngIf=\"enableBreakdown()\">\n        <button\n          class=\"button is-small\"\n          [class.is-selected]=\"selectedView === View.breakdown\"\n          (click)=\"selectedView = View.breakdown\">\n          <span class=\"icon is-small\">\n            <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n          </span>\n          <span class=\"is-hidden-mobile\">Breakdown view</span>\n        </button>\n      </div>\n      <div class=\"control\" *ngIf=\"!isOriginal()\">\n        <button\n          class=\"button is-small\"\n          [class.is-selected]=\"selectedView === View.logs\"\n          (click)=\"selectedView = View.logs\">\n          <span class=\"icon is-small\">\n            <fa-icon icon=\"calculator\" aria-hidden=\"true\"></fa-icon>\n          </span>\n          <span class=\"is-hidden-mobile\">Recalculations logs</span>\n        </button>\n      </div>\n    </div>\n  </div>\n</div>\n\n<div class=\"px-3 pb-3\" [class.is-hidden]=\"selectedView !== View.table\">\n  <ng-container *ngIf=\"hasData(); else emptyTable\">\n    <he-data-table class=\"mb-1 is-small\" [small]=\"true\" maxHeight=\"320\">\n      <table class=\"table is-fullwidth is-narrow is-striped\">\n        <thead>\n          <tr class=\"has-text-weight-semibold\">\n            <th class=\"width-auto py-0\">\n              <div class=\"select is-small\" *ngIf=\"enableFilterMethodModel\">\n                <select (change)=\"updateSelectedMethodModel($event)\">\n                  <option [ngValue]=\"undefined\">Filter Model</option>\n                  <option *ngFor=\"let term of methodModels()\" [ngValue]=\"term\">{{ term.name }}</option>\n                </select>\n              </div>\n            </th>\n            <th class=\"has-border-right\"></th>\n            <th class=\"has-border-right\"></th>\n            <th *ngFor=\"let indicator of indicators()\" [attr.title]=\"indicator.value.term.name\">\n              <he-node-link [node]=\"indicator.value.term\">\n                <span [innerHtml]=\"indicator.value.term.name | ellipsis: 30 | compound\"></span>\n              </he-node-link>\n            </th>\n          </tr>\n          <tr class=\"is-italic has-text-weight-semibold\">\n            <th class=\"width-auto has-border-right\"></th>\n            <th class=\"has-border-right\">\n              <a [href]=\"baseUrl + '/schema/ImpactAssessment#functionalUnit'\" target=\"_blank\">Functional unit</a>\n            </th>\n            <th class=\"has-border-right\">\n              <a [href]=\"baseUrl + '/schema/ImpactAssessment#product'\" target=\"_blank\">Product</a>\n            </th>\n            <th *ngFor=\"let indicator of indicators()\" [attr.title]=\"indicator.value.term.units\">\n              <span [innerHtml]=\"indicator.value.term.units | compound\"></span>\n              <he-terms-units-description\n                class=\"is-inline-block is-ml-2\"\n                [term]=\"indicator.value.term\"></he-terms-units-description>\n            </th>\n          </tr>\n        </thead>\n        <tbody>\n          <ng-container *ngFor=\"let impactAssessment of impactAssessments(); trackBy: trackById; let i = index\">\n            <tr>\n              <td class=\"width-auto has-border-right\" [attr.title]=\"impactName(impactAssessment)\">\n                <he-node-link [node]=\"impactAssessment\">\n                  <span>{{ i + 1 }}. {{ impactName(impactAssessment) }}</span>\n                </he-node-link>\n              </td>\n              <td class=\"has-border-right\" [attr.title]=\"impactAssessment.product?.term?.units\">\n                <span>1 {{ impactAssessment.product?.term?.units }}</span>\n              </td>\n              <td class=\"has-border-right\" [attr.title]=\"impactAssessment.product?.term?.name\">\n                <he-node-link *ngIf=\"impactAssessment.product?.term\" [node]=\"impactAssessment.product.term\">\n                  <span>{{ impactAssessment.product.term.name | ellipsis: 30 }}</span>\n                </he-node-link>\n              </td>\n              <td class=\"is-nowrap\" *ngFor=\"let indicator of indicators()\">\n                <span\n                  *ngIf=\"indicator.value.values[impactAssessment['@id']]; else emptyValue\"\n                  class=\"trigger-popover\"\n                  [ngbPopover]=\"details\"\n                  [autoClose]=\"'outside'\"\n                  popoverClass=\"is-narrow\"\n                  triggers=\"manual\"\n                  #p=\"ngbPopover\"\n                  placement=\"left\"\n                  container=\"body\"\n                  (click)=\"\n                    togglePopover(p, {\n                      data: indicator.value.values[impactAssessment['@id']],\n                      impactAssessment: impactAssessment\n                    })\n                  \">\n                  <span pointer>\n                    {{\n                      propertyValue(indicator.value.values[impactAssessment['@id']].value, indicator.value.term['@id'])\n                        | precision: 3\n                        | default: '-'\n                    }}\n                  </span>\n                  <he-blank-node-state\n                    class=\"ml-1\"\n                    [dataState]=\"_dataState()\"\n                    [node]=\"indicator.value.values[impactAssessment['@id']].node\"\n                    key=\"value\"\n                    [state]=\"impactAssessment.aggregated ? 'aggregated' : undefined\"></he-blank-node-state>\n                </span>\n              </td>\n            </tr>\n          </ng-container>\n        </tbody>\n      </table>\n    </he-data-table>\n\n    <he-blank-node-state-notice [dataState]=\"_dataState()\"></he-blank-node-state-notice>\n  </ng-container>\n</div>\n\n<he-impact-assessments-indicator-breakdown-chart\n  *ngIf=\"selectedView === View.breakdown\"\n  [impactAssessment]=\"impactAssessments()[0]\"\n  [indicators]=\"impactAssessments()[0][_nodeKey()]\"></he-impact-assessments-indicator-breakdown-chart>\n\n<he-impact-assessments-indicators-chart\n  *ngIf=\"selectedView === View.chart && impactAssessments().length > 1\"\n  [key]=\"_nodeKey()\"\n  [filterTermTypes]=\"_filterTermTypes()\"></he-impact-assessments-indicators-chart>\n\n<ng-container *ngIf=\"selectedView === View.logs && !isOriginal()\">\n  <div class=\"field has-addons pt-2 px-3\" *ngIf=\"impactAssessments().length > 1\">\n    <div class=\"control\">\n      <span class=\"button is-small is-static is-secondary\">Select an Impact Assessment</span>\n    </div>\n    <div class=\"control is-expanded\">\n      <div class=\"select is-small is-fullwidth is-secondary\">\n        <select (change)=\"selectIndex($event)\">\n          <option *ngFor=\"let value of impactAssessments(); let i = index\" [value]=\"i\">\n            {{ i + 1 }}. {{ impactName(value) }}\n          </option>\n        </select>\n      </div>\n    </div>\n  </div>\n\n  <he-node-logs-models\n    *ngIf=\"selectedNode()\"\n    [node]=\"selectedNode()\"\n    [nodeKey]=\"_nodeKey()\"\n    [filterTermTypes]=\"_filterTermTypes()\"\n    [originalValues]=\"selectedOriginalValues()\"\n    [recalculatedValues]=\"selectedRecalculatedValues()\"\n    [noDataMessage]=\"noDataMessage()\"></he-node-logs-models>\n</ng-container>\n\n<he-node-csv-export-confirm\n  *ngIf=\"showDownload\"\n  [nodes]=\"impactAssessments()\"\n  [filename]=\"'impact-' + _nodeKey() + '.csv'\"\n  [isUpload]=\"false\"\n  [headerKeys]=\"['impactAssessment.id', 'impactAssessment.@id', 'impactAssessment.' + _nodeKey() + '.']\"\n  (closed)=\"showDownload = false\"></he-node-csv-export-confirm>\n\n<ng-template #emptyTable>\n  <div class=\"has-text-centered\">\n    <span>No data available</span>\n    <span class=\"is-pl-1\" [class.is-hidden]=\"!filterTerm()\">matching your search criteria</span>\n    <span>.</span>\n    <span [class.is-hidden]=\"!isOriginal()\">\n      Switch to\n      <code>recalculated</code>\n      version.\n    </span>\n  </div>\n</ng-template>\n\n<ng-template #emptyValue>\n  <span>-</span>\n</ng-template>\n\n<ng-template #details let-node=\"impactAssessment\" let-data=\"data\">\n  <p *bindOnce=\"node\">\n    <b>\n      <span *ngIf=\"data.cycle\">{{ cycleLabel(node.cycle) }}</span>\n      <span *ngIf=\"!data.cycle\">{{ data.name }}</span>\n    </b>\n  </p>\n  <he-node-value-details\n    [data]=\"data\"\n    [dataState]=\"_dataState()\"\n    [nodeType]=\"node['@type']\"\n    [dataKey]=\"_nodeKey()\"></he-node-value-details>\n</ng-template>\n\n<ng-template #suggestion let-impact=\"result\" let-t=\"term\">\n  <div class=\"is-block\">\n    <ngb-highlight [result]=\"impact.name || impact.cycle.name\" [term]=\"t\"></ngb-highlight>\n  </div>\n  <div class=\"columns is-flex\">\n    <div class=\"column\" *ngIf=\"impact.country\">\n      <span class=\"pr-1 has-text-underline\">Country:</span>\n      <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.country.name\" [term]=\"t\"></ngb-highlight></span>\n    </div>\n    <div class=\"column\" *ngIf=\"impact.product\">\n      <span class=\"pr-1 has-text-underline\">Product:</span>\n      <span class=\"is-inline-flex\">\n        <ngb-highlight [result]=\"impact.product.term?.name || impact.product.name\" [term]=\"t\"></ngb-highlight>\n      </span>\n    </div>\n    <div class=\"column\" *ngIf=\"impact.endDate\">\n      <span class=\"pr-1 has-text-underline\">Date:</span>\n      <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.endDate\" [term]=\"t\"></ngb-highlight></span>\n    </div>\n  </div>\n</ng-template>\n"]}