@hestia-earth/ui-components 0.23.5 → 0.23.7

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.
@@ -31,7 +31,7 @@ export declare class CyclesActivityComponent implements OnChanges {
31
31
  togglePopover(popover: any, context: any): any;
32
32
  private filterNode;
33
33
  filter(term: string): void;
34
- showSwitchToRecalculated(): boolean;
34
+ get showSwitchToRecalculated(): boolean;
35
35
  private updateSelectedIndex;
36
36
  selectIndex({ target: { value } }: {
37
37
  target: {
@@ -21,7 +21,7 @@ export declare class CyclesAnimalsComponent implements OnChanges {
21
21
  togglePopover(popover: any, context: any): any;
22
22
  private filterNode;
23
23
  filter(term: string): void;
24
- showSwitchToRecalculated(): boolean;
24
+ get showSwitchToRecalculated(): boolean;
25
25
  static ɵfac: i0.ɵɵFactoryDeclaration<CyclesAnimalsComponent, never>;
26
26
  static ɵcmp: i0.ɵɵComponentDeclaration<CyclesAnimalsComponent, "he-cycles-animals", never, { "originalValues": { "alias": "originalValues"; "required": false; }; "cycles": { "alias": "cycles"; "required": false; }; "dataState": { "alias": "dataState"; "required": false; }; }, {}, never, never, false, never>;
27
27
  }
@@ -38,7 +38,7 @@ export declare class CyclesEmissionsComponent implements OnChanges {
38
38
  get isTransformation(): boolean;
39
39
  private filterNode;
40
40
  filter(term: string): void;
41
- showSwitchToRecalculated(): boolean;
41
+ get showSwitchToRecalculated(): boolean;
42
42
  private updateSelectedIndex;
43
43
  selectIndex({ target: { value } }: {
44
44
  target: {
@@ -31,7 +31,7 @@ export declare class CyclesPracticesComponent implements OnChanges {
31
31
  get showTimeline(): boolean;
32
32
  private filterNode;
33
33
  filter(term: string): void;
34
- showSwitchToRecalculated(): boolean;
34
+ get showSwitchToRecalculated(): boolean;
35
35
  private updateSelectedIndex;
36
36
  selectIndex({ target: { value } }: {
37
37
  target: {
@@ -89,7 +89,7 @@ export class CyclesActivityComponent {
89
89
  return this.update();
90
90
  }
91
91
  // Recalculation logs
92
- showSwitchToRecalculated() {
92
+ get showSwitchToRecalculated() {
93
93
  return this.isOriginal && this.cycles.some(cycle => !cycle.aggregated);
94
94
  }
95
95
  updateSelectedIndex(index) {
@@ -115,4 +115,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.8", ngImpor
115
115
  }], enableCompare: [{
116
116
  type: Input
117
117
  }] } });
118
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cycles-activity.component.js","sourceRoot":"","sources":["../../../../src/cycles/cycles-activity/cycles-activity.component.ts","../../../../src/cycles/cycles-activity/cycles-activity.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnChanges, SimpleChanges } from '@angular/core';\nimport { DataState } from '@hestia-earth/api';\nimport { ICycleJSONLD, Input as HestiaInput, Product } from '@hestia-earth/schema';\nimport orderBy from 'lodash.orderby';\nimport { propertyValue } from '@hestia-earth/utils/dist/term';\n\nimport { groupNodesByTerm, IGroupedKeys, grouppedKeys } from '../../common/node-utils';\nimport { baseUrl, defaultLabel } from '../../common/utils';\n\nenum View {\n  table = 'table',\n  chart = 'chart',\n  logs = 'logs'\n}\n\nconst headerKeys = [\n  'cycle.id',\n  'cycle.@id',\n  'cycle.inputs.',\n  'cycle.products.',\n  'transformation.id',\n  'transformation.@id',\n  'transformation.inputs.',\n  'transformation.products.'\n];\n\n@Component({\n  selector: 'he-cycles-activity',\n  templateUrl: './cycles-activity.component.html',\n  styleUrls: ['./cycles-activity.component.scss']\n})\nexport class CyclesActivityComponent implements OnChanges {\n  @Input()\n  public originalValues: ICycleJSONLD[] = [];\n  @Input()\n  public cycles: ICycleJSONLD[] = [];\n  @Input()\n  public dataState?: DataState;\n  @Input()\n  public enableCompare = true;\n\n  public baseUrl = baseUrl();\n  public propertyValue = propertyValue;\n  public defaultLabel = defaultLabel;\n  public headerKeys = headerKeys;\n  public showDownload = false;\n  public View = View;\n  public selectedView = View.table;\n  public selectedIndex = 0;\n\n  public inputs: IGroupedKeys<HestiaInput>[] = [];\n  public products: IGroupedKeys<Product>[] = [];\n  public filterTerm = '';\n\n  ngOnChanges(changes: SimpleChanges) {\n    if ('dataState' in changes) {\n      this.selectedView = View.table;\n    }\n    if ('cycles' in changes) {\n      this.updateSelectedIndex(this.selectedIndex);\n      return this.update();\n    }\n  }\n\n  public trackById(_index: number, item: ICycleJSONLD) {\n    return item['@id'];\n  }\n\n  public get isOriginal() {\n    return this.dataState === DataState.original;\n  }\n\n  private update() {\n    const inputsPerCycle = groupNodesByTerm<ICycleJSONLD, HestiaInput>(\n      this.cycles,\n      'inputs',\n      this.originalValues,\n      node => this.filterNode(node)\n    );\n    this.inputs = orderBy(grouppedKeys(inputsPerCycle), ['key'], ['asc']);\n    const productsPerCycle = groupNodesByTerm<ICycleJSONLD, Product>(\n      this.cycles,\n      'products',\n      this.originalValues,\n      node => this.filterNode(node)\n    );\n    this.products = orderBy(grouppedKeys(productsPerCycle), ['key'], ['asc']);\n  }\n\n  public togglePopover(popover: any, context: any) {\n    return popover.isOpen() ? popover.close() : popover.open(context);\n  }\n\n  private filterNode({ term }: HestiaInput | Product) {\n    return !this.filterTerm || term?.name?.toLowerCase()?.includes(this.filterTerm.toLowerCase());\n  }\n\n  public filter(term: string) {\n    this.filterTerm = term;\n    return this.update();\n  }\n\n  // Recalculation logs\n\n  public showSwitchToRecalculated() {\n    return this.isOriginal && this.cycles.some(cycle => !cycle.aggregated);\n  }\n\n  private updateSelectedIndex(index: number) {\n    this.selectedIndex = -1;\n    // force a refresh of the logs\n    setTimeout(() => (this.selectedIndex = index));\n  }\n\n  public selectIndex({ target: { value } }) {\n    return this.updateSelectedIndex(+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 && (inputs.length || products.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 inputs & products by name\"\n      (search)=\"filter($event)\"></he-search-extend>\n  </div>\n  <div class=\"column is-narrow\">\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\">\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=\"!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=\"inputs.length || products.length; 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-bold\">\n            <th class=\"width-auto\"></th>\n            <th></th>\n            <th *ngIf=\"inputs.length\" [attr.colspan]=\"inputs.length\" [class.has-border-right]=\"products.length\">\n              Inputs\n            </th>\n            <th *ngIf=\"products.length\" [attr.colspan]=\"products.length\">Products</th>\n          </tr>\n          <tr class=\"has-text-weight-semibold\">\n            <th class=\"width-auto has-border-right\"></th>\n            <th class=\"has-border-right\"></th>\n            <th\n              *ngFor=\"let input of inputs; let pl = last\"\n              [attr.title]=\"input.value.term.name\"\n              [class.has-border-right]=\"products.length && pl\">\n              <he-node-link [node]=\"input.value.term\">\n                <span [innerHtml]=\"input.value.term.name | ellipsis: 30 | compound\"></span>\n              </he-node-link>\n            </th>\n            <th *ngFor=\"let product of products; let pl = last\" [attr.title]=\"product.value.term.name\">\n              <he-node-link [node]=\"product.value.term\">\n                <span [innerHtml]=\"product.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/Cycle#functionalUnit'\" target=\"_blank\">Functional unit</a>\n            </th>\n            <th\n              *ngFor=\"let input of inputs; let pl = last\"\n              [attr.title]=\"input.value.term.units\"\n              [class.has-border-right]=\"products.length && pl\">\n              <span [innerHtml]=\"input.value.term.units | compound\"></span>\n              <he-terms-units-description\n                class=\"is-inline-block is-ml-2\"\n                [term]=\"input.value.term\"></he-terms-units-description>\n            </th>\n            <th *ngFor=\"let product of products; let pl = last\" [attr.title]=\"product.value.term.units\">\n              <span [innerHtml]=\"product.value.term.units | compound\"></span>\n              <he-terms-units-description\n                class=\"is-inline-block is-ml-2\"\n                [term]=\"product.value.term\"></he-terms-units-description>\n            </th>\n          </tr>\n        </thead>\n        <tbody>\n          <ng-container *ngFor=\"let cycle of cycles; trackBy: trackById; let i = index\">\n            <tr>\n              <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n                <he-node-link [node]=\"cycle.term || cycle\">\n                  <span>{{ i + 1 }}. {{ defaultLabel(cycle) }}</span>\n                </he-node-link>\n              </td>\n              <td class=\"has-border-right\">\n                <he-cycles-functional-unit-measure [cycle]=\"cycle\"></he-cycles-functional-unit-measure>\n              </td>\n              <td\n                class=\"is-nowrap\"\n                *ngFor=\"let input of inputs; let pl = last\"\n                [class.has-border-right]=\"products.length && pl\">\n                <span\n                  *ngIf=\"input.value.values[cycle['@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)=\"togglePopover(p, { data: input.value.values[cycle['@id']], cycle: cycle, key: 'inputs' })\">\n                  <span pointer>\n                    {{\n                      propertyValue(input.value.values[cycle['@id']].value, input.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]=\"input.value.values[cycle['@id']].node\"\n                    key=\"value\"></he-blank-node-state>\n                </span>\n              </td>\n              <td class=\"is-nowrap\" *ngFor=\"let product of products; let pl = last\">\n                <span\n                  *ngIf=\"product.value.values[cycle['@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, { data: product.value.values[cycle['@id']], cycle: cycle, key: 'products' })\n                  \">\n                  <span pointer>\n                    {{\n                      propertyValue(product.value.values[cycle['@id']].value, product.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]=\"product.value.values[cycle['@id']].node\"\n                    key=\"value\"></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-cycles-result *ngIf=\"selectedView === View.chart\" [cycles]=\"cycles\"></he-cycles-result>\n\n<ng-container *ngIf=\"selectedView === View.logs && !isOriginal\">\n  <div class=\"field has-addons pt-2 px-3\" *ngIf=\"cycles.length > 1\">\n    <div class=\"control\">\n      <span class=\"button is-small is-static is-secondary\">Select a Cycle</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 cycles; let i = index\" [value]=\"i\">\n            {{ i + 1 }}. {{ defaultLabel(value) }}\n          </option>\n        </select>\n      </div>\n    </div>\n  </div>\n\n  <he-cycles-activity-logs\n    *ngIf=\"selectedIndex >= 0\"\n    [cycle]=\"cycles[selectedIndex]\"\n    [original]=\"originalValues[selectedIndex]\"\n    [recalculated]=\"cycles[selectedIndex]\"></he-cycles-activity-logs>\n</ng-container>\n\n<he-node-csv-export-confirm\n  *ngIf=\"showDownload\"\n  [nodes]=\"cycles\"\n  filename=\"inputs-products.csv\"\n  [isUpload]=\"false\"\n  [headerKeys]=\"headerKeys\"\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 *ngIf=\"showSwitchToRecalculated\">\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=\"cycle\" let-data=\"data\" let-key=\"key\">\n  <p>\n    <b>{{ defaultLabel(node) }}</b>\n  </p>\n  <he-node-value-details\n    [data]=\"data\"\n    [dataState]=\"dataState\"\n    [nodeType]=\"node['@type']\"\n    [dataKey]=\"key\"></he-node-value-details>\n</ng-template>\n"]}
118
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cycles-activity.component.js","sourceRoot":"","sources":["../../../../src/cycles/cycles-activity/cycles-activity.component.ts","../../../../src/cycles/cycles-activity/cycles-activity.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnChanges, SimpleChanges } from '@angular/core';\nimport { DataState } from '@hestia-earth/api';\nimport { ICycleJSONLD, Input as HestiaInput, Product } from '@hestia-earth/schema';\nimport orderBy from 'lodash.orderby';\nimport { propertyValue } from '@hestia-earth/utils/dist/term';\n\nimport { groupNodesByTerm, IGroupedKeys, grouppedKeys } from '../../common/node-utils';\nimport { baseUrl, defaultLabel } from '../../common/utils';\n\nenum View {\n  table = 'table',\n  chart = 'chart',\n  logs = 'logs'\n}\n\nconst headerKeys = [\n  'cycle.id',\n  'cycle.@id',\n  'cycle.inputs.',\n  'cycle.products.',\n  'transformation.id',\n  'transformation.@id',\n  'transformation.inputs.',\n  'transformation.products.'\n];\n\n@Component({\n  selector: 'he-cycles-activity',\n  templateUrl: './cycles-activity.component.html',\n  styleUrls: ['./cycles-activity.component.scss']\n})\nexport class CyclesActivityComponent implements OnChanges {\n  @Input()\n  public originalValues: ICycleJSONLD[] = [];\n  @Input()\n  public cycles: ICycleJSONLD[] = [];\n  @Input()\n  public dataState?: DataState;\n  @Input()\n  public enableCompare = true;\n\n  public baseUrl = baseUrl();\n  public propertyValue = propertyValue;\n  public defaultLabel = defaultLabel;\n  public headerKeys = headerKeys;\n  public showDownload = false;\n  public View = View;\n  public selectedView = View.table;\n  public selectedIndex = 0;\n\n  public inputs: IGroupedKeys<HestiaInput>[] = [];\n  public products: IGroupedKeys<Product>[] = [];\n  public filterTerm = '';\n\n  ngOnChanges(changes: SimpleChanges) {\n    if ('dataState' in changes) {\n      this.selectedView = View.table;\n    }\n    if ('cycles' in changes) {\n      this.updateSelectedIndex(this.selectedIndex);\n      return this.update();\n    }\n  }\n\n  public trackById(_index: number, item: ICycleJSONLD) {\n    return item['@id'];\n  }\n\n  public get isOriginal() {\n    return this.dataState === DataState.original;\n  }\n\n  private update() {\n    const inputsPerCycle = groupNodesByTerm<ICycleJSONLD, HestiaInput>(\n      this.cycles,\n      'inputs',\n      this.originalValues,\n      node => this.filterNode(node)\n    );\n    this.inputs = orderBy(grouppedKeys(inputsPerCycle), ['key'], ['asc']);\n    const productsPerCycle = groupNodesByTerm<ICycleJSONLD, Product>(\n      this.cycles,\n      'products',\n      this.originalValues,\n      node => this.filterNode(node)\n    );\n    this.products = orderBy(grouppedKeys(productsPerCycle), ['key'], ['asc']);\n  }\n\n  public togglePopover(popover: any, context: any) {\n    return popover.isOpen() ? popover.close() : popover.open(context);\n  }\n\n  private filterNode({ term }: HestiaInput | Product) {\n    return !this.filterTerm || term?.name?.toLowerCase()?.includes(this.filterTerm.toLowerCase());\n  }\n\n  public filter(term: string) {\n    this.filterTerm = term;\n    return this.update();\n  }\n\n  // Recalculation logs\n\n  public get showSwitchToRecalculated() {\n    return this.isOriginal && this.cycles.some(cycle => !cycle.aggregated);\n  }\n\n  private updateSelectedIndex(index: number) {\n    this.selectedIndex = -1;\n    // force a refresh of the logs\n    setTimeout(() => (this.selectedIndex = index));\n  }\n\n  public selectIndex({ target: { value } }) {\n    return this.updateSelectedIndex(+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 && (inputs.length || products.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 inputs & products by name\"\n      (search)=\"filter($event)\"></he-search-extend>\n  </div>\n  <div class=\"column is-narrow\">\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\">\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=\"!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=\"inputs.length || products.length; 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-bold\">\n            <th class=\"width-auto\"></th>\n            <th></th>\n            <th *ngIf=\"inputs.length\" [attr.colspan]=\"inputs.length\" [class.has-border-right]=\"products.length\">\n              Inputs\n            </th>\n            <th *ngIf=\"products.length\" [attr.colspan]=\"products.length\">Products</th>\n          </tr>\n          <tr class=\"has-text-weight-semibold\">\n            <th class=\"width-auto has-border-right\"></th>\n            <th class=\"has-border-right\"></th>\n            <th\n              *ngFor=\"let input of inputs; let pl = last\"\n              [attr.title]=\"input.value.term.name\"\n              [class.has-border-right]=\"products.length && pl\">\n              <he-node-link [node]=\"input.value.term\">\n                <span [innerHtml]=\"input.value.term.name | ellipsis: 30 | compound\"></span>\n              </he-node-link>\n            </th>\n            <th *ngFor=\"let product of products; let pl = last\" [attr.title]=\"product.value.term.name\">\n              <he-node-link [node]=\"product.value.term\">\n                <span [innerHtml]=\"product.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/Cycle#functionalUnit'\" target=\"_blank\">Functional unit</a>\n            </th>\n            <th\n              *ngFor=\"let input of inputs; let pl = last\"\n              [attr.title]=\"input.value.term.units\"\n              [class.has-border-right]=\"products.length && pl\">\n              <span [innerHtml]=\"input.value.term.units | compound\"></span>\n              <he-terms-units-description\n                class=\"is-inline-block is-ml-2\"\n                [term]=\"input.value.term\"></he-terms-units-description>\n            </th>\n            <th *ngFor=\"let product of products; let pl = last\" [attr.title]=\"product.value.term.units\">\n              <span [innerHtml]=\"product.value.term.units | compound\"></span>\n              <he-terms-units-description\n                class=\"is-inline-block is-ml-2\"\n                [term]=\"product.value.term\"></he-terms-units-description>\n            </th>\n          </tr>\n        </thead>\n        <tbody>\n          <ng-container *ngFor=\"let cycle of cycles; trackBy: trackById; let i = index\">\n            <tr>\n              <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n                <he-node-link [node]=\"cycle.term || cycle\">\n                  <span>{{ i + 1 }}. {{ defaultLabel(cycle) }}</span>\n                </he-node-link>\n              </td>\n              <td class=\"has-border-right\">\n                <he-cycles-functional-unit-measure [cycle]=\"cycle\"></he-cycles-functional-unit-measure>\n              </td>\n              <td\n                class=\"is-nowrap\"\n                *ngFor=\"let input of inputs; let pl = last\"\n                [class.has-border-right]=\"products.length && pl\">\n                <span\n                  *ngIf=\"input.value.values[cycle['@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)=\"togglePopover(p, { data: input.value.values[cycle['@id']], cycle: cycle, key: 'inputs' })\">\n                  <span pointer>\n                    {{\n                      propertyValue(input.value.values[cycle['@id']].value, input.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]=\"input.value.values[cycle['@id']].node\"\n                    key=\"value\"></he-blank-node-state>\n                </span>\n              </td>\n              <td class=\"is-nowrap\" *ngFor=\"let product of products; let pl = last\">\n                <span\n                  *ngIf=\"product.value.values[cycle['@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, { data: product.value.values[cycle['@id']], cycle: cycle, key: 'products' })\n                  \">\n                  <span pointer>\n                    {{\n                      propertyValue(product.value.values[cycle['@id']].value, product.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]=\"product.value.values[cycle['@id']].node\"\n                    key=\"value\"></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-cycles-result *ngIf=\"selectedView === View.chart\" [cycles]=\"cycles\"></he-cycles-result>\n\n<ng-container *ngIf=\"selectedView === View.logs && !isOriginal\">\n  <div class=\"field has-addons pt-2 px-3\" *ngIf=\"cycles.length > 1\">\n    <div class=\"control\">\n      <span class=\"button is-small is-static is-secondary\">Select a Cycle</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 cycles; let i = index\" [value]=\"i\">\n            {{ i + 1 }}. {{ defaultLabel(value) }}\n          </option>\n        </select>\n      </div>\n    </div>\n  </div>\n\n  <he-cycles-activity-logs\n    *ngIf=\"selectedIndex >= 0\"\n    [cycle]=\"cycles[selectedIndex]\"\n    [original]=\"originalValues[selectedIndex]\"\n    [recalculated]=\"cycles[selectedIndex]\"></he-cycles-activity-logs>\n</ng-container>\n\n<he-node-csv-export-confirm\n  *ngIf=\"showDownload\"\n  [nodes]=\"cycles\"\n  filename=\"inputs-products.csv\"\n  [isUpload]=\"false\"\n  [headerKeys]=\"headerKeys\"\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 *ngIf=\"showSwitchToRecalculated\">\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=\"cycle\" let-data=\"data\" let-key=\"key\">\n  <p>\n    <b>{{ defaultLabel(node) }}</b>\n  </p>\n  <he-node-value-details\n    [data]=\"data\"\n    [dataState]=\"dataState\"\n    [nodeType]=\"node['@type']\"\n    [dataKey]=\"key\"></he-node-value-details>\n</ng-template>\n"]}
@@ -60,7 +60,7 @@ export class CyclesAnimalsComponent {
60
60
  return this.update();
61
61
  }
62
62
  // Recalculation logs
63
- showSwitchToRecalculated() {
63
+ get showSwitchToRecalculated() {
64
64
  return this.isOriginal && this.cycles.some(cycle => !cycle.aggregated);
65
65
  }
66
66
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.8", ngImport: i0, type: CyclesAnimalsComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
@@ -76,4 +76,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.8", ngImpor
76
76
  }], dataState: [{
77
77
  type: Input
78
78
  }] } });
79
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cycles-animals.component.js","sourceRoot":"","sources":["../../../../src/cycles/cycles-animals/cycles-animals.component.ts","../../../../src/cycles/cycles-animals/cycles-animals.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAA4B,MAAM,eAAe,CAAC;AAC3E,OAAO,EAAE,SAAS,EAAE,MAAM,mBAAmB,CAAC;AAE9C,OAAO,EAAE,aAAa,EAAE,MAAM,+BAA+B,CAAC;AAC9D,OAAO,OAAO,MAAM,gBAAgB,CAAC;AAErC,OAAO,EAAE,gBAAgB,EAAgB,YAAY,EAAE,MAAM,yBAAyB,CAAC;AACvF,OAAO,EAAE,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;;;;;;;;;;;;;;;;;;AAE3D,MAAM,UAAU,GAAG,CAAC,UAAU,EAAE,WAAW,EAAE,gBAAgB,CAAC,CAAC;AAO/D,MAAM,OAAO,sBAAsB;IALnC;QAOS,mBAAc,GAAmB,EAAE,CAAC;QAEpC,WAAM,GAAmB,EAAE,CAAC;QAI5B,YAAO,GAAG,OAAO,EAAE,CAAC;QACpB,kBAAa,GAAG,aAAa,CAAC;QAC9B,iBAAY,GAAG,YAAY,CAAC;QAC5B,eAAU,GAAG,UAAU,CAAC;QACxB,iBAAY,GAAG,KAAK,CAAC;QAErB,YAAO,GAA2B,EAAE,CAAC;QACrC,eAAU,GAAG,EAAE,CAAC;KAyCxB;IAvCC,WAAW,CAAC,OAAsB;QAChC,IAAI,QAAQ,IAAI,OAAO,EAAE;YACvB,OAAO,IAAI,CAAC,MAAM,EAAE,CAAC;SACtB;IACH,CAAC;IAEM,SAAS,CAAC,MAAc,EAAE,IAAkB;QACjD,OAAO,IAAI,CAAC,KAAK,CAAC,CAAC;IACrB,CAAC;IAED,IAAW,UAAU;QACnB,OAAO,IAAI,CAAC,SAAS,KAAK,SAAS,CAAC,QAAQ,CAAC;IAC/C,CAAC;IAEO,MAAM;QACZ,MAAM,eAAe,GAAG,gBAAgB,CAAuB,IAAI,CAAC,MAAM,EAAE,SAAS,EAAE,IAAI,CAAC,cAAc,EAAE,IAAI,CAAC,EAAE,CACjH,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,CACtB,CAAC;QACF,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC,YAAY,CAAC,eAAe,CAAC,EAAE,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC;IAC1E,CAAC;IAEM,aAAa,CAAC,OAAY,EAAE,OAAY;QAC7C,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACpE,CAAC;IAEO,UAAU,CAAC,EAAE,IAAI,EAAU;QACjC,OAAO,CAAC,IAAI,CAAC,UAAU,IAAI,IAAI,EAAE,IAAI,EAAE,WAAW,EAAE,EAAE,QAAQ,CAAC,IAAI,CAAC,UAAU,CAAC,WAAW,EAAE,CAAC,CAAC;IAChG,CAAC;IAEM,MAAM,CAAC,IAAY;QACxB,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;QACvB,OAAO,IAAI,CAAC,MAAM,EAAE,CAAC;IACvB,CAAC;IAED,qBAAqB;IAEd,wBAAwB;QAC7B,OAAO,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;IACzE,CAAC;8GAvDU,sBAAsB;kGAAtB,sBAAsB,sKChBnC,i8JAiIA;;2FDjHa,sBAAsB;kBALlC,SAAS;+BACE,mBAAmB;8BAMtB,cAAc;sBADpB,KAAK;gBAGC,MAAM;sBADZ,KAAK;gBAGC,SAAS;sBADf,KAAK","sourcesContent":["import { Component, Input, OnChanges, SimpleChanges } from '@angular/core';\nimport { DataState } from '@hestia-earth/api';\nimport { Animal, ICycleJSONLD } from '@hestia-earth/schema';\nimport { propertyValue } from '@hestia-earth/utils/dist/term';\nimport orderBy from 'lodash.orderby';\n\nimport { groupNodesByTerm, IGroupedKeys, grouppedKeys } from '../../common/node-utils';\nimport { baseUrl, defaultLabel } from '../../common/utils';\n\nconst headerKeys = ['cycle.id', 'cycle.@id', 'cycle.animals.'];\n\n@Component({\n  selector: 'he-cycles-animals',\n  templateUrl: './cycles-animals.component.html',\n  styleUrls: ['./cycles-animals.component.scss']\n})\nexport class CyclesAnimalsComponent implements OnChanges {\n  @Input()\n  public originalValues: ICycleJSONLD[] = [];\n  @Input()\n  public cycles: ICycleJSONLD[] = [];\n  @Input()\n  public dataState?: DataState;\n\n  public baseUrl = baseUrl();\n  public propertyValue = propertyValue;\n  public defaultLabel = defaultLabel;\n  public headerKeys = headerKeys;\n  public showDownload = false;\n\n  public animals: IGroupedKeys<Animal>[] = [];\n  public filterTerm = '';\n\n  ngOnChanges(changes: SimpleChanges) {\n    if ('cycles' in changes) {\n      return this.update();\n    }\n  }\n\n  public trackById(_index: number, item: ICycleJSONLD) {\n    return item['@id'];\n  }\n\n  public get isOriginal() {\n    return this.dataState === DataState.original;\n  }\n\n  private update() {\n    const animalsPerCycle = groupNodesByTerm<ICycleJSONLD, Animal>(this.cycles, 'animals', this.originalValues, node =>\n      this.filterNode(node)\n    );\n    this.animals = orderBy(grouppedKeys(animalsPerCycle), ['key'], ['asc']);\n  }\n\n  public togglePopover(popover: any, context: any) {\n    return popover.isOpen() ? popover.close() : popover.open(context);\n  }\n\n  private filterNode({ term }: Animal) {\n    return !this.filterTerm || term?.name?.toLowerCase()?.includes(this.filterTerm.toLowerCase());\n  }\n\n  public filter(term: string) {\n    this.filterTerm = term;\n    return this.update();\n  }\n\n  // Recalculation logs\n\n  public showSwitchToRecalculated() {\n    return this.isOriginal && this.cycles.some(cycle => !cycle.aggregated);\n  }\n}\n","<div class=\"columns is-variable is-align-items-center is-2 m-0\">\n  <div class=\"column\">\n    <ng-container *ngIf=\"animals.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\">\n    <he-search-extend\n      class=\"is-secondary is-small\"\n      placeholder=\"Filter animals by name\"\n      (search)=\"filter($event)\"></he-search-extend>\n  </div>\n</div>\n\n<div class=\"px-3 pb-3\">\n  <ng-container *ngIf=\"animals.length; 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 has-border-right\"></th>\n            <th class=\"has-border-right\"></th>\n            <th *ngFor=\"let animal of animals\" [attr.title]=\"animal.value.term.name\">\n              <he-node-link [node]=\"animal.value.term\">\n                <span [innerHtml]=\"animal.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/Cycle#functionalUnit'\" target=\"_blank\">Functional unit</a>\n            </th>\n            <th *ngFor=\"let animal of animals\" [attr.title]=\"animal.value.term.units\">\n              <span [innerHtml]=\"animal.value.term.units | compound\"></span>\n              <he-terms-units-description\n                class=\"is-inline-block is-ml-2\"\n                [term]=\"animal.value.term\"></he-terms-units-description>\n            </th>\n          </tr>\n        </thead>\n        <tbody>\n          <ng-container *ngFor=\"let cycle of cycles; trackBy: trackById; let i = index\">\n            <tr>\n              <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n                <he-node-link [node]=\"cycle.term || cycle\">\n                  <span>{{ i + 1 }}. {{ defaultLabel(cycle) }}</span>\n                </he-node-link>\n              </td>\n              <td class=\"has-border-right\">\n                <he-cycles-functional-unit-measure [cycle]=\"cycles[0]\"></he-cycles-functional-unit-measure>\n              </td>\n              <td class=\"is-nowrap\" *ngFor=\"let animal of animals\">\n                <span\n                  *ngIf=\"animal.value.values[cycle['@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)=\"togglePopover(p, { data: animal.value.values[cycle['@id']], cycle: cycle, key: 'animals' })\">\n                  <span pointer>\n                    {{\n                      propertyValue(animal.value.values[cycle['@id']].value, animal.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]=\"animal.value.values[cycle['@id']].node\"\n                    key=\"value\"></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-node-csv-export-confirm\n  *ngIf=\"showDownload\"\n  [nodes]=\"cycles\"\n  filename=\"animals.csv\"\n  [isUpload]=\"false\"\n  [headerKeys]=\"headerKeys\"\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 *ngIf=\"showSwitchToRecalculated\">\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=\"cycle\" let-data=\"data\" let-key=\"key\">\n  <p>\n    <b>{{ defaultLabel(node) }}</b>\n  </p>\n  <he-node-value-details\n    [data]=\"data\"\n    [dataState]=\"dataState\"\n    [nodeType]=\"node['@type']\"\n    [dataKey]=\"key\"></he-node-value-details>\n</ng-template>\n"]}
79
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cycles-animals.component.js","sourceRoot":"","sources":["../../../../src/cycles/cycles-animals/cycles-animals.component.ts","../../../../src/cycles/cycles-animals/cycles-animals.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAA4B,MAAM,eAAe,CAAC;AAC3E,OAAO,EAAE,SAAS,EAAE,MAAM,mBAAmB,CAAC;AAE9C,OAAO,EAAE,aAAa,EAAE,MAAM,+BAA+B,CAAC;AAC9D,OAAO,OAAO,MAAM,gBAAgB,CAAC;AAErC,OAAO,EAAE,gBAAgB,EAAgB,YAAY,EAAE,MAAM,yBAAyB,CAAC;AACvF,OAAO,EAAE,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;;;;;;;;;;;;;;;;;;AAE3D,MAAM,UAAU,GAAG,CAAC,UAAU,EAAE,WAAW,EAAE,gBAAgB,CAAC,CAAC;AAO/D,MAAM,OAAO,sBAAsB;IALnC;QAOS,mBAAc,GAAmB,EAAE,CAAC;QAEpC,WAAM,GAAmB,EAAE,CAAC;QAI5B,YAAO,GAAG,OAAO,EAAE,CAAC;QACpB,kBAAa,GAAG,aAAa,CAAC;QAC9B,iBAAY,GAAG,YAAY,CAAC;QAC5B,eAAU,GAAG,UAAU,CAAC;QACxB,iBAAY,GAAG,KAAK,CAAC;QAErB,YAAO,GAA2B,EAAE,CAAC;QACrC,eAAU,GAAG,EAAE,CAAC;KAyCxB;IAvCC,WAAW,CAAC,OAAsB;QAChC,IAAI,QAAQ,IAAI,OAAO,EAAE;YACvB,OAAO,IAAI,CAAC,MAAM,EAAE,CAAC;SACtB;IACH,CAAC;IAEM,SAAS,CAAC,MAAc,EAAE,IAAkB;QACjD,OAAO,IAAI,CAAC,KAAK,CAAC,CAAC;IACrB,CAAC;IAED,IAAW,UAAU;QACnB,OAAO,IAAI,CAAC,SAAS,KAAK,SAAS,CAAC,QAAQ,CAAC;IAC/C,CAAC;IAEO,MAAM;QACZ,MAAM,eAAe,GAAG,gBAAgB,CAAuB,IAAI,CAAC,MAAM,EAAE,SAAS,EAAE,IAAI,CAAC,cAAc,EAAE,IAAI,CAAC,EAAE,CACjH,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,CACtB,CAAC;QACF,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC,YAAY,CAAC,eAAe,CAAC,EAAE,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC;IAC1E,CAAC;IAEM,aAAa,CAAC,OAAY,EAAE,OAAY;QAC7C,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACpE,CAAC;IAEO,UAAU,CAAC,EAAE,IAAI,EAAU;QACjC,OAAO,CAAC,IAAI,CAAC,UAAU,IAAI,IAAI,EAAE,IAAI,EAAE,WAAW,EAAE,EAAE,QAAQ,CAAC,IAAI,CAAC,UAAU,CAAC,WAAW,EAAE,CAAC,CAAC;IAChG,CAAC;IAEM,MAAM,CAAC,IAAY;QACxB,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;QACvB,OAAO,IAAI,CAAC,MAAM,EAAE,CAAC;IACvB,CAAC;IAED,qBAAqB;IAErB,IAAW,wBAAwB;QACjC,OAAO,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;IACzE,CAAC;8GAvDU,sBAAsB;kGAAtB,sBAAsB,sKChBnC,i8JAiIA;;2FDjHa,sBAAsB;kBALlC,SAAS;+BACE,mBAAmB;8BAMtB,cAAc;sBADpB,KAAK;gBAGC,MAAM;sBADZ,KAAK;gBAGC,SAAS;sBADf,KAAK","sourcesContent":["import { Component, Input, OnChanges, SimpleChanges } from '@angular/core';\nimport { DataState } from '@hestia-earth/api';\nimport { Animal, ICycleJSONLD } from '@hestia-earth/schema';\nimport { propertyValue } from '@hestia-earth/utils/dist/term';\nimport orderBy from 'lodash.orderby';\n\nimport { groupNodesByTerm, IGroupedKeys, grouppedKeys } from '../../common/node-utils';\nimport { baseUrl, defaultLabel } from '../../common/utils';\n\nconst headerKeys = ['cycle.id', 'cycle.@id', 'cycle.animals.'];\n\n@Component({\n  selector: 'he-cycles-animals',\n  templateUrl: './cycles-animals.component.html',\n  styleUrls: ['./cycles-animals.component.scss']\n})\nexport class CyclesAnimalsComponent implements OnChanges {\n  @Input()\n  public originalValues: ICycleJSONLD[] = [];\n  @Input()\n  public cycles: ICycleJSONLD[] = [];\n  @Input()\n  public dataState?: DataState;\n\n  public baseUrl = baseUrl();\n  public propertyValue = propertyValue;\n  public defaultLabel = defaultLabel;\n  public headerKeys = headerKeys;\n  public showDownload = false;\n\n  public animals: IGroupedKeys<Animal>[] = [];\n  public filterTerm = '';\n\n  ngOnChanges(changes: SimpleChanges) {\n    if ('cycles' in changes) {\n      return this.update();\n    }\n  }\n\n  public trackById(_index: number, item: ICycleJSONLD) {\n    return item['@id'];\n  }\n\n  public get isOriginal() {\n    return this.dataState === DataState.original;\n  }\n\n  private update() {\n    const animalsPerCycle = groupNodesByTerm<ICycleJSONLD, Animal>(this.cycles, 'animals', this.originalValues, node =>\n      this.filterNode(node)\n    );\n    this.animals = orderBy(grouppedKeys(animalsPerCycle), ['key'], ['asc']);\n  }\n\n  public togglePopover(popover: any, context: any) {\n    return popover.isOpen() ? popover.close() : popover.open(context);\n  }\n\n  private filterNode({ term }: Animal) {\n    return !this.filterTerm || term?.name?.toLowerCase()?.includes(this.filterTerm.toLowerCase());\n  }\n\n  public filter(term: string) {\n    this.filterTerm = term;\n    return this.update();\n  }\n\n  // Recalculation logs\n\n  public get showSwitchToRecalculated() {\n    return this.isOriginal && this.cycles.some(cycle => !cycle.aggregated);\n  }\n}\n","<div class=\"columns is-variable is-align-items-center is-2 m-0\">\n  <div class=\"column\">\n    <ng-container *ngIf=\"animals.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\">\n    <he-search-extend\n      class=\"is-secondary is-small\"\n      placeholder=\"Filter animals by name\"\n      (search)=\"filter($event)\"></he-search-extend>\n  </div>\n</div>\n\n<div class=\"px-3 pb-3\">\n  <ng-container *ngIf=\"animals.length; 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 has-border-right\"></th>\n            <th class=\"has-border-right\"></th>\n            <th *ngFor=\"let animal of animals\" [attr.title]=\"animal.value.term.name\">\n              <he-node-link [node]=\"animal.value.term\">\n                <span [innerHtml]=\"animal.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/Cycle#functionalUnit'\" target=\"_blank\">Functional unit</a>\n            </th>\n            <th *ngFor=\"let animal of animals\" [attr.title]=\"animal.value.term.units\">\n              <span [innerHtml]=\"animal.value.term.units | compound\"></span>\n              <he-terms-units-description\n                class=\"is-inline-block is-ml-2\"\n                [term]=\"animal.value.term\"></he-terms-units-description>\n            </th>\n          </tr>\n        </thead>\n        <tbody>\n          <ng-container *ngFor=\"let cycle of cycles; trackBy: trackById; let i = index\">\n            <tr>\n              <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n                <he-node-link [node]=\"cycle.term || cycle\">\n                  <span>{{ i + 1 }}. {{ defaultLabel(cycle) }}</span>\n                </he-node-link>\n              </td>\n              <td class=\"has-border-right\">\n                <he-cycles-functional-unit-measure [cycle]=\"cycles[0]\"></he-cycles-functional-unit-measure>\n              </td>\n              <td class=\"is-nowrap\" *ngFor=\"let animal of animals\">\n                <span\n                  *ngIf=\"animal.value.values[cycle['@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)=\"togglePopover(p, { data: animal.value.values[cycle['@id']], cycle: cycle, key: 'animals' })\">\n                  <span pointer>\n                    {{\n                      propertyValue(animal.value.values[cycle['@id']].value, animal.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]=\"animal.value.values[cycle['@id']].node\"\n                    key=\"value\"></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-node-csv-export-confirm\n  *ngIf=\"showDownload\"\n  [nodes]=\"cycles\"\n  filename=\"animals.csv\"\n  [isUpload]=\"false\"\n  [headerKeys]=\"headerKeys\"\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 *ngIf=\"showSwitchToRecalculated\">\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=\"cycle\" let-data=\"data\" let-key=\"key\">\n  <p>\n    <b>{{ defaultLabel(node) }}</b>\n  </p>\n  <he-node-value-details\n    [data]=\"data\"\n    [dataState]=\"dataState\"\n    [nodeType]=\"node['@type']\"\n    [dataKey]=\"key\"></he-node-value-details>\n</ng-template>\n"]}
@@ -100,7 +100,7 @@ export class CyclesEmissionsComponent {
100
100
  return this.update();
101
101
  }
102
102
  // Recalculation logs
103
- showSwitchToRecalculated() {
103
+ get showSwitchToRecalculated() {
104
104
  return this.isOriginal && this.cycles.some(cycle => !cycle.aggregated);
105
105
  }
106
106
  updateSelectedIndex(index) {
@@ -124,4 +124,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.8", ngImpor
124
124
  }], dataState: [{
125
125
  type: Input
126
126
  }] } });
127
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cycles-emissions.component.js","sourceRoot":"","sources":["../../../../src/cycles/cycles-emissions/cycles-emissions.component.ts","../../../../src/cycles/cycles-emissions/cycles-emissions.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAA4B,MAAM,eAAe,CAAC;AAC3E,OAAO,EAAE,SAAS,EAAE,MAAM,mBAAmB,CAAC;AAC9C,OAAO,EAA0B,QAAQ,EAAE,UAAU,EAAsB,MAAM,sBAAsB,CAAC;AACxG,OAAO,EAAE,aAAa,EAAE,MAAM,+BAA+B,CAAC;AAC9D,OAAO,OAAO,MAAM,gBAAgB,CAAC;AAErC,OAAO,EAAE,gBAAgB,EAAgB,YAAY,EAAE,eAAe,EAAE,MAAM,yBAAyB,CAAC;AACxG,OAAO,EAAE,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;;;;;;;;;;;;;;;;;;;;;;AAE3D,IAAK,IAIJ;AAJD,WAAK,IAAI;IACP,uBAAe,CAAA;IACf,uBAAe,CAAA;IACf,qBAAa,CAAA;AACf,CAAC,EAJI,IAAI,KAAJ,IAAI,QAIR;AAED,MAAM,UAAU,GAAG;IACjB,UAAU;IACV,WAAW;IACX,kBAAkB;IAClB,mBAAmB;IACnB,oBAAoB;IACpB,2BAA2B;CAC5B,CAAC;AAiBF,MAAM,iBAAiB,GAAG,CAAC,SAAmC,EAAE,EAAE,CAChE,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE;IACxB,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,UAAU,CAAC,IAAI;QAC/C,eAAe,EAAE,eAAe,CAAC,CAAC,CAAC,KAAK,CAAC,UAAU,CAAC;QACpD,SAAS,EAAE,EAAE;KACd,CAAC;IACF,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IACxC,OAAO,CAAC,CAAC;AACX,CAAC,EAAE,EAAqB,CAAC,CAAC;AAO5B,MAAM,OAAO,wBAAwB;IALrC;QAOS,mBAAc,GAAmB,EAAE,CAAC;QAEpC,WAAM,GAAmB,EAAE,CAAC;QAI5B,YAAO,GAAG,OAAO,EAAE,CAAC;QACpB,kBAAa,GAAG,aAAa,CAAC;QAC9B,iBAAY,GAAG,YAAY,CAAC;QAC5B,eAAU,GAAG,UAAU,CAAC;QACxB,iBAAY,GAAG,KAAK,CAAC;QACrB,SAAI,GAAG,IAAI,CAAC;QACZ,iBAAY,GAAG,IAAI,CAAC,KAAK,CAAC;QAC1B,kBAAa,GAAG,CAAC,CAAC;QAElB,iBAAY,GAAG,KAAK,CAAC;QAErB,eAAU,GAAG,EAAE,CAAC;KAkExB;IAhEC,WAAW,CAAC,OAAsB;QAChC,IAAI,WAAW,IAAI,OAAO,EAAE;YAC1B,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,KAAK,CAAC;SAChC;QACD,IAAI,QAAQ,IAAI,OAAO,EAAE;YACvB,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;YAC7C,OAAO,IAAI,CAAC,MAAM,EAAE,CAAC;SACtB;IACH,CAAC;IAEM,SAAS,CAAC,MAAc,EAAE,IAAkB;QACjD,OAAO,IAAI,CAAC,KAAK,CAAC,CAAC;IACrB,CAAC;IAED,IAAW,UAAU;QACnB,OAAO,IAAI,CAAC,SAAS,KAAK,SAAS,CAAC,QAAQ,CAAC;IAC/C,CAAC;IAEO,MAAM;QACZ,MAAM,iBAAiB,GAAG,gBAAgB,CACxC,IAAI,CAAC,MAAM,EACX,WAAW,EACX,IAAI,CAAC,cAAc,EACnB,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,CAC9B,CAAC;QACF,MAAM,SAAS,GAAG,OAAO,CAAC,YAAY,CAAC,iBAAiB,CAAC,EAAE,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC;QAC7E,IAAI,CAAC,YAAY,GAAG,SAAS,CAAC,MAAM,GAAG,CAAC,CAAC;QACzC,IAAI,CAAC,SAAS,GAAG,OAAO,CAAC,YAAY,CAAC,iBAAiB,CAAC,SAAS,CAAC,CAAC,EAAE,CAAC,uBAAuB,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC;IAC3G,CAAC;IAEM,aAAa,CAAC,OAAY,EAAE,OAAY;QAC7C,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACpE,CAAC;IAED,IAAW,gBAAgB;QACzB,MAAM,SAAS,GAAG,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,EAAE,OAAO,EAAE,QAAQ,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACpE,MAAM,QAAQ,GAAI,SAAiB,CAAC,IAAI,IAAI,SAAS,CAAC,OAAO,CAAC,CAAC;QAC/D,OAAO,QAAQ,KAAK,UAAU,CAAC,cAAc,CAAC;IAChD,CAAC;IAEO,UAAU,CAAC,EAAE,IAAI,EAAY;QACnC,OAAO,CAAC,IAAI,CAAC,UAAU,IAAI,IAAI,EAAE,IAAI,EAAE,WAAW,EAAE,EAAE,QAAQ,CAAC,IAAI,CAAC,UAAU,CAAC,WAAW,EAAE,CAAC,CAAC;IAChG,CAAC;IAEM,MAAM,CAAC,IAAY;QACxB,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;QACvB,OAAO,IAAI,CAAC,MAAM,EAAE,CAAC;IACvB,CAAC;IAED,qBAAqB;IAEd,wBAAwB;QAC7B,OAAO,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;IACzE,CAAC;IAEO,mBAAmB,CAAC,KAAa;QACvC,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC,CAAC;QACxB,8BAA8B;QAC9B,UAAU,CAAC,GAAG,EAAE,CAAC,CAAC,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC,CAAC,CAAC;IACjD,CAAC;IAEM,WAAW,CAAC,EAAE,MAAM,EAAE,EAAE,KAAK,EAAE,EAAE;QACtC,OAAO,IAAI,CAAC,mBAAmB,CAAC,CAAC,KAAK,CAAC,CAAC;IAC1C,CAAC;8GApFU,wBAAwB;kGAAxB,wBAAwB,wKCtDrC,siSA4NA;;2FDtKa,wBAAwB;kBALpC,SAAS;+BACE,qBAAqB;8BAMxB,cAAc;sBADpB,KAAK;gBAGC,MAAM;sBADZ,KAAK;gBAGC,SAAS;sBADf,KAAK","sourcesContent":["import { Component, Input, OnChanges, SimpleChanges } from '@angular/core';\nimport { DataState } from '@hestia-earth/api';\nimport { ICycleJSONLD, Emission, NodeType, SchemaType, EmissionMethodTier } from '@hestia-earth/schema';\nimport { propertyValue } from '@hestia-earth/utils/dist/term';\nimport orderBy from 'lodash.orderby';\n\nimport { groupNodesByTerm, IGroupedKeys, grouppedKeys, methodTierOrder } from '../../common/node-utils';\nimport { baseUrl, defaultLabel } from '../../common/utils';\n\nenum View {\n  table = 'table',\n  chart = 'chart',\n  logs = 'logs'\n}\n\nconst headerKeys = [\n  'cycle.id',\n  'cycle.@id',\n  'cycle.emissions.',\n  'transformation.id',\n  'transformation.@id',\n  'transformation.emissions.'\n];\n\ntype groupedEmission = {\n  [methodTier in EmissionMethodTier]: {\n    methodTierOrder: number;\n    emissions: IGroupedKeys<Emission>[];\n  };\n};\n\ninterface IGroupedEmissionByKey {\n  key: string;\n  value: {\n    methodTierOrder: number;\n    emissions: IGroupedKeys<Emission>[];\n  };\n}\n\nconst groupByMethodTier = (emissions: IGroupedKeys<Emission>[]) =>\n  emissions.reduce((p, e) => {\n    p[e.value.methodTier] = p[e.value.methodTier] || {\n      methodTierOrder: methodTierOrder(e.value.methodTier),\n      emissions: []\n    };\n    p[e.value.methodTier].emissions.push(e);\n    return p;\n  }, {} as groupedEmission);\n\n@Component({\n  selector: 'he-cycles-emissions',\n  templateUrl: './cycles-emissions.component.html',\n  styleUrls: ['./cycles-emissions.component.scss']\n})\nexport class CyclesEmissionsComponent implements OnChanges {\n  @Input()\n  public originalValues: ICycleJSONLD[] = [];\n  @Input()\n  public cycles: ICycleJSONLD[] = [];\n  @Input()\n  public dataState?: DataState;\n\n  public baseUrl = baseUrl();\n  public propertyValue = propertyValue;\n  public defaultLabel = defaultLabel;\n  public headerKeys = headerKeys;\n  public showDownload = false;\n  public View = View;\n  public selectedView = View.table;\n  public selectedIndex = 0;\n\n  public hasEmissions = false;\n  public emissions: IGroupedEmissionByKey[];\n  public filterTerm = '';\n\n  ngOnChanges(changes: SimpleChanges) {\n    if ('dataState' in changes) {\n      this.selectedView = View.table;\n    }\n    if ('cycles' in changes) {\n      this.updateSelectedIndex(this.selectedIndex);\n      return this.update();\n    }\n  }\n\n  public trackById(_index: number, item: ICycleJSONLD) {\n    return item['@id'];\n  }\n\n  public get isOriginal() {\n    return this.dataState === DataState.original;\n  }\n\n  private update() {\n    const emissionsPerCycle = groupNodesByTerm<ICycleJSONLD, Emission>(\n      this.cycles,\n      'emissions',\n      this.originalValues,\n      node => this.filterNode(node)\n    );\n    const emissions = orderBy(grouppedKeys(emissionsPerCycle), ['key'], ['asc']);\n    this.hasEmissions = emissions.length > 0;\n    this.emissions = orderBy(grouppedKeys(groupByMethodTier(emissions)), ['value.methodTierOrder'], ['asc']);\n  }\n\n  public togglePopover(popover: any, context: any) {\n    return popover.isOpen() ? popover.close() : popover.open(context);\n  }\n\n  public get isTransformation() {\n    const firstNode = (this.cycles || [{ '@type': NodeType.Cycle }])[0];\n    const nodeType = (firstNode as any).type || firstNode['@type'];\n    return nodeType === SchemaType.Transformation;\n  }\n\n  private filterNode({ term }: Emission) {\n    return !this.filterTerm || term?.name?.toLowerCase()?.includes(this.filterTerm.toLowerCase());\n  }\n\n  public filter(term: string) {\n    this.filterTerm = term;\n    return this.update();\n  }\n\n  // Recalculation logs\n\n  public showSwitchToRecalculated() {\n    return this.isOriginal && this.cycles.some(cycle => !cycle.aggregated);\n  }\n\n  private updateSelectedIndex(index: number) {\n    this.selectedIndex = -1;\n    // force a refresh of the logs\n    setTimeout(() => (this.selectedIndex = index));\n  }\n\n  public selectIndex({ target: { value } }) {\n    return this.updateSelectedIndex(+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 && hasEmissions\">\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 emissions by name\"\n      (search)=\"filter($event)\"></he-search-extend>\n  </div>\n  <div class=\"column is-narrow\" *ngIf=\"cycles.length > 1 || !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=\"cycles.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=\"!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=\"hasEmissions; else emptyTable\">\n    <he-data-table class=\"mb-1\" [small]=\"true\" maxHeight=\"320\">\n      <table class=\"table is-fullwidth is-narrow is-striped\">\n        <thead>\n          <tr class=\"has-text-weight-bold\">\n            <th class=\"width-auto\"></th>\n            <th [class.is-hidden]=\"isTransformation\"></th>\n            <ng-container *ngFor=\"let item of emissions; let lastGroup = last\">\n              <th [class.has-border-right]=\"!lastGroup\" [attr.colspan]=\"item.value.emissions.length\">\n                {{ item.key | keyToLabel }}\n              </th>\n            </ng-container>\n          </tr>\n          <tr class=\"has-text-weight-semibold\">\n            <th class=\"width-auto has-border-right\"></th>\n            <th class=\"has-border-right\" [class.is-hidden]=\"isTransformation\"></th>\n            <ng-container *ngFor=\"let item of emissions; let lastGroup = last\">\n              <th\n                *ngFor=\"let emission of item.value.emissions; let lastEmission = last\"\n                [class.has-border-right]=\"lastEmission && !lastGroup\"\n                [attr.title]=\"emission.value.term.name\">\n                <he-node-link [node]=\"emission.value.term\">\n                  <span [innerHtml]=\"emission.value.term.name | ellipsis: 30 | compound\"></span>\n                </he-node-link>\n              </th>\n            </ng-container>\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\" [class.is-hidden]=\"isTransformation\">\n              <a [href]=\"baseUrl + '/schema/Cycle#functionalUnit'\" target=\"_blank\">Functional unit</a>\n            </th>\n            <ng-container *ngFor=\"let item of emissions; let lastGroup = last\">\n              <th\n                *ngFor=\"let emission of item.value.emissions; let lastEmission = last\"\n                [class.has-border-right]=\"lastEmission && !lastGroup\"\n                [attr.title]=\"emission.value.term.units\">\n                <span [innerHtml]=\"emission.value.term.units | compound\"></span>\n                <he-terms-units-description\n                  class=\"is-inline-block is-ml-2\"\n                  [term]=\"emission.value.term\"></he-terms-units-description>\n              </th>\n            </ng-container>\n          </tr>\n        </thead>\n        <tbody>\n          <ng-container *ngFor=\"let cycle of cycles; trackBy: trackById; let i = index\">\n            <tr>\n              <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n                <he-node-link [node]=\"cycle.term || cycle\">\n                  <span>{{ i + 1 }}. {{ defaultLabel(cycle) }}</span>\n                </he-node-link>\n              </td>\n              <td class=\"has-border-right\" [class.is-hidden]=\"isTransformation\">\n                <he-cycles-functional-unit-measure [cycle]=\"cycles[0]\"></he-cycles-functional-unit-measure>\n              </td>\n              <ng-container *ngFor=\"let item of emissions; let lastGroup = last\">\n                <td\n                  class=\"is-nowrap\"\n                  *ngFor=\"let emission of item.value.emissions; let lastEmission = last\"\n                  [class.has-border-right]=\"lastEmission && !lastGroup\">\n                  <span\n                    *ngIf=\"emission.value.values[cycle['@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, { data: emission.value.values[cycle['@id']], cycle: cycle, key: 'emissions' })\n                    \">\n                    <span pointer>\n                      {{\n                        propertyValue(emission.value.values[cycle['@id']].value, emission.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]=\"emission.value.values[cycle['@id']].node\"\n                      key=\"value\"></he-blank-node-state>\n                  </span>\n                </td>\n              </ng-container>\n            </tr>\n          </ng-container>\n        </tbody>\n      </table>\n    </he-data-table>\n\n    <he-blank-node-state-notice [dataState]=\"dataState\" [showDeleted]=\"true\"></he-blank-node-state-notice>\n  </ng-container>\n</div>\n\n<he-cycles-emissions-chart\n  *ngIf=\"selectedView === View.chart && cycles.length > 1\"\n  [cycles]=\"cycles\"></he-cycles-emissions-chart>\n\n<ng-container *ngIf=\"selectedView === View.logs && !isOriginal\">\n  <div class=\"field has-addons pt-2 px-3\" *ngIf=\"cycles.length > 1\">\n    <div class=\"control\">\n      <span class=\"button is-small is-static is-secondary\">Select a Cycle</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 cycles; let i = index\" [value]=\"i\">\n            {{ i + 1 }}. {{ defaultLabel(value) }}\n          </option>\n        </select>\n      </div>\n    </div>\n  </div>\n\n  <he-cycles-emissions-logs\n    *ngIf=\"selectedIndex >= 0\"\n    [cycle]=\"cycles[selectedIndex]\"\n    [originalValues]=\"originalValues[selectedIndex]?.emissions\"\n    [recalculatedValues]=\"cycles[selectedIndex]?.emissions\"></he-cycles-emissions-logs>\n</ng-container>\n\n<he-node-csv-export-confirm\n  *ngIf=\"showDownload\"\n  [nodes]=\"cycles\"\n  filename=\"emissions.csv\"\n  [isUpload]=\"false\"\n  [headerKeys]=\"headerKeys\"\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 *ngIf=\"showSwitchToRecalculated\">\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=\"cycle\" let-data=\"data\" let-key=\"key\">\n  <p>\n    <b>{{ defaultLabel(node) }}</b>\n  </p>\n  <he-node-value-details\n    [data]=\"data\"\n    [dataState]=\"dataState\"\n    [dataState]=\"dataState\"\n    [nodeType]=\"node['@type']\"\n    [dataKey]=\"key\"></he-node-value-details>\n</ng-template>\n"]}
127
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cycles-emissions.component.js","sourceRoot":"","sources":["../../../../src/cycles/cycles-emissions/cycles-emissions.component.ts","../../../../src/cycles/cycles-emissions/cycles-emissions.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnChanges, SimpleChanges } from '@angular/core';\nimport { DataState } from '@hestia-earth/api';\nimport { ICycleJSONLD, Emission, NodeType, SchemaType, EmissionMethodTier } from '@hestia-earth/schema';\nimport { propertyValue } from '@hestia-earth/utils/dist/term';\nimport orderBy from 'lodash.orderby';\n\nimport { groupNodesByTerm, IGroupedKeys, grouppedKeys, methodTierOrder } from '../../common/node-utils';\nimport { baseUrl, defaultLabel } from '../../common/utils';\n\nenum View {\n  table = 'table',\n  chart = 'chart',\n  logs = 'logs'\n}\n\nconst headerKeys = [\n  'cycle.id',\n  'cycle.@id',\n  'cycle.emissions.',\n  'transformation.id',\n  'transformation.@id',\n  'transformation.emissions.'\n];\n\ntype groupedEmission = {\n  [methodTier in EmissionMethodTier]: {\n    methodTierOrder: number;\n    emissions: IGroupedKeys<Emission>[];\n  };\n};\n\ninterface IGroupedEmissionByKey {\n  key: string;\n  value: {\n    methodTierOrder: number;\n    emissions: IGroupedKeys<Emission>[];\n  };\n}\n\nconst groupByMethodTier = (emissions: IGroupedKeys<Emission>[]) =>\n  emissions.reduce((p, e) => {\n    p[e.value.methodTier] = p[e.value.methodTier] || {\n      methodTierOrder: methodTierOrder(e.value.methodTier),\n      emissions: []\n    };\n    p[e.value.methodTier].emissions.push(e);\n    return p;\n  }, {} as groupedEmission);\n\n@Component({\n  selector: 'he-cycles-emissions',\n  templateUrl: './cycles-emissions.component.html',\n  styleUrls: ['./cycles-emissions.component.scss']\n})\nexport class CyclesEmissionsComponent implements OnChanges {\n  @Input()\n  public originalValues: ICycleJSONLD[] = [];\n  @Input()\n  public cycles: ICycleJSONLD[] = [];\n  @Input()\n  public dataState?: DataState;\n\n  public baseUrl = baseUrl();\n  public propertyValue = propertyValue;\n  public defaultLabel = defaultLabel;\n  public headerKeys = headerKeys;\n  public showDownload = false;\n  public View = View;\n  public selectedView = View.table;\n  public selectedIndex = 0;\n\n  public hasEmissions = false;\n  public emissions: IGroupedEmissionByKey[];\n  public filterTerm = '';\n\n  ngOnChanges(changes: SimpleChanges) {\n    if ('dataState' in changes) {\n      this.selectedView = View.table;\n    }\n    if ('cycles' in changes) {\n      this.updateSelectedIndex(this.selectedIndex);\n      return this.update();\n    }\n  }\n\n  public trackById(_index: number, item: ICycleJSONLD) {\n    return item['@id'];\n  }\n\n  public get isOriginal() {\n    return this.dataState === DataState.original;\n  }\n\n  private update() {\n    const emissionsPerCycle = groupNodesByTerm<ICycleJSONLD, Emission>(\n      this.cycles,\n      'emissions',\n      this.originalValues,\n      node => this.filterNode(node)\n    );\n    const emissions = orderBy(grouppedKeys(emissionsPerCycle), ['key'], ['asc']);\n    this.hasEmissions = emissions.length > 0;\n    this.emissions = orderBy(grouppedKeys(groupByMethodTier(emissions)), ['value.methodTierOrder'], ['asc']);\n  }\n\n  public togglePopover(popover: any, context: any) {\n    return popover.isOpen() ? popover.close() : popover.open(context);\n  }\n\n  public get isTransformation() {\n    const firstNode = (this.cycles || [{ '@type': NodeType.Cycle }])[0];\n    const nodeType = (firstNode as any).type || firstNode['@type'];\n    return nodeType === SchemaType.Transformation;\n  }\n\n  private filterNode({ term }: Emission) {\n    return !this.filterTerm || term?.name?.toLowerCase()?.includes(this.filterTerm.toLowerCase());\n  }\n\n  public filter(term: string) {\n    this.filterTerm = term;\n    return this.update();\n  }\n\n  // Recalculation logs\n\n  public get showSwitchToRecalculated() {\n    return this.isOriginal && this.cycles.some(cycle => !cycle.aggregated);\n  }\n\n  private updateSelectedIndex(index: number) {\n    this.selectedIndex = -1;\n    // force a refresh of the logs\n    setTimeout(() => (this.selectedIndex = index));\n  }\n\n  public selectIndex({ target: { value } }) {\n    return this.updateSelectedIndex(+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 && hasEmissions\">\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 emissions by name\"\n      (search)=\"filter($event)\"></he-search-extend>\n  </div>\n  <div class=\"column is-narrow\" *ngIf=\"cycles.length > 1 || !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=\"cycles.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=\"!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=\"hasEmissions; else emptyTable\">\n    <he-data-table class=\"mb-1\" [small]=\"true\" maxHeight=\"320\">\n      <table class=\"table is-fullwidth is-narrow is-striped\">\n        <thead>\n          <tr class=\"has-text-weight-bold\">\n            <th class=\"width-auto\"></th>\n            <th [class.is-hidden]=\"isTransformation\"></th>\n            <ng-container *ngFor=\"let item of emissions; let lastGroup = last\">\n              <th [class.has-border-right]=\"!lastGroup\" [attr.colspan]=\"item.value.emissions.length\">\n                {{ item.key | keyToLabel }}\n              </th>\n            </ng-container>\n          </tr>\n          <tr class=\"has-text-weight-semibold\">\n            <th class=\"width-auto has-border-right\"></th>\n            <th class=\"has-border-right\" [class.is-hidden]=\"isTransformation\"></th>\n            <ng-container *ngFor=\"let item of emissions; let lastGroup = last\">\n              <th\n                *ngFor=\"let emission of item.value.emissions; let lastEmission = last\"\n                [class.has-border-right]=\"lastEmission && !lastGroup\"\n                [attr.title]=\"emission.value.term.name\">\n                <he-node-link [node]=\"emission.value.term\">\n                  <span [innerHtml]=\"emission.value.term.name | ellipsis: 30 | compound\"></span>\n                </he-node-link>\n              </th>\n            </ng-container>\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\" [class.is-hidden]=\"isTransformation\">\n              <a [href]=\"baseUrl + '/schema/Cycle#functionalUnit'\" target=\"_blank\">Functional unit</a>\n            </th>\n            <ng-container *ngFor=\"let item of emissions; let lastGroup = last\">\n              <th\n                *ngFor=\"let emission of item.value.emissions; let lastEmission = last\"\n                [class.has-border-right]=\"lastEmission && !lastGroup\"\n                [attr.title]=\"emission.value.term.units\">\n                <span [innerHtml]=\"emission.value.term.units | compound\"></span>\n                <he-terms-units-description\n                  class=\"is-inline-block is-ml-2\"\n                  [term]=\"emission.value.term\"></he-terms-units-description>\n              </th>\n            </ng-container>\n          </tr>\n        </thead>\n        <tbody>\n          <ng-container *ngFor=\"let cycle of cycles; trackBy: trackById; let i = index\">\n            <tr>\n              <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n                <he-node-link [node]=\"cycle.term || cycle\">\n                  <span>{{ i + 1 }}. {{ defaultLabel(cycle) }}</span>\n                </he-node-link>\n              </td>\n              <td class=\"has-border-right\" [class.is-hidden]=\"isTransformation\">\n                <he-cycles-functional-unit-measure [cycle]=\"cycles[0]\"></he-cycles-functional-unit-measure>\n              </td>\n              <ng-container *ngFor=\"let item of emissions; let lastGroup = last\">\n                <td\n                  class=\"is-nowrap\"\n                  *ngFor=\"let emission of item.value.emissions; let lastEmission = last\"\n                  [class.has-border-right]=\"lastEmission && !lastGroup\">\n                  <span\n                    *ngIf=\"emission.value.values[cycle['@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, { data: emission.value.values[cycle['@id']], cycle: cycle, key: 'emissions' })\n                    \">\n                    <span pointer>\n                      {{\n                        propertyValue(emission.value.values[cycle['@id']].value, emission.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]=\"emission.value.values[cycle['@id']].node\"\n                      key=\"value\"></he-blank-node-state>\n                  </span>\n                </td>\n              </ng-container>\n            </tr>\n          </ng-container>\n        </tbody>\n      </table>\n    </he-data-table>\n\n    <he-blank-node-state-notice [dataState]=\"dataState\" [showDeleted]=\"true\"></he-blank-node-state-notice>\n  </ng-container>\n</div>\n\n<he-cycles-emissions-chart\n  *ngIf=\"selectedView === View.chart && cycles.length > 1\"\n  [cycles]=\"cycles\"></he-cycles-emissions-chart>\n\n<ng-container *ngIf=\"selectedView === View.logs && !isOriginal\">\n  <div class=\"field has-addons pt-2 px-3\" *ngIf=\"cycles.length > 1\">\n    <div class=\"control\">\n      <span class=\"button is-small is-static is-secondary\">Select a Cycle</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 cycles; let i = index\" [value]=\"i\">\n            {{ i + 1 }}. {{ defaultLabel(value) }}\n          </option>\n        </select>\n      </div>\n    </div>\n  </div>\n\n  <he-cycles-emissions-logs\n    *ngIf=\"selectedIndex >= 0\"\n    [cycle]=\"cycles[selectedIndex]\"\n    [originalValues]=\"originalValues[selectedIndex]?.emissions\"\n    [recalculatedValues]=\"cycles[selectedIndex]?.emissions\"></he-cycles-emissions-logs>\n</ng-container>\n\n<he-node-csv-export-confirm\n  *ngIf=\"showDownload\"\n  [nodes]=\"cycles\"\n  filename=\"emissions.csv\"\n  [isUpload]=\"false\"\n  [headerKeys]=\"headerKeys\"\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 *ngIf=\"showSwitchToRecalculated\">\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=\"cycle\" let-data=\"data\" let-key=\"key\">\n  <p>\n    <b>{{ defaultLabel(node) }}</b>\n  </p>\n  <he-node-value-details\n    [data]=\"data\"\n    [dataState]=\"dataState\"\n    [dataState]=\"dataState\"\n    [nodeType]=\"node['@type']\"\n    [dataKey]=\"key\"></he-node-value-details>\n</ng-template>\n"]}
@@ -89,7 +89,7 @@ export class CyclesPracticesComponent {
89
89
  return this.update();
90
90
  }
91
91
  // Recalculation logs
92
- showSwitchToRecalculated() {
92
+ get showSwitchToRecalculated() {
93
93
  return this.isOriginal && this.cycles.some(cycle => !cycle.aggregated);
94
94
  }
95
95
  updateSelectedIndex(index) {
@@ -113,4 +113,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.8", ngImpor
113
113
  }], dataState: [{
114
114
  type: Input
115
115
  }] } });
116
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cycles-practices.component.js","sourceRoot":"","sources":["../../../../src/cycles/cycles-practices/cycles-practices.component.ts","../../../../src/cycles/cycles-practices/cycles-practices.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnChanges, SimpleChanges } from '@angular/core';\nimport { DataState } from '@hestia-earth/api';\nimport { ICycleJSONLD, Practice, TermTermType } from '@hestia-earth/schema';\nimport { propertyValue } from '@hestia-earth/utils/dist/term';\nimport orderBy from 'lodash.orderby';\n\nimport { groupNodesByTerm, IGroupedKeys, grouppedKeys } from '../../common/node-utils';\nimport { baseUrl, defaultLabel } from '../../common/utils';\n\nenum View {\n  table = 'table',\n  timeline = 'timeline',\n  logs = 'logs'\n}\n\nconst headerKeys = [\n  'cycle.id',\n  'cycle.@id',\n  'cycle.practices.',\n  'transformation.id',\n  'transformation.@id',\n  'transformation.practices.'\n];\n\nconst timelineTermType = [TermTermType.operation];\n\n@Component({\n  selector: 'he-cycles-practices',\n  templateUrl: './cycles-practices.component.html',\n  styleUrls: ['./cycles-practices.component.scss']\n})\nexport class CyclesPracticesComponent implements OnChanges {\n  @Input()\n  public originalValues: ICycleJSONLD[] = [];\n  @Input()\n  public cycles: ICycleJSONLD[] = [];\n  @Input()\n  public dataState?: DataState;\n\n  public baseUrl = baseUrl();\n  public propertyValue = propertyValue;\n  public defaultLabel = defaultLabel;\n  public headerKeys = headerKeys;\n  public timelineTermType = timelineTermType;\n  public showDownload = false;\n  public View = View;\n  public selectedView = View.table;\n  public selectedIndex = 0;\n\n  public practices: IGroupedKeys<Practice>[] = [];\n  public filterTerm = '';\n\n  ngOnChanges(changes: SimpleChanges) {\n    if ('dataState' in changes) {\n      this.selectedView = View.table;\n    }\n    if ('cycles' in changes) {\n      this.updateSelectedIndex(this.selectedIndex);\n      return this.update();\n    }\n  }\n\n  public trackById(_index: number, item: ICycleJSONLD) {\n    return item['@id'];\n  }\n\n  public get isOriginal() {\n    return this.dataState === DataState.original;\n  }\n\n  private update() {\n    const practicesPerCycle = groupNodesByTerm<ICycleJSONLD, Practice>(\n      this.cycles,\n      'practices',\n      this.originalValues,\n      node => this.filterNode(node)\n    );\n    this.practices = orderBy(grouppedKeys(practicesPerCycle), ['key'], ['asc']);\n  }\n\n  public togglePopover(popover: any, context: any) {\n    return popover.isOpen() ? popover.close() : popover.open(context);\n  }\n\n  public get showTimeline() {\n    return (this.cycles?.[0]?.practices || []).some(p => timelineTermType.includes(p.term?.termType));\n  }\n\n  private filterNode({ term }: Practice) {\n    return !this.filterTerm || term?.name?.toLowerCase()?.includes(this.filterTerm.toLowerCase());\n  }\n\n  public filter(term: string) {\n    this.filterTerm = term;\n    return this.update();\n  }\n\n  // Recalculation logs\n\n  public showSwitchToRecalculated() {\n    return this.isOriginal && this.cycles.some(cycle => !cycle.aggregated);\n  }\n\n  private updateSelectedIndex(index: number) {\n    this.selectedIndex = -1;\n    // force a refresh of the logs\n    setTimeout(() => (this.selectedIndex = index));\n  }\n\n  public selectIndex({ target: { value } }) {\n    return this.updateSelectedIndex(+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 && practices.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 practices by name\"\n      (search)=\"filter($event)\"></he-search-extend>\n  </div>\n  <div class=\"column is-narrow\" *ngIf=\"showTimeline || !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=\"showTimeline\">\n        <button\n          class=\"button is-small\"\n          [class.is-selected]=\"selectedView === View.timeline\"\n          (click)=\"selectedView = View.timeline\">\n          <span class=\"icon is-small\">\n            <fa-icon icon=\"list-alt\" aria-hidden=\"true\"></fa-icon>\n          </span>\n          <span class=\"is-hidden-mobile\">Operations timeline</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=\"practices.length; 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 has-border-right\"></th>\n            <th class=\"has-border-right\"></th>\n            <th *ngFor=\"let practice of practices\" [attr.title]=\"practice.value.term.name\">\n              <he-node-link [node]=\"practice.value.term\">\n                <span [innerHtml]=\"practice.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/Cycle#functionalUnit'\" target=\"_blank\">Functional unit</a>\n            </th>\n            <th *ngFor=\"let practice of practices\" [attr.title]=\"practice.value.term.units\">\n              <span [innerHtml]=\"practice.value.term.units | compound\"></span>\n              <he-terms-units-description\n                class=\"is-inline-block is-ml-2\"\n                [term]=\"practice.value.term\"></he-terms-units-description>\n            </th>\n          </tr>\n        </thead>\n        <tbody>\n          <ng-container *ngFor=\"let cycle of cycles; trackBy: trackById; let i = index\">\n            <tr>\n              <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n                <he-node-link [node]=\"cycle.term || cycle\">\n                  <span>{{ i + 1 }}. {{ defaultLabel(cycle) }}</span>\n                </he-node-link>\n              </td>\n              <td class=\"has-border-right\">\n                <he-cycles-functional-unit-measure [cycle]=\"cycles[0]\"></he-cycles-functional-unit-measure>\n              </td>\n              <td class=\"is-nowrap\" *ngFor=\"let practice of practices\">\n                <span\n                  *ngIf=\"practice.value.values[cycle['@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, { data: practice.value.values[cycle['@id']], cycle: cycle, key: 'practices' })\n                  \">\n                  <span pointer>\n                    {{\n                      propertyValue(practice.value.values[cycle['@id']].value, practice.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]=\"practice.value.values[cycle['@id']].node\"\n                    key=\"value\"></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<div class=\"px-3 pb-3\" *ngIf=\"selectedView === View.timeline\">\n  <he-cycles-practices-timeline\n    [dataState]=\"dataState\"\n    [cycle]=\"cycles[0]\"\n    [filterTermTypes]=\"timelineTermType\"></he-cycles-practices-timeline>\n\n  <he-blank-node-state-notice [dataState]=\"dataState\"></he-blank-node-state-notice>\n</div>\n\n<ng-container *ngIf=\"selectedView === View.logs && !isOriginal\">\n  <div class=\"field has-addons pt-2 px-3\" *ngIf=\"cycles.length > 1\">\n    <div class=\"control\">\n      <span class=\"button is-small is-static is-secondary\">Select a Cycle</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 cycles; let i = index\" [value]=\"i\">\n            {{ i + 1 }}. {{ defaultLabel(value) }}\n          </option>\n        </select>\n      </div>\n    </div>\n  </div>\n\n  <he-cycles-practices-logs\n    *ngIf=\"selectedIndex >= 0\"\n    [cycle]=\"cycles[selectedIndex]\"\n    [originalValues]=\"originalValues[selectedIndex]?.practices\"\n    [recalculatedValues]=\"cycles[selectedIndex]?.practices\"></he-cycles-practices-logs>\n</ng-container>\n\n<he-node-csv-export-confirm\n  *ngIf=\"showDownload\"\n  [nodes]=\"cycles\"\n  filename=\"practices.csv\"\n  [isUpload]=\"false\"\n  [headerKeys]=\"headerKeys\"\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 *ngIf=\"showSwitchToRecalculated\">\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=\"cycle\" let-data=\"data\" let-key=\"key\">\n  <p>\n    <b>{{ defaultLabel(node) }}</b>\n  </p>\n  <he-node-value-details\n    [data]=\"data\"\n    [dataState]=\"dataState\"\n    [nodeType]=\"node['@type']\"\n    [dataKey]=\"key\"></he-node-value-details>\n</ng-template>\n"]}
116
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cycles-practices.component.js","sourceRoot":"","sources":["../../../../src/cycles/cycles-practices/cycles-practices.component.ts","../../../../src/cycles/cycles-practices/cycles-practices.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnChanges, SimpleChanges } from '@angular/core';\nimport { DataState } from '@hestia-earth/api';\nimport { ICycleJSONLD, Practice, TermTermType } from '@hestia-earth/schema';\nimport { propertyValue } from '@hestia-earth/utils/dist/term';\nimport orderBy from 'lodash.orderby';\n\nimport { groupNodesByTerm, IGroupedKeys, grouppedKeys } from '../../common/node-utils';\nimport { baseUrl, defaultLabel } from '../../common/utils';\n\nenum View {\n  table = 'table',\n  timeline = 'timeline',\n  logs = 'logs'\n}\n\nconst headerKeys = [\n  'cycle.id',\n  'cycle.@id',\n  'cycle.practices.',\n  'transformation.id',\n  'transformation.@id',\n  'transformation.practices.'\n];\n\nconst timelineTermType = [TermTermType.operation];\n\n@Component({\n  selector: 'he-cycles-practices',\n  templateUrl: './cycles-practices.component.html',\n  styleUrls: ['./cycles-practices.component.scss']\n})\nexport class CyclesPracticesComponent implements OnChanges {\n  @Input()\n  public originalValues: ICycleJSONLD[] = [];\n  @Input()\n  public cycles: ICycleJSONLD[] = [];\n  @Input()\n  public dataState?: DataState;\n\n  public baseUrl = baseUrl();\n  public propertyValue = propertyValue;\n  public defaultLabel = defaultLabel;\n  public headerKeys = headerKeys;\n  public timelineTermType = timelineTermType;\n  public showDownload = false;\n  public View = View;\n  public selectedView = View.table;\n  public selectedIndex = 0;\n\n  public practices: IGroupedKeys<Practice>[] = [];\n  public filterTerm = '';\n\n  ngOnChanges(changes: SimpleChanges) {\n    if ('dataState' in changes) {\n      this.selectedView = View.table;\n    }\n    if ('cycles' in changes) {\n      this.updateSelectedIndex(this.selectedIndex);\n      return this.update();\n    }\n  }\n\n  public trackById(_index: number, item: ICycleJSONLD) {\n    return item['@id'];\n  }\n\n  public get isOriginal() {\n    return this.dataState === DataState.original;\n  }\n\n  private update() {\n    const practicesPerCycle = groupNodesByTerm<ICycleJSONLD, Practice>(\n      this.cycles,\n      'practices',\n      this.originalValues,\n      node => this.filterNode(node)\n    );\n    this.practices = orderBy(grouppedKeys(practicesPerCycle), ['key'], ['asc']);\n  }\n\n  public togglePopover(popover: any, context: any) {\n    return popover.isOpen() ? popover.close() : popover.open(context);\n  }\n\n  public get showTimeline() {\n    return (this.cycles?.[0]?.practices || []).some(p => timelineTermType.includes(p.term?.termType));\n  }\n\n  private filterNode({ term }: Practice) {\n    return !this.filterTerm || term?.name?.toLowerCase()?.includes(this.filterTerm.toLowerCase());\n  }\n\n  public filter(term: string) {\n    this.filterTerm = term;\n    return this.update();\n  }\n\n  // Recalculation logs\n\n  public get showSwitchToRecalculated() {\n    return this.isOriginal && this.cycles.some(cycle => !cycle.aggregated);\n  }\n\n  private updateSelectedIndex(index: number) {\n    this.selectedIndex = -1;\n    // force a refresh of the logs\n    setTimeout(() => (this.selectedIndex = index));\n  }\n\n  public selectIndex({ target: { value } }) {\n    return this.updateSelectedIndex(+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 && practices.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 practices by name\"\n      (search)=\"filter($event)\"></he-search-extend>\n  </div>\n  <div class=\"column is-narrow\" *ngIf=\"showTimeline || !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=\"showTimeline\">\n        <button\n          class=\"button is-small\"\n          [class.is-selected]=\"selectedView === View.timeline\"\n          (click)=\"selectedView = View.timeline\">\n          <span class=\"icon is-small\">\n            <fa-icon icon=\"list-alt\" aria-hidden=\"true\"></fa-icon>\n          </span>\n          <span class=\"is-hidden-mobile\">Operations timeline</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=\"practices.length; 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 has-border-right\"></th>\n            <th class=\"has-border-right\"></th>\n            <th *ngFor=\"let practice of practices\" [attr.title]=\"practice.value.term.name\">\n              <he-node-link [node]=\"practice.value.term\">\n                <span [innerHtml]=\"practice.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/Cycle#functionalUnit'\" target=\"_blank\">Functional unit</a>\n            </th>\n            <th *ngFor=\"let practice of practices\" [attr.title]=\"practice.value.term.units\">\n              <span [innerHtml]=\"practice.value.term.units | compound\"></span>\n              <he-terms-units-description\n                class=\"is-inline-block is-ml-2\"\n                [term]=\"practice.value.term\"></he-terms-units-description>\n            </th>\n          </tr>\n        </thead>\n        <tbody>\n          <ng-container *ngFor=\"let cycle of cycles; trackBy: trackById; let i = index\">\n            <tr>\n              <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n                <he-node-link [node]=\"cycle.term || cycle\">\n                  <span>{{ i + 1 }}. {{ defaultLabel(cycle) }}</span>\n                </he-node-link>\n              </td>\n              <td class=\"has-border-right\">\n                <he-cycles-functional-unit-measure [cycle]=\"cycles[0]\"></he-cycles-functional-unit-measure>\n              </td>\n              <td class=\"is-nowrap\" *ngFor=\"let practice of practices\">\n                <span\n                  *ngIf=\"practice.value.values[cycle['@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, { data: practice.value.values[cycle['@id']], cycle: cycle, key: 'practices' })\n                  \">\n                  <span pointer>\n                    {{\n                      propertyValue(practice.value.values[cycle['@id']].value, practice.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]=\"practice.value.values[cycle['@id']].node\"\n                    key=\"value\"></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<div class=\"px-3 pb-3\" *ngIf=\"selectedView === View.timeline\">\n  <he-cycles-practices-timeline\n    [dataState]=\"dataState\"\n    [cycle]=\"cycles[0]\"\n    [filterTermTypes]=\"timelineTermType\"></he-cycles-practices-timeline>\n\n  <he-blank-node-state-notice [dataState]=\"dataState\"></he-blank-node-state-notice>\n</div>\n\n<ng-container *ngIf=\"selectedView === View.logs && !isOriginal\">\n  <div class=\"field has-addons pt-2 px-3\" *ngIf=\"cycles.length > 1\">\n    <div class=\"control\">\n      <span class=\"button is-small is-static is-secondary\">Select a Cycle</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 cycles; let i = index\" [value]=\"i\">\n            {{ i + 1 }}. {{ defaultLabel(value) }}\n          </option>\n        </select>\n      </div>\n    </div>\n  </div>\n\n  <he-cycles-practices-logs\n    *ngIf=\"selectedIndex >= 0\"\n    [cycle]=\"cycles[selectedIndex]\"\n    [originalValues]=\"originalValues[selectedIndex]?.practices\"\n    [recalculatedValues]=\"cycles[selectedIndex]?.practices\"></he-cycles-practices-logs>\n</ng-container>\n\n<he-node-csv-export-confirm\n  *ngIf=\"showDownload\"\n  [nodes]=\"cycles\"\n  filename=\"practices.csv\"\n  [isUpload]=\"false\"\n  [headerKeys]=\"headerKeys\"\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 *ngIf=\"showSwitchToRecalculated\">\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=\"cycle\" let-data=\"data\" let-key=\"key\">\n  <p>\n    <b>{{ defaultLabel(node) }}</b>\n  </p>\n  <he-node-value-details\n    [data]=\"data\"\n    [dataState]=\"dataState\"\n    [nodeType]=\"node['@type']\"\n    [dataKey]=\"key\"></he-node-value-details>\n</ng-template>\n"]}