@hestia-earth/ui-components 0.29.2 → 0.29.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/common/maps-utils.d.ts +4 -0
- package/common/navigation-menu/navigation-menu.component.d.ts +1 -0
- package/common/node-utils.d.ts +4 -2
- package/common/semver-utils.d.ts +2 -3
- package/common/utils.d.ts +0 -5
- package/cycles/cycles-nodes/cycles-nodes.component.d.ts +1 -1
- package/engine/engine-models-stage-deep/engine-models-stage-deep.component.d.ts +1 -1
- package/engine/engine-models-version-link/engine-models-version-link.component.d.ts +1 -1
- package/engine/engine.service.d.ts +1 -1
- package/esm2022/bibliographies/bibliographies-search-confirm/bibliographies-search-confirm.component.mjs +3 -3
- package/esm2022/chart/chart.component.mjs +3 -3
- package/esm2022/chart/line-chart/line-chart.component.mjs +3 -3
- package/esm2022/common/blank-node-state/blank-node-state.component.mjs +3 -3
- package/esm2022/common/clipboard/clipboard.component.mjs +3 -3
- package/esm2022/common/collapsible-box/collapsible-box.component.mjs +3 -3
- package/esm2022/common/data-table/data-table.component.mjs +3 -3
- package/esm2022/common/drawer-container/drawer-container.component.mjs +3 -3
- package/esm2022/common/link-key-value/link-key-value.component.mjs +3 -3
- package/esm2022/common/maps-drawing-confirm/maps-drawing-confirm.component.mjs +3 -3
- package/esm2022/common/maps-utils.mjs +3 -1
- package/esm2022/common/mobile-shell/mobile-shell.component.mjs +3 -3
- package/esm2022/common/navigation-menu/navigation-menu.component.mjs +7 -3
- package/esm2022/common/node-utils.mjs +12 -7
- package/esm2022/common/schema-version-link/schema-version-link.component.mjs +3 -3
- package/esm2022/common/search-extend/search-extend.component.mjs +3 -3
- package/esm2022/common/semver-utils.mjs +3 -4
- package/esm2022/common/shelf-dialog/shelf-dialog.component.mjs +3 -3
- package/esm2022/common/shell/shell.component.mjs +3 -3
- package/esm2022/common/sort-select/sort-select.component.mjs +3 -3
- package/esm2022/common/utils.mjs +1 -9
- package/esm2022/cycles/cycles-completeness/cycles-completeness.component.mjs +3 -3
- package/esm2022/cycles/cycles-nodes/cycles-nodes.component.mjs +5 -5
- package/esm2022/cycles/cycles-nodes-timeline/cycles-nodes-timeline.component.mjs +3 -3
- package/esm2022/engine/engine-models-stage-deep/engine-models-stage-deep.component.mjs +9 -8
- package/esm2022/engine/engine-models-version-link/engine-models-version-link.component.mjs +3 -3
- package/esm2022/engine/engine-requirements-form/engine-requirements-form.component.mjs +3 -3
- package/esm2022/engine/engine.service.mjs +3 -2
- package/esm2022/files/files-error-summary/files-error-summary-item/files-error-summary-item.component.mjs +3 -3
- package/esm2022/files/files-form.model.mjs +3 -2
- package/esm2022/files/files-upload-errors/files-upload-errors.component.mjs +3 -3
- package/esm2022/impact-assessments/impact-assessments-graph/impact-assessments-graph.component.mjs +3 -3
- package/esm2022/impact-assessments/impact-assessments-indicator-breakdown-chart/impact-assessments-indicator-breakdown-chart.component.mjs +3 -3
- package/esm2022/impact-assessments/impact-assessments-products/impact-assessments-products.component.mjs +7 -4
- package/esm2022/node/node-csv-export-confirm/node-csv-export-confirm.component.mjs +3 -3
- package/esm2022/node/node-csv-select-headers/node-csv-select-headers.component.mjs +3 -3
- package/esm2022/node/node-diffs/node-diffs.component.mjs +3 -3
- package/esm2022/node/node-link/node-link.component.mjs +3 -3
- package/esm2022/node/node-logs-file/node-logs-file.component.mjs +3 -3
- package/esm2022/node/node-logs-models/node-logs-models.component.mjs +34 -41
- package/esm2022/node/node-logs-models/node-logs-models.model.mjs +10 -6
- package/esm2022/node/node-missing-lookup-factors/node-missing-lookup-factors.component.mjs +3 -3
- package/esm2022/node/node-recommendations/node-recommendations.component.mjs +3 -3
- package/esm2022/node/node-store.service.mjs +1 -1
- package/esm2022/node/node-value-details/node-value-details.component.mjs +5 -3
- package/esm2022/schema/schema-info/schema-info.component.mjs +3 -3
- package/esm2022/select/select-option/select-option.component.mjs +3 -3
- package/esm2022/select/select-option-group/select-option-group.component.mjs +3 -3
- package/esm2022/select/select.component.mjs +3 -3
- package/esm2022/sites/sites-management-chart/sites-management-chart.component.mjs +8 -12
- package/esm2022/sites/sites-nodes/sites-nodes.component.mjs +6 -5
- package/esm2022/terms/terms-property-content/terms-property-content.component.mjs +3 -3
- package/esm2022/terms/terms-sub-class-of-content/terms-sub-class-of-content.component.mjs +3 -3
- package/esm2022/terms/terms-units-description/terms-units-description.component.mjs +3 -3
- package/fesm2022/hestia-earth-ui-components.mjs +172 -170
- package/fesm2022/hestia-earth-ui-components.mjs.map +1 -1
- package/impact-assessments/impact-assessments-products/impact-assessments-products.component.d.ts +2 -0
- package/node/node-logs-models/node-logs-models.component.d.ts +4 -4
- package/node/node-logs-models/node-logs-models.model.d.ts +1 -1
- package/node/node-store.service.d.ts +1 -1
- package/package.json +1 -1
- package/sites/sites-management-chart/sites-management-chart.component.d.ts +1 -0
- package/sites/sites-nodes/sites-nodes.component.d.ts +2 -1
- package/styles.scss +5 -34
|
@@ -92,7 +92,7 @@ export class CyclesCompletenessComponent {
|
|
|
92
92
|
component.headerKeys.set(headerKeys);
|
|
93
93
|
}
|
|
94
94
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.11", ngImport: i0, type: CyclesCompletenessComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
95
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.11", type: CyclesCompletenessComponent, isStandalone: true, selector: "he-cycles-completeness", inputs: { dataState: { classPropertyName: "dataState", publicName: "dataState", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: "<div class=\"columns is-variable is-align-items-center is-2 m-0\">\n <div class=\"column\">\n @if (selectedView() === View.table) {\n <button class=\"button is-small is-ghost\" (click)=\"showDownload()\">\n <fa-icon [icon]=\"faDownload\" size=\"lg\"
|
|
95
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.11", type: CyclesCompletenessComponent, isStandalone: true, selector: "he-cycles-completeness", inputs: { dataState: { classPropertyName: "dataState", publicName: "dataState", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: "<div class=\"columns is-variable is-align-items-center is-2 m-0\">\n <div class=\"column\">\n @if (selectedView() === View.table) {\n <button class=\"button is-small is-ghost\" (click)=\"showDownload()\">\n <fa-icon [icon]=\"faDownload\" size=\"lg\" />\n </button>\n }\n </div>\n @if (views()?.length > 1) {\n <div class=\"column is-narrow\">\n <div class=\"field has-addons\">\n @for (view of views(); track view) {\n <div class=\"control\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView() === view\"\n (click)=\"selectedView.set(view)\">\n <span class=\"icon is-small\">\n <fa-icon [icon]=\"viewIcon[view]\" aria-hidden=\"true\" />\n </span>\n <span class=\"is-hidden-mobile\">{{ view }}</span>\n </button>\n </div>\n }\n </div>\n </div>\n }\n</div>\n\n@switch (selectedView()) {\n @case (View.table) {\n <div class=\"px-3 pb-3\">\n @if (hasData()) {\n <he-data-table class=\"mb-1 is-small\" [small]=\"true\" maxHeight=\"320\">\n <table class=\"table is-narrow is-striped\">\n <thead>\n <tr class=\"has-text-weight-semibold\">\n <th class=\"width-auto has-border-right\"></th>\n @for (completeness of completenessKeys(); track completeness) {\n <th [attr.title]=\"completeness\">\n <a [href]=\"schemaBaseUrl + '/Completeness#' + completeness\" target=\"_blank\">\n {{ keyToLabel(completeness) }}\n </a>\n </th>\n }\n </tr>\n </thead>\n <tbody>\n @for (cycle of cycles(); track trackById(i, cycle); let i = $index) {\n <tr>\n <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n <he-node-link [node]=\"cycle\">\n <span>{{ i + 1 }}. {{ defaultLabel(cycle) }}</span>\n </he-node-link>\n </td>\n @for (key of completenessKeys(); track key) {\n <td class=\"is-nowrap\">\n <span>{{ getCompleteness(cycle)[key] ? 'Complete' : 'Incomplete' }}</span>\n <he-blank-node-state\n class=\"ml-1\"\n [dataState]=\"dataState()\"\n [node]=\"getCompleteness(cycle)\"\n [key]=\"key\"></he-blank-node-state>\n </td>\n }\n </tr>\n }\n </tbody>\n </table>\n </he-data-table>\n <he-blank-node-state-notice [dataState]=\"dataState()\"></he-blank-node-state-notice>\n } @else {\n <div class=\"panel-block\">\n <span>No completeness data</span>\n </div>\n }\n </div>\n }\n @case (View.logs) {\n @if (cycles().length > 1) {\n <div class=\"field has-addons pt-2 px-3\">\n <div class=\"control\">\n <span class=\"button is-small is-static is-secondary\">Select 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 @for (value of cycles(); track value; let i = $index) {\n <option [value]=\"i\">{{ i + 1 }}. {{ defaultLabel(value) }}</option>\n }\n </select>\n </div>\n </div>\n </div>\n }\n @if (selectedNode()) {\n <he-node-logs-models\n [node]=\"selectedNode()\"\n [nodeKey]=\"nodeKey\"\n [logsKey]=\"selectedLogsKey()\"\n [originalValues]=\"selectedOriginalValues()\"\n [recalculatedValues]=\"selectedRecalculatedValues()\"></he-node-logs-models>\n }\n }\n}\n", styles: [""], dependencies: [{ kind: "component", type: FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "mask", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "transform", "a11yRole"] }, { kind: "component", type: DataTableComponent, selector: "he-data-table", inputs: ["minHeight", "maxHeight", "small"] }, { kind: "component", type: NodeLinkComponent, selector: "he-node-link", inputs: ["node", "showExternalLink", "linkClass"] }, { kind: "component", type: BlankNodeStateComponent, selector: "he-blank-node-state", inputs: ["dataState", "nodeType", "dataKey", "key", "node", "state", "linkClass"] }, { kind: "component", type: BlankNodeStateNoticeComponent, selector: "he-blank-node-state-notice", inputs: ["dataState", "showDeleted"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i1.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "component", type: NodeLogsModelsComponent, selector: "he-node-logs-models", inputs: ["node", "nodeKey", "originalValues", "recalculatedValues", "terms", "filterTermTypes", "filterTermTypesLabel", "logsKey", "noDataMessage"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
96
96
|
}
|
|
97
97
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.11", ngImport: i0, type: CyclesCompletenessComponent, decorators: [{
|
|
98
98
|
type: Component,
|
|
@@ -105,6 +105,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.11", ngImpo
|
|
|
105
105
|
BlankNodeStateNoticeComponent,
|
|
106
106
|
FormsModule,
|
|
107
107
|
NodeLogsModelsComponent
|
|
108
|
-
], template: "<div class=\"columns is-variable is-align-items-center is-2 m-0\">\n <div class=\"column\">\n @if (selectedView() === View.table) {\n <button class=\"button is-small is-ghost\" (click)=\"showDownload()\">\n <fa-icon [icon]=\"faDownload\" size=\"lg\"
|
|
108
|
+
], template: "<div class=\"columns is-variable is-align-items-center is-2 m-0\">\n <div class=\"column\">\n @if (selectedView() === View.table) {\n <button class=\"button is-small is-ghost\" (click)=\"showDownload()\">\n <fa-icon [icon]=\"faDownload\" size=\"lg\" />\n </button>\n }\n </div>\n @if (views()?.length > 1) {\n <div class=\"column is-narrow\">\n <div class=\"field has-addons\">\n @for (view of views(); track view) {\n <div class=\"control\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView() === view\"\n (click)=\"selectedView.set(view)\">\n <span class=\"icon is-small\">\n <fa-icon [icon]=\"viewIcon[view]\" aria-hidden=\"true\" />\n </span>\n <span class=\"is-hidden-mobile\">{{ view }}</span>\n </button>\n </div>\n }\n </div>\n </div>\n }\n</div>\n\n@switch (selectedView()) {\n @case (View.table) {\n <div class=\"px-3 pb-3\">\n @if (hasData()) {\n <he-data-table class=\"mb-1 is-small\" [small]=\"true\" maxHeight=\"320\">\n <table class=\"table is-narrow is-striped\">\n <thead>\n <tr class=\"has-text-weight-semibold\">\n <th class=\"width-auto has-border-right\"></th>\n @for (completeness of completenessKeys(); track completeness) {\n <th [attr.title]=\"completeness\">\n <a [href]=\"schemaBaseUrl + '/Completeness#' + completeness\" target=\"_blank\">\n {{ keyToLabel(completeness) }}\n </a>\n </th>\n }\n </tr>\n </thead>\n <tbody>\n @for (cycle of cycles(); track trackById(i, cycle); let i = $index) {\n <tr>\n <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n <he-node-link [node]=\"cycle\">\n <span>{{ i + 1 }}. {{ defaultLabel(cycle) }}</span>\n </he-node-link>\n </td>\n @for (key of completenessKeys(); track key) {\n <td class=\"is-nowrap\">\n <span>{{ getCompleteness(cycle)[key] ? 'Complete' : 'Incomplete' }}</span>\n <he-blank-node-state\n class=\"ml-1\"\n [dataState]=\"dataState()\"\n [node]=\"getCompleteness(cycle)\"\n [key]=\"key\"></he-blank-node-state>\n </td>\n }\n </tr>\n }\n </tbody>\n </table>\n </he-data-table>\n <he-blank-node-state-notice [dataState]=\"dataState()\"></he-blank-node-state-notice>\n } @else {\n <div class=\"panel-block\">\n <span>No completeness data</span>\n </div>\n }\n </div>\n }\n @case (View.logs) {\n @if (cycles().length > 1) {\n <div class=\"field has-addons pt-2 px-3\">\n <div class=\"control\">\n <span class=\"button is-small is-static is-secondary\">Select 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 @for (value of cycles(); track value; let i = $index) {\n <option [value]=\"i\">{{ i + 1 }}. {{ defaultLabel(value) }}</option>\n }\n </select>\n </div>\n </div>\n </div>\n }\n @if (selectedNode()) {\n <he-node-logs-models\n [node]=\"selectedNode()\"\n [nodeKey]=\"nodeKey\"\n [logsKey]=\"selectedLogsKey()\"\n [originalValues]=\"selectedOriginalValues()\"\n [recalculatedValues]=\"selectedRecalculatedValues()\"></he-node-logs-models>\n }\n }\n}\n" }]
|
|
109
109
|
}], ctorParameters: () => [] });
|
|
110
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cycles-completeness.component.js","sourceRoot":"","sources":["../../../../src/cycles/cycles-completeness/cycles-completeness.component.ts","../../../../src/cycles/cycles-completeness/cycles-completeness.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, computed, effect, inject, input, signal } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { FormsModule } from '@angular/forms';\nimport { FaIconComponent } from '@fortawesome/angular-fontawesome';\nimport { IconDefinition, faCalculator, faDownload, faList } from '@fortawesome/free-solid-svg-icons';\nimport { NgbModal, NgbTooltip } from '@ng-bootstrap/ng-bootstrap';\nimport { DataState } from '@hestia-earth/api';\nimport { ICycleJSONLD, NodeType, NonBlankNodesKey } from '@hestia-earth/schema';\nimport { keyToLabel } from '@hestia-earth/utils';\n\nimport { HeNodeStoreService } from '../../node/node-store.service';\nimport { isValidKey } from '../../common/node-utils';\nimport { defaultLabel, schemaBaseUrl } from '../../common/utils';\nimport { logsKey } from '../cycles.model';\nimport { NodeCsvExportConfirmComponent } from '../../node/node-csv-export-confirm/node-csv-export-confirm.component';\nimport { NodeLogsModelsComponent } from '../../node/node-logs-models/node-logs-models.component';\nimport { BlankNodeStateNoticeComponent } from '../../common/blank-node-state-notice/blank-node-state-notice.component';\nimport { BlankNodeStateComponent } from '../../common/blank-node-state/blank-node-state.component';\nimport { NodeLinkComponent } from '../../node/node-link/node-link.component';\nimport { DataTableComponent } from '../../common/data-table/data-table.component';\n\nenum View {\n  table = 'Table view',\n  logs = 'Recalculations logs'\n}\n\nconst viewIcon: {\n  [view in View]: IconDefinition;\n} = {\n  [View.logs]: faCalculator,\n  [View.table]: faList\n};\n\nconst headerKeys = ['cycle.id', 'cycle.@id', 'cycle.completeness.'];\n\n// backward compatibility with schema version < 14\nconst getCompleteness = (cycle: any) => cycle?.completeness || cycle?.dataCompleteness || {};\n\n@Component({\n  selector: 'he-cycles-completeness',\n  templateUrl: './cycles-completeness.component.html',\n  styleUrls: ['./cycles-completeness.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [\n    NgbTooltip,\n    FaIconComponent,\n    DataTableComponent,\n    NodeLinkComponent,\n    BlankNodeStateComponent,\n    BlankNodeStateNoticeComponent,\n    FormsModule,\n    NodeLogsModelsComponent\n  ]\n})\nexport class CyclesCompletenessComponent {\n  private readonly modalService = inject(NgbModal);\n  private readonly nodeStoreService = inject(HeNodeStoreService);\n\n  protected readonly faDownload = faDownload;\n\n  protected readonly dataState = input(DataState.original);\n\n  protected readonly schemaBaseUrl = schemaBaseUrl();\n  protected readonly defaultLabel = defaultLabel;\n  protected readonly keyToLabel = keyToLabel;\n  protected readonly getCompleteness = getCompleteness;\n  protected readonly nodeKey = NonBlankNodesKey.completeness;\n\n  protected readonly View = View;\n  protected readonly viewIcon = viewIcon;\n  private readonly showView = computed(\n    () =>\n      ({\n        [View.logs]: !this.isOriginal(),\n        [View.table]: true\n      }) as { [view in View]: boolean }\n  );\n  protected readonly views = computed(() => Object.values(View).filter(view => this.showView()[view]) ?? []);\n  protected readonly selectedView = signal(View.table);\n\n  private readonly originalCycles = toSignal(\n    this.nodeStoreService.findByState$<ICycleJSONLD>(NodeType.Cycle, DataState.original)\n  );\n  private readonly _allCycles = toSignal(this.nodeStoreService.find$<ICycleJSONLD>(NodeType.Cycle));\n  protected readonly cycles = computed(() => this._allCycles()?.map(data => data[this.dataState()]) || []);\n\n  private readonly selectedIndex = signal(0);\n  private readonly ogirinalSelectedCycle = computed(() => this.originalCycles()?.[this.selectedIndex()]);\n  private readonly selectedCycle = computed(() => this.cycles()?.[this.selectedIndex()]);\n  protected readonly selectedLogsKey = computed(() => logsKey(this.selectedCycle()));\n  protected readonly selectedOriginalValues = computed(() => getCompleteness(this.ogirinalSelectedCycle()));\n  protected readonly selectedRecalculatedValues = computed(() => getCompleteness(this.selectedCycle()));\n  protected readonly selectedNode = computed(() =>\n    this.selectedCycle()\n      ? {\n          ...this.selectedCycle(),\n          '@type': NodeType.Cycle,\n          type: NodeType.Cycle,\n          dataState: DataState.recalculated\n        }\n      : null\n  );\n\n  protected readonly hasData = computed(() => this.cycles().length);\n  protected readonly isOriginal = computed(() => this.dataState() === DataState.original);\n\n  protected readonly completenessKeys = computed(() =>\n    Object.keys(getCompleteness(this.selectedCycle())).filter(isValidKey).sort()\n  );\n\n  constructor() {\n    effect(() => {\n      // make sure logs does not remain displayed when switching back to original view\n      if (this.isOriginal() && this.selectedView() === View.logs) {\n        this.selectedView.set(View.table);\n      }\n    });\n  }\n\n  protected trackById(_index: number, item: ICycleJSONLD) {\n    return item['@id'];\n  }\n\n  protected selectIndex({ target: { value } }) {\n    this.selectedIndex.set(+value);\n  }\n\n  protected showDownload() {\n    const instance = this.modalService.open(NodeCsvExportConfirmComponent);\n    const component = instance.componentInstance as NodeCsvExportConfirmComponent;\n    component.nodes.set(this.cycles());\n    component.filename.set('completeness.csv');\n    component.isUpload.set(false);\n    component.headerKeys.set(headerKeys);\n  }\n}\n","<div class=\"columns is-variable is-align-items-center is-2 m-0\">\n  <div class=\"column\">\n    @if (selectedView() === View.table) {\n      <button class=\"button is-small is-ghost\" (click)=\"showDownload()\">\n        <fa-icon [icon]=\"faDownload\" size=\"lg\"></fa-icon>\n      </button>\n    }\n  </div>\n  @if (views()?.length > 1) {\n    <div class=\"column is-narrow\">\n      <div class=\"field has-addons\">\n        @for (view of views(); track view) {\n          <div class=\"control\">\n            <button\n              class=\"button is-small\"\n              [class.is-selected]=\"selectedView() === view\"\n              (click)=\"selectedView.set(view)\">\n              <span class=\"icon is-small\">\n                <fa-icon [icon]=\"viewIcon[view]\" aria-hidden=\"true\"></fa-icon>\n              </span>\n              <span class=\"is-hidden-mobile\">{{ view }}</span>\n            </button>\n          </div>\n        }\n      </div>\n    </div>\n  }\n</div>\n\n@switch (selectedView()) {\n  @case (View.table) {\n    <div class=\"px-3 pb-3\">\n      @if (hasData()) {\n        <he-data-table class=\"mb-1 is-small\" [small]=\"true\" maxHeight=\"320\">\n          <table class=\"table is-narrow is-striped\">\n            <thead>\n              <tr class=\"has-text-weight-semibold\">\n                <th class=\"width-auto has-border-right\"></th>\n                @for (completeness of completenessKeys(); track completeness) {\n                  <th [attr.title]=\"completeness\">\n                    <a [href]=\"schemaBaseUrl + '/Completeness#' + completeness\" target=\"_blank\">\n                      {{ keyToLabel(completeness) }}\n                    </a>\n                  </th>\n                }\n              </tr>\n            </thead>\n            <tbody>\n              @for (cycle of cycles(); track trackById(i, cycle); let i = $index) {\n                <tr>\n                  <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n                    <he-node-link [node]=\"cycle\">\n                      <span>{{ i + 1 }}. {{ defaultLabel(cycle) }}</span>\n                    </he-node-link>\n                  </td>\n                  @for (key of completenessKeys(); track key) {\n                    <td class=\"is-nowrap\">\n                      <span>{{ getCompleteness(cycle)[key] ? 'Complete' : 'Incomplete' }}</span>\n                      <he-blank-node-state\n                        class=\"ml-1\"\n                        [dataState]=\"dataState()\"\n                        [node]=\"getCompleteness(cycle)\"\n                        [key]=\"key\"></he-blank-node-state>\n                    </td>\n                  }\n                </tr>\n              }\n            </tbody>\n          </table>\n        </he-data-table>\n        <he-blank-node-state-notice [dataState]=\"dataState()\"></he-blank-node-state-notice>\n      } @else {\n        <div class=\"panel-block\">\n          <span>No completeness data</span>\n        </div>\n      }\n    </div>\n  }\n  @case (View.logs) {\n    @if (cycles().length > 1) {\n      <div class=\"field has-addons pt-2 px-3\">\n        <div class=\"control\">\n          <span class=\"button is-small is-static is-secondary\">Select 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              @for (value of cycles(); track value; let i = $index) {\n                <option [value]=\"i\">{{ i + 1 }}. {{ defaultLabel(value) }}</option>\n              }\n            </select>\n          </div>\n        </div>\n      </div>\n    }\n    @if (selectedNode()) {\n      <he-node-logs-models\n        [node]=\"selectedNode()\"\n        [nodeKey]=\"nodeKey\"\n        [logsKey]=\"selectedLogsKey()\"\n        [originalValues]=\"selectedOriginalValues()\"\n        [recalculatedValues]=\"selectedRecalculatedValues()\"></he-node-logs-models>\n    }\n  }\n}\n"]}
|
|
110
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cycles-completeness.component.js","sourceRoot":"","sources":["../../../../src/cycles/cycles-completeness/cycles-completeness.component.ts","../../../../src/cycles/cycles-completeness/cycles-completeness.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, computed, effect, inject, input, signal } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { FormsModule } from '@angular/forms';\nimport { FaIconComponent } from '@fortawesome/angular-fontawesome';\nimport { IconDefinition, faCalculator, faDownload, faList } from '@fortawesome/free-solid-svg-icons';\nimport { NgbModal, NgbTooltip } from '@ng-bootstrap/ng-bootstrap';\nimport { DataState } from '@hestia-earth/api';\nimport { ICycleJSONLD, NodeType, NonBlankNodesKey } from '@hestia-earth/schema';\nimport { keyToLabel } from '@hestia-earth/utils';\n\nimport { HeNodeStoreService } from '../../node/node-store.service';\nimport { isValidKey } from '../../common/node-utils';\nimport { defaultLabel, schemaBaseUrl } from '../../common/utils';\nimport { logsKey } from '../cycles.model';\nimport { NodeCsvExportConfirmComponent } from '../../node/node-csv-export-confirm/node-csv-export-confirm.component';\nimport { NodeLogsModelsComponent } from '../../node/node-logs-models/node-logs-models.component';\nimport { BlankNodeStateNoticeComponent } from '../../common/blank-node-state-notice/blank-node-state-notice.component';\nimport { BlankNodeStateComponent } from '../../common/blank-node-state/blank-node-state.component';\nimport { NodeLinkComponent } from '../../node/node-link/node-link.component';\nimport { DataTableComponent } from '../../common/data-table/data-table.component';\n\nenum View {\n  table = 'Table view',\n  logs = 'Recalculations logs'\n}\n\nconst viewIcon: {\n  [view in View]: IconDefinition;\n} = {\n  [View.logs]: faCalculator,\n  [View.table]: faList\n};\n\nconst headerKeys = ['cycle.id', 'cycle.@id', 'cycle.completeness.'];\n\n// backward compatibility with schema version < 14\nconst getCompleteness = (cycle: any) => cycle?.completeness || cycle?.dataCompleteness || {};\n\n@Component({\n  selector: 'he-cycles-completeness',\n  templateUrl: './cycles-completeness.component.html',\n  styleUrls: ['./cycles-completeness.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [\n    NgbTooltip,\n    FaIconComponent,\n    DataTableComponent,\n    NodeLinkComponent,\n    BlankNodeStateComponent,\n    BlankNodeStateNoticeComponent,\n    FormsModule,\n    NodeLogsModelsComponent\n  ]\n})\nexport class CyclesCompletenessComponent {\n  private readonly modalService = inject(NgbModal);\n  private readonly nodeStoreService = inject(HeNodeStoreService);\n\n  protected readonly faDownload = faDownload;\n\n  protected readonly dataState = input(DataState.original);\n\n  protected readonly schemaBaseUrl = schemaBaseUrl();\n  protected readonly defaultLabel = defaultLabel;\n  protected readonly keyToLabel = keyToLabel;\n  protected readonly getCompleteness = getCompleteness;\n  protected readonly nodeKey = NonBlankNodesKey.completeness;\n\n  protected readonly View = View;\n  protected readonly viewIcon = viewIcon;\n  private readonly showView = computed(\n    () =>\n      ({\n        [View.logs]: !this.isOriginal(),\n        [View.table]: true\n      }) as { [view in View]: boolean }\n  );\n  protected readonly views = computed(() => Object.values(View).filter(view => this.showView()[view]) ?? []);\n  protected readonly selectedView = signal(View.table);\n\n  private readonly originalCycles = toSignal(\n    this.nodeStoreService.findByState$<ICycleJSONLD>(NodeType.Cycle, DataState.original)\n  );\n  private readonly _allCycles = toSignal(this.nodeStoreService.find$<ICycleJSONLD>(NodeType.Cycle));\n  protected readonly cycles = computed(() => this._allCycles()?.map(data => data[this.dataState()]) || []);\n\n  private readonly selectedIndex = signal(0);\n  private readonly ogirinalSelectedCycle = computed(() => this.originalCycles()?.[this.selectedIndex()]);\n  private readonly selectedCycle = computed(() => this.cycles()?.[this.selectedIndex()]);\n  protected readonly selectedLogsKey = computed(() => logsKey(this.selectedCycle()));\n  protected readonly selectedOriginalValues = computed(() => getCompleteness(this.ogirinalSelectedCycle()));\n  protected readonly selectedRecalculatedValues = computed(() => getCompleteness(this.selectedCycle()));\n  protected readonly selectedNode = computed(() =>\n    this.selectedCycle()\n      ? {\n          ...this.selectedCycle(),\n          '@type': NodeType.Cycle,\n          type: NodeType.Cycle,\n          dataState: DataState.recalculated\n        }\n      : null\n  );\n\n  protected readonly hasData = computed(() => this.cycles().length);\n  protected readonly isOriginal = computed(() => this.dataState() === DataState.original);\n\n  protected readonly completenessKeys = computed(() =>\n    Object.keys(getCompleteness(this.selectedCycle())).filter(isValidKey).sort()\n  );\n\n  constructor() {\n    effect(() => {\n      // make sure logs does not remain displayed when switching back to original view\n      if (this.isOriginal() && this.selectedView() === View.logs) {\n        this.selectedView.set(View.table);\n      }\n    });\n  }\n\n  protected trackById(_index: number, item: ICycleJSONLD) {\n    return item['@id'];\n  }\n\n  protected selectIndex({ target: { value } }) {\n    this.selectedIndex.set(+value);\n  }\n\n  protected showDownload() {\n    const instance = this.modalService.open(NodeCsvExportConfirmComponent);\n    const component = instance.componentInstance as NodeCsvExportConfirmComponent;\n    component.nodes.set(this.cycles());\n    component.filename.set('completeness.csv');\n    component.isUpload.set(false);\n    component.headerKeys.set(headerKeys);\n  }\n}\n","<div class=\"columns is-variable is-align-items-center is-2 m-0\">\n  <div class=\"column\">\n    @if (selectedView() === View.table) {\n      <button class=\"button is-small is-ghost\" (click)=\"showDownload()\">\n        <fa-icon [icon]=\"faDownload\" size=\"lg\" />\n      </button>\n    }\n  </div>\n  @if (views()?.length > 1) {\n    <div class=\"column is-narrow\">\n      <div class=\"field has-addons\">\n        @for (view of views(); track view) {\n          <div class=\"control\">\n            <button\n              class=\"button is-small\"\n              [class.is-selected]=\"selectedView() === view\"\n              (click)=\"selectedView.set(view)\">\n              <span class=\"icon is-small\">\n                <fa-icon [icon]=\"viewIcon[view]\" aria-hidden=\"true\" />\n              </span>\n              <span class=\"is-hidden-mobile\">{{ view }}</span>\n            </button>\n          </div>\n        }\n      </div>\n    </div>\n  }\n</div>\n\n@switch (selectedView()) {\n  @case (View.table) {\n    <div class=\"px-3 pb-3\">\n      @if (hasData()) {\n        <he-data-table class=\"mb-1 is-small\" [small]=\"true\" maxHeight=\"320\">\n          <table class=\"table is-narrow is-striped\">\n            <thead>\n              <tr class=\"has-text-weight-semibold\">\n                <th class=\"width-auto has-border-right\"></th>\n                @for (completeness of completenessKeys(); track completeness) {\n                  <th [attr.title]=\"completeness\">\n                    <a [href]=\"schemaBaseUrl + '/Completeness#' + completeness\" target=\"_blank\">\n                      {{ keyToLabel(completeness) }}\n                    </a>\n                  </th>\n                }\n              </tr>\n            </thead>\n            <tbody>\n              @for (cycle of cycles(); track trackById(i, cycle); let i = $index) {\n                <tr>\n                  <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n                    <he-node-link [node]=\"cycle\">\n                      <span>{{ i + 1 }}. {{ defaultLabel(cycle) }}</span>\n                    </he-node-link>\n                  </td>\n                  @for (key of completenessKeys(); track key) {\n                    <td class=\"is-nowrap\">\n                      <span>{{ getCompleteness(cycle)[key] ? 'Complete' : 'Incomplete' }}</span>\n                      <he-blank-node-state\n                        class=\"ml-1\"\n                        [dataState]=\"dataState()\"\n                        [node]=\"getCompleteness(cycle)\"\n                        [key]=\"key\"></he-blank-node-state>\n                    </td>\n                  }\n                </tr>\n              }\n            </tbody>\n          </table>\n        </he-data-table>\n        <he-blank-node-state-notice [dataState]=\"dataState()\"></he-blank-node-state-notice>\n      } @else {\n        <div class=\"panel-block\">\n          <span>No completeness data</span>\n        </div>\n      }\n    </div>\n  }\n  @case (View.logs) {\n    @if (cycles().length > 1) {\n      <div class=\"field has-addons pt-2 px-3\">\n        <div class=\"control\">\n          <span class=\"button is-small is-static is-secondary\">Select 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              @for (value of cycles(); track value; let i = $index) {\n                <option [value]=\"i\">{{ i + 1 }}. {{ defaultLabel(value) }}</option>\n              }\n            </select>\n          </div>\n        </div>\n      </div>\n    }\n    @if (selectedNode()) {\n      <he-node-logs-models\n        [node]=\"selectedNode()\"\n        [nodeKey]=\"nodeKey\"\n        [logsKey]=\"selectedLogsKey()\"\n        [originalValues]=\"selectedOriginalValues()\"\n        [recalculatedValues]=\"selectedRecalculatedValues()\"></he-node-logs-models>\n    }\n  }\n}\n"]}
|
|
@@ -102,7 +102,7 @@ export class CyclesNodesComponent {
|
|
|
102
102
|
this.viewIcon = viewIcon;
|
|
103
103
|
this.showView = computed(() => ({
|
|
104
104
|
[View.chart]: this.selectedNodeKey() === BlankNodesKey.emissions ? this.cycles().length > 1 : this.hasData(),
|
|
105
|
-
[View.logs]: !this.isOriginal(),
|
|
105
|
+
[View.logs]: !this.isOriginal() && this.hasRecalculatedNodes(),
|
|
106
106
|
[View.table]: true,
|
|
107
107
|
[View.timeline]: this.enableTimeline()
|
|
108
108
|
}));
|
|
@@ -135,7 +135,7 @@ export class CyclesNodesComponent {
|
|
|
135
135
|
: null);
|
|
136
136
|
this.isOriginal = computed(() => this.dataState() === DataState.original);
|
|
137
137
|
this.originalValues = computed(() => (this.isOriginal() ? null : this.originalCycles()) || []);
|
|
138
|
-
this.hasRecalculatedNodes = computed(() => this.cycles().some(
|
|
138
|
+
this.hasRecalculatedNodes = computed(() => this.cycles().some(({ aggregated }) => !aggregated));
|
|
139
139
|
this.showSwitchToRecalculated = computed(() => this.isOriginal() && this.hasRecalculatedNodes());
|
|
140
140
|
this.timelineValues = computed(() => filterValuesTimeline(this.selectedCycle()?.[this.selectedNodeKey()] || []));
|
|
141
141
|
this.enableTimeline = computed(() => this.timelineValues().length > 0);
|
|
@@ -201,7 +201,7 @@ export class CyclesNodesComponent {
|
|
|
201
201
|
component.headerKeys.set(this.headerKeys());
|
|
202
202
|
}
|
|
203
203
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.11", ngImport: i0, type: CyclesNodesComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
204
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.11", type: CyclesNodesComponent, isStandalone: true, selector: "he-cycles-nodes", inputs: { dataState: { classPropertyName: "dataState", publicName: "dataState", isSignal: true, isRequired: false, transformFunction: null }, nodeKeys: { classPropertyName: "nodeKeys", publicName: "nodeKeys", isSignal: true, isRequired: true, transformFunction: null }, nodeKeyGroup: { classPropertyName: "nodeKeyGroup", publicName: "nodeKeyGroup", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: "@if (isGroupNode()) {\n <div class=\"tabs is-mb-1\">\n <ul>\n @for (term of groupNodeTerms(); track term.name) {\n <li [class.is-active]=\"selectedGroupTerm() === term\">\n <a (click)=\"selectedGroupTerm.set(term)\">{{ term.name }}</a>\n </li>\n }\n </ul>\n </div>\n}\n\n<div class=\"columns is-variable is-align-items-center is-2 is-m-0\">\n <div class=\"column\">\n @if (hasData() && selectedView() === View.table) {\n <button class=\"button is-small is-ghost\" (click)=\"showDownload()\">\n <fa-icon [icon]=\"faDownload\" size=\"lg\"></fa-icon>\n </button>\n }\n </div>\n @if (selectedView() === View.table) {\n <div class=\"column is-narrow\">\n <he-search-extend\n class=\"is-secondary is-small\"\n placeholder=\"Filter terms by name\"\n (search)=\"filterTerm.set($event)\"></he-search-extend>\n </div>\n }\n @if (hasData() && views()?.length > 1) {\n <div class=\"column is-narrow\">\n <div class=\"field has-addons\">\n @for (view of views(); track view) {\n <div class=\"control\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView() === view\"\n (click)=\"selectedView.set(view)\">\n <span class=\"icon is-small\">\n <fa-icon [icon]=\"viewIcon[view]\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span class=\"is-hidden-mobile\">{{ view }}</span>\n </button>\n </div>\n }\n </div>\n </div>\n }\n</div>\n\n@if (isNodeKeyAllowed()) {\n @switch (selectedView()) {\n @case (View.table) {\n <div class=\"is-px-3 is-pb-3\">\n @if (hasData()) {\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 @if (dataKeys().length > 1) {\n <tr class=\"has-text-weight-bold\">\n <th class=\"width-auto\"></th>\n <th [class.is-hidden]=\"isGroupNode()\"></th>\n @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n @if (data()[dataKey]?.length) {\n <th [attr.colspan]=\"data()[dataKey].length\" [class.has-border-right]=\"!dataKeyLast\">\n <span>{{ dataKey | keyToLabel }}</span>\n </th>\n }\n }\n </tr>\n }\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]=\"isGroupNode()\"></th>\n @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n @for (node of data()[dataKey]; track node.value.term.name; let nodeLast = $last) {\n <th\n [attr.title]=\"node.value.term.name\"\n [class.has-border-right]=\"dataKeys().length > 1 && !dataKeyLast && nodeLast\">\n <he-node-link [node]=\"node.value.term\">\n <span\n [innerHtml]=\"\n node.value.term.name | ellipsis: 30 | compound: node.value.term.termType\n \"></span>\n </he-node-link>\n </th>\n }\n }\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]=\"isGroupNode()\">\n <a [href]=\"schemaBaseUrl + '/Cycle#functionalUnit'\" target=\"_blank\">Functional unit</a>\n </th>\n @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n @for (node of data()[dataKey]; track node.value.term.name; let nodeLast = $last) {\n <th\n [attr.title]=\"node.value.term.units\"\n [class.has-border-right]=\"dataKeys().length > 1 && !dataKeyLast && nodeLast\">\n <span [innerHtml]=\"node.value.term.units | compound: node.value.term.termType\"></span>\n <he-terms-units-description\n class=\"is-inline-block is-ml-2\"\n [term]=\"node.value.term\"></he-terms-units-description>\n </th>\n }\n }\n </tr>\n </thead>\n <tbody>\n @for (cycle of cycles(); track trackById(cycleIndex, cycle); let cycleIndex = $index) {\n <tr>\n <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n <he-node-link [node]=\"cycleNode(cycle)\">\n <span>{{ cycleIndex + 1 }}. {{ defaultLabel(cycle) }}</span>\n </he-node-link>\n </td>\n <td class=\"has-border-right\" [class.is-hidden]=\"isGroupNode()\">\n <he-cycles-functional-unit-measure [cycle]=\"cycle\"></he-cycles-functional-unit-measure>\n </td>\n @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n @for (node of data()[dataKey]; track node.value.term.name; let nodeLast = $last) {\n <td\n class=\"is-nowrap\"\n [class.has-border-right]=\"dataKeys().length > 1 && !dataKeyLast && nodeLast\">\n @if (node.value.values[cycle['@id']]) {\n <span\n class=\"trigger-popover\"\n [ngbPopover]=\"details\"\n autoClose=\"outside\"\n popoverClass=\"is-narrow\"\n placement=\"left\"\n container=\"body\"\n [popoverContext]=\"{ data: node.value.values[cycle['@id']], cycle, key: dataKey }\">\n <span pointer>\n {{\n propertyValue(node.value.values[cycle['@id']].value, node.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]=\"node.value.values[cycle['@id']].node\"\n key=\"value\"></he-blank-node-state>\n </span>\n } @else {\n <span>-</span>\n }\n </td>\n }\n }\n </tr>\n }\n </tbody>\n </table>\n </he-data-table>\n <he-blank-node-state-notice\n [dataState]=\"dataState()\"\n [showDeleted]=\"firstNodeKey() === BlankNodesKey.emissions\"></he-blank-node-state-notice>\n } @else {\n <div class=\"is-pt-3 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 @if (showSwitchToRecalculated()) {\n <span>\n Switch to\n <code>recalculated</code>\n version.\n </span>\n }\n </div>\n }\n </div>\n }\n @case (View.chart) {\n @switch (firstNodeKey()) {\n @case (BlankNodesKey.inputs) {\n <he-cycles-result [cycles]=\"cycles()\"></he-cycles-result>\n }\n @case (BlankNodesKey.emissions) {\n <he-cycles-emissions-chart [cycles]=\"cycles()\"></he-cycles-emissions-chart>\n }\n }\n }\n @case (View.timeline) {\n <ng-container *ngTemplateOutlet=\"selectCycle\"></ng-container>\n <div class=\"is-px-3 is-pb-3\">\n <he-cycles-nodes-timeline\n [recalculatedValues]=\"timelineValues()\"\n [dataState]=\"dataState()\"></he-cycles-nodes-timeline>\n <he-blank-node-state-notice [dataState]=\"dataState()\"></he-blank-node-state-notice>\n </div>\n }\n @case (View.logs) {\n <ng-container *ngTemplateOutlet=\"selectCycle\"></ng-container>\n @if (nodeKeys().length > 1) {\n <div class=\"tabs is-mb-2\">\n <ul>\n @for (nodeKey of nodeKeys(); track nodeKey) {\n <li [class.is-active]=\"selectedNodeKey() === nodeKey\">\n <a (click)=\"selectedNodeKey.set(nodeKey)\">{{ nodeKey | keyToLabel }}</a>\n </li>\n }\n </ul>\n </div>\n }\n @if (selectedNode()) {\n <he-node-logs-models\n [node]=\"selectedNode()\"\n [nodeKey]=\"selectedNodeKey()\"\n [logsKey]=\"selectedLogsKey()\"\n [originalValues]=\"selectedOriginalValues()\"\n [recalculatedValues]=\"selectedRecalculatedValues()\"\n [filterTermTypes]=\"filterTermTypes()\"></he-node-logs-models>\n }\n }\n }\n}\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\n<ng-template #selectCycle>\n @if (cycles().length > 1) {\n <div class=\"field has-addons is-py-2 is-px-3\">\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 @for (value of cycles(); track value; let cycleIndex = $index) {\n <option [value]=\"cycleIndex\">{{ cycleIndex + 1 }}. {{ defaultLabel(value) }}</option>\n }\n </select>\n </div>\n </div>\n </div>\n }\n</ng-template>\n", styles: ["fa-icon{display:inline-block;width:10px}he-data-table ::ng-deep .table thead tr th:nth-child(2),he-data-table ::ng-deep .table tbody tr td:nth-child(2){max-width:102px;width:102px}\n"], dependencies: [{ kind: "component", type: FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "mask", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "transform", "a11yRole"] }, { kind: "component", type: SearchExtendComponent, selector: "he-search-extend", inputs: ["value", "disabled", "placeholder", "class"], outputs: ["valueChange", "search"] }, { kind: "component", type: DataTableComponent, selector: "he-data-table", inputs: ["minHeight", "maxHeight", "small"] }, { kind: "component", type: NodeLinkComponent, selector: "he-node-link", inputs: ["node", "showExternalLink", "linkClass"] }, { kind: "component", type: TermsUnitsDescriptionComponent, selector: "he-terms-units-description", inputs: ["term", "iconTemplate"] }, { kind: "component", type: CyclesFunctionalUnitMeasureComponent, selector: "he-cycles-functional-unit-measure", inputs: ["cycle"] }, { kind: "directive", type: NgbPopover, selector: "[ngbPopover]", inputs: ["animation", "autoClose", "ngbPopover", "popoverTitle", "placement", "popperOptions", "triggers", "positionTarget", "container", "disablePopover", "popoverClass", "popoverContext", "openDelay", "closeDelay"], outputs: ["shown", "hidden"], exportAs: ["ngbPopover"] }, { kind: "component", type: BlankNodeStateComponent, selector: "he-blank-node-state", inputs: ["dataState", "nodeType", "dataKey", "key", "node", "state", "linkClass"] }, { kind: "component", type: BlankNodeStateNoticeComponent, selector: "he-blank-node-state-notice", inputs: ["dataState", "showDeleted"] }, { kind: "component", type: CyclesResultComponent, selector: "he-cycles-result", inputs: ["cycles"] }, { kind: "component", type: CyclesEmissionsChartComponent, selector: "he-cycles-emissions-chart", inputs: ["cycles"] }, { kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: CyclesNodesTimelineComponent, selector: "he-cycles-nodes-timeline", inputs: ["recalculatedValues", "dataState"] }, { kind: "component", type: NodeLogsModelsComponent, selector: "he-node-logs-models", inputs: ["node", "nodeKey", "originalValues", "recalculatedValues", "terms", "filterTermTypes", "filterTermTypesLabel", "logsKey", "noDataMessage"] }, { kind: "component", type: NodeValueDetailsComponent, selector: "he-node-value-details", inputs: ["data", "nodeType", "dataState", "dataKey"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i1.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "pipe", type: CompoundPipe, name: "compound" }, { kind: "pipe", type: DefaultPipe, name: "default" }, { kind: "pipe", type: EllipsisPipe, name: "ellipsis" }, { kind: "pipe", type: KeyToLabelPipe, name: "keyToLabel" }, { kind: "pipe", type: PrecisionPipe, name: "precision" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
204
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.11", type: CyclesNodesComponent, isStandalone: true, selector: "he-cycles-nodes", inputs: { dataState: { classPropertyName: "dataState", publicName: "dataState", isSignal: true, isRequired: false, transformFunction: null }, nodeKeys: { classPropertyName: "nodeKeys", publicName: "nodeKeys", isSignal: true, isRequired: true, transformFunction: null }, nodeKeyGroup: { classPropertyName: "nodeKeyGroup", publicName: "nodeKeyGroup", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: "@if (isGroupNode()) {\n <div class=\"tabs is-mb-1\">\n <ul>\n @for (term of groupNodeTerms(); track term.name) {\n <li [class.is-active]=\"selectedGroupTerm() === term\">\n <a (click)=\"selectedGroupTerm.set(term)\">{{ term.name }}</a>\n </li>\n }\n </ul>\n </div>\n}\n\n<div class=\"columns is-variable is-align-items-center is-2 is-m-0\">\n <div class=\"column\">\n @if (hasData() && selectedView() === View.table) {\n <button class=\"button is-small is-ghost\" (click)=\"showDownload()\">\n <fa-icon [icon]=\"faDownload\" size=\"lg\" />\n </button>\n }\n </div>\n @if (selectedView() === View.table) {\n <div class=\"column is-narrow\">\n <he-search-extend\n class=\"is-secondary is-small\"\n placeholder=\"Filter terms by name\"\n (search)=\"filterTerm.set($event)\" />\n </div>\n }\n @if (views()?.length > 1) {\n <div class=\"column is-narrow\">\n <div class=\"field has-addons\">\n @for (view of views(); track view) {\n <div class=\"control\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView() === view\"\n (click)=\"selectedView.set(view)\">\n <span class=\"icon is-small\">\n <fa-icon [icon]=\"viewIcon[view]\" aria-hidden=\"true\" />\n </span>\n <span class=\"is-hidden-mobile\">{{ view }}</span>\n </button>\n </div>\n }\n </div>\n </div>\n }\n</div>\n\n@if (isNodeKeyAllowed()) {\n @switch (selectedView()) {\n @case (View.table) {\n <div class=\"is-px-3 is-pb-3\">\n @if (hasData()) {\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 @if (dataKeys().length > 1) {\n <tr class=\"has-text-weight-bold\">\n <th class=\"width-auto\"></th>\n <th [class.is-hidden]=\"isGroupNode()\"></th>\n @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n @if (data()[dataKey]?.length) {\n <th [attr.colspan]=\"data()[dataKey].length\" [class.has-border-right]=\"!dataKeyLast\">\n <span>{{ dataKey | keyToLabel }}</span>\n </th>\n }\n }\n </tr>\n }\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]=\"isGroupNode()\"></th>\n @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n @for (node of data()[dataKey]; track node.value.term.name; let nodeLast = $last) {\n <th\n [attr.title]=\"node.value.term.name\"\n [class.has-border-right]=\"dataKeys().length > 1 && !dataKeyLast && nodeLast\">\n <he-node-link [node]=\"node.value.term\">\n <span\n [innerHtml]=\"\n node.value.term.name | ellipsis: 30 | compound: node.value.term.termType\n \"></span>\n </he-node-link>\n </th>\n }\n }\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]=\"isGroupNode()\">\n <a [href]=\"schemaBaseUrl + '/Cycle#functionalUnit'\" target=\"_blank\">Functional unit</a>\n </th>\n @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n @for (node of data()[dataKey]; track node.value.term.name; let nodeLast = $last) {\n <th\n [attr.title]=\"node.value.term.units\"\n [class.has-border-right]=\"dataKeys().length > 1 && !dataKeyLast && nodeLast\">\n <span [innerHtml]=\"node.value.term.units | compound: node.value.term.termType\"></span>\n <he-terms-units-description class=\"is-inline-block is-ml-2\" [term]=\"node.value.term\" />\n </th>\n }\n }\n </tr>\n </thead>\n <tbody>\n @for (cycle of cycles(); track trackById(cycleIndex, cycle); let cycleIndex = $index) {\n <tr>\n <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n <he-node-link [node]=\"cycleNode(cycle)\">\n <span>{{ cycleIndex + 1 }}. {{ defaultLabel(cycle) }}</span>\n </he-node-link>\n </td>\n <td class=\"has-border-right\" [class.is-hidden]=\"isGroupNode()\">\n <he-cycles-functional-unit-measure [cycle]=\"cycle\" />\n </td>\n @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n @for (node of data()[dataKey]; track node.value.term.name; let nodeLast = $last) {\n <td\n class=\"is-nowrap\"\n [class.has-border-right]=\"dataKeys().length > 1 && !dataKeyLast && nodeLast\">\n @if (node.value.values[cycle['@id']]) {\n <span\n class=\"trigger-popover\"\n [ngbPopover]=\"details\"\n autoClose=\"outside\"\n popoverClass=\"is-narrow is-overflow-visible\"\n placement=\"left\"\n container=\"body\"\n [popoverContext]=\"{ data: node.value.values[cycle['@id']], cycle, key: dataKey }\">\n <span pointer>\n {{\n propertyValue(node.value.values[cycle['@id']].value, node.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]=\"node.value.values[cycle['@id']].node\"\n key=\"value\" />\n </span>\n } @else {\n <span>-</span>\n }\n </td>\n }\n }\n </tr>\n }\n </tbody>\n </table>\n </he-data-table>\n <he-blank-node-state-notice\n [dataState]=\"dataState()\"\n [showDeleted]=\"firstNodeKey() === BlankNodesKey.emissions\" />\n } @else {\n <div class=\"is-pt-3 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 @if (showSwitchToRecalculated()) {\n <span>\n Switch to\n <code>recalculated</code>\n version.\n </span>\n }\n </div>\n }\n </div>\n }\n @case (View.chart) {\n @switch (firstNodeKey()) {\n @case (BlankNodesKey.inputs) {\n <he-cycles-result [cycles]=\"cycles()\"></he-cycles-result>\n }\n @case (BlankNodesKey.emissions) {\n <he-cycles-emissions-chart [cycles]=\"cycles()\" />\n }\n }\n }\n @case (View.timeline) {\n <ng-container *ngTemplateOutlet=\"selectCycle\" />\n <div class=\"is-px-3 is-pb-3\">\n <he-cycles-nodes-timeline [recalculatedValues]=\"timelineValues()\" [dataState]=\"dataState()\" />\n <he-blank-node-state-notice [dataState]=\"dataState()\" />\n </div>\n }\n @case (View.logs) {\n <ng-container *ngTemplateOutlet=\"selectCycle\" />\n @if (nodeKeys().length > 1) {\n <div class=\"tabs is-mb-2\">\n <ul>\n @for (nodeKey of nodeKeys(); track nodeKey) {\n <li [class.is-active]=\"selectedNodeKey() === nodeKey\">\n <a (click)=\"selectedNodeKey.set(nodeKey)\">{{ nodeKey | keyToLabel }}</a>\n </li>\n }\n </ul>\n </div>\n }\n @if (selectedNode()) {\n <he-node-logs-models\n [node]=\"selectedNode()\"\n [nodeKey]=\"selectedNodeKey()\"\n [logsKey]=\"selectedLogsKey()\"\n [originalValues]=\"selectedOriginalValues()\"\n [recalculatedValues]=\"selectedRecalculatedValues()\"\n [filterTermTypes]=\"filterTermTypes()\" />\n }\n }\n }\n}\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 class=\"is-overflow-visible\"\n [data]=\"data\"\n [dataState]=\"dataState()\"\n [nodeType]=\"node['@type']\"\n [dataKey]=\"key\" />\n</ng-template>\n\n<ng-template #selectCycle>\n @if (cycles().length > 1) {\n <div class=\"field has-addons is-py-2 is-px-3\">\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 @for (value of cycles(); track value; let cycleIndex = $index) {\n <option [value]=\"cycleIndex\">{{ cycleIndex + 1 }}. {{ defaultLabel(value) }}</option>\n }\n </select>\n </div>\n </div>\n </div>\n }\n</ng-template>\n", styles: ["fa-icon{display:inline-block;width:10px}he-data-table ::ng-deep .table thead tr th:nth-child(2),he-data-table ::ng-deep .table tbody tr td:nth-child(2){max-width:102px;width:102px}\n"], dependencies: [{ kind: "component", type: FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "mask", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "transform", "a11yRole"] }, { kind: "component", type: SearchExtendComponent, selector: "he-search-extend", inputs: ["value", "disabled", "placeholder", "class"], outputs: ["valueChange", "search"] }, { kind: "component", type: DataTableComponent, selector: "he-data-table", inputs: ["minHeight", "maxHeight", "small"] }, { kind: "component", type: NodeLinkComponent, selector: "he-node-link", inputs: ["node", "showExternalLink", "linkClass"] }, { kind: "component", type: TermsUnitsDescriptionComponent, selector: "he-terms-units-description", inputs: ["term", "iconTemplate"] }, { kind: "component", type: CyclesFunctionalUnitMeasureComponent, selector: "he-cycles-functional-unit-measure", inputs: ["cycle"] }, { kind: "directive", type: NgbPopover, selector: "[ngbPopover]", inputs: ["animation", "autoClose", "ngbPopover", "popoverTitle", "placement", "popperOptions", "triggers", "positionTarget", "container", "disablePopover", "popoverClass", "popoverContext", "openDelay", "closeDelay"], outputs: ["shown", "hidden"], exportAs: ["ngbPopover"] }, { kind: "component", type: BlankNodeStateComponent, selector: "he-blank-node-state", inputs: ["dataState", "nodeType", "dataKey", "key", "node", "state", "linkClass"] }, { kind: "component", type: BlankNodeStateNoticeComponent, selector: "he-blank-node-state-notice", inputs: ["dataState", "showDeleted"] }, { kind: "component", type: CyclesResultComponent, selector: "he-cycles-result", inputs: ["cycles"] }, { kind: "component", type: CyclesEmissionsChartComponent, selector: "he-cycles-emissions-chart", inputs: ["cycles"] }, { kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: CyclesNodesTimelineComponent, selector: "he-cycles-nodes-timeline", inputs: ["recalculatedValues", "dataState"] }, { kind: "component", type: NodeLogsModelsComponent, selector: "he-node-logs-models", inputs: ["node", "nodeKey", "originalValues", "recalculatedValues", "terms", "filterTermTypes", "filterTermTypesLabel", "logsKey", "noDataMessage"] }, { kind: "component", type: NodeValueDetailsComponent, selector: "he-node-value-details", inputs: ["data", "nodeType", "dataState", "dataKey"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i1.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "pipe", type: CompoundPipe, name: "compound" }, { kind: "pipe", type: DefaultPipe, name: "default" }, { kind: "pipe", type: EllipsisPipe, name: "ellipsis" }, { kind: "pipe", type: KeyToLabelPipe, name: "keyToLabel" }, { kind: "pipe", type: PrecisionPipe, name: "precision" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
205
205
|
}
|
|
206
206
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.11", ngImport: i0, type: CyclesNodesComponent, decorators: [{
|
|
207
207
|
type: Component,
|
|
@@ -228,6 +228,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.11", ngImpo
|
|
|
228
228
|
EllipsisPipe,
|
|
229
229
|
KeyToLabelPipe,
|
|
230
230
|
PrecisionPipe
|
|
231
|
-
], template: "@if (isGroupNode()) {\n <div class=\"tabs is-mb-1\">\n <ul>\n @for (term of groupNodeTerms(); track term.name) {\n <li [class.is-active]=\"selectedGroupTerm() === term\">\n <a (click)=\"selectedGroupTerm.set(term)\">{{ term.name }}</a>\n </li>\n }\n </ul>\n </div>\n}\n\n<div class=\"columns is-variable is-align-items-center is-2 is-m-0\">\n <div class=\"column\">\n @if (hasData() && selectedView() === View.table) {\n <button class=\"button is-small is-ghost\" (click)=\"showDownload()\">\n <fa-icon [icon]=\"faDownload\" size=\"lg\"></fa-icon>\n </button>\n }\n </div>\n @if (selectedView() === View.table) {\n <div class=\"column is-narrow\">\n <he-search-extend\n class=\"is-secondary is-small\"\n placeholder=\"Filter terms by name\"\n (search)=\"filterTerm.set($event)\"></he-search-extend>\n </div>\n }\n @if (hasData() && views()?.length > 1) {\n <div class=\"column is-narrow\">\n <div class=\"field has-addons\">\n @for (view of views(); track view) {\n <div class=\"control\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView() === view\"\n (click)=\"selectedView.set(view)\">\n <span class=\"icon is-small\">\n <fa-icon [icon]=\"viewIcon[view]\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span class=\"is-hidden-mobile\">{{ view }}</span>\n </button>\n </div>\n }\n </div>\n </div>\n }\n</div>\n\n@if (isNodeKeyAllowed()) {\n @switch (selectedView()) {\n @case (View.table) {\n <div class=\"is-px-3 is-pb-3\">\n @if (hasData()) {\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 @if (dataKeys().length > 1) {\n <tr class=\"has-text-weight-bold\">\n <th class=\"width-auto\"></th>\n <th [class.is-hidden]=\"isGroupNode()\"></th>\n @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n @if (data()[dataKey]?.length) {\n <th [attr.colspan]=\"data()[dataKey].length\" [class.has-border-right]=\"!dataKeyLast\">\n <span>{{ dataKey | keyToLabel }}</span>\n </th>\n }\n }\n </tr>\n }\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]=\"isGroupNode()\"></th>\n @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n @for (node of data()[dataKey]; track node.value.term.name; let nodeLast = $last) {\n <th\n [attr.title]=\"node.value.term.name\"\n [class.has-border-right]=\"dataKeys().length > 1 && !dataKeyLast && nodeLast\">\n <he-node-link [node]=\"node.value.term\">\n <span\n [innerHtml]=\"\n node.value.term.name | ellipsis: 30 | compound: node.value.term.termType\n \"></span>\n </he-node-link>\n </th>\n }\n }\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]=\"isGroupNode()\">\n <a [href]=\"schemaBaseUrl + '/Cycle#functionalUnit'\" target=\"_blank\">Functional unit</a>\n </th>\n @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n @for (node of data()[dataKey]; track node.value.term.name; let nodeLast = $last) {\n <th\n [attr.title]=\"node.value.term.units\"\n [class.has-border-right]=\"dataKeys().length > 1 && !dataKeyLast && nodeLast\">\n <span [innerHtml]=\"node.value.term.units | compound: node.value.term.termType\"></span>\n <he-terms-units-description\n class=\"is-inline-block is-ml-2\"\n [term]=\"node.value.term\"></he-terms-units-description>\n </th>\n }\n }\n </tr>\n </thead>\n <tbody>\n @for (cycle of cycles(); track trackById(cycleIndex, cycle); let cycleIndex = $index) {\n <tr>\n <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n <he-node-link [node]=\"cycleNode(cycle)\">\n <span>{{ cycleIndex + 1 }}. {{ defaultLabel(cycle) }}</span>\n </he-node-link>\n </td>\n <td class=\"has-border-right\" [class.is-hidden]=\"isGroupNode()\">\n <he-cycles-functional-unit-measure [cycle]=\"cycle\"></he-cycles-functional-unit-measure>\n </td>\n @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n @for (node of data()[dataKey]; track node.value.term.name; let nodeLast = $last) {\n <td\n class=\"is-nowrap\"\n [class.has-border-right]=\"dataKeys().length > 1 && !dataKeyLast && nodeLast\">\n @if (node.value.values[cycle['@id']]) {\n <span\n class=\"trigger-popover\"\n [ngbPopover]=\"details\"\n autoClose=\"outside\"\n popoverClass=\"is-narrow\"\n placement=\"left\"\n container=\"body\"\n [popoverContext]=\"{ data: node.value.values[cycle['@id']], cycle, key: dataKey }\">\n <span pointer>\n {{\n propertyValue(node.value.values[cycle['@id']].value, node.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]=\"node.value.values[cycle['@id']].node\"\n key=\"value\"></he-blank-node-state>\n </span>\n } @else {\n <span>-</span>\n }\n </td>\n }\n }\n </tr>\n }\n </tbody>\n </table>\n </he-data-table>\n <he-blank-node-state-notice\n [dataState]=\"dataState()\"\n [showDeleted]=\"firstNodeKey() === BlankNodesKey.emissions\"></he-blank-node-state-notice>\n } @else {\n <div class=\"is-pt-3 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 @if (showSwitchToRecalculated()) {\n <span>\n Switch to\n <code>recalculated</code>\n version.\n </span>\n }\n </div>\n }\n </div>\n }\n @case (View.chart) {\n @switch (firstNodeKey()) {\n @case (BlankNodesKey.inputs) {\n <he-cycles-result [cycles]=\"cycles()\"></he-cycles-result>\n }\n @case (BlankNodesKey.emissions) {\n <he-cycles-emissions-chart [cycles]=\"cycles()\"></he-cycles-emissions-chart>\n }\n }\n }\n @case (View.timeline) {\n <ng-container *ngTemplateOutlet=\"selectCycle\"></ng-container>\n <div class=\"is-px-3 is-pb-3\">\n <he-cycles-nodes-timeline\n [recalculatedValues]=\"timelineValues()\"\n [dataState]=\"dataState()\"></he-cycles-nodes-timeline>\n <he-blank-node-state-notice [dataState]=\"dataState()\"></he-blank-node-state-notice>\n </div>\n }\n @case (View.logs) {\n <ng-container *ngTemplateOutlet=\"selectCycle\"></ng-container>\n @if (nodeKeys().length > 1) {\n <div class=\"tabs is-mb-2\">\n <ul>\n @for (nodeKey of nodeKeys(); track nodeKey) {\n <li [class.is-active]=\"selectedNodeKey() === nodeKey\">\n <a (click)=\"selectedNodeKey.set(nodeKey)\">{{ nodeKey | keyToLabel }}</a>\n </li>\n }\n </ul>\n </div>\n }\n @if (selectedNode()) {\n <he-node-logs-models\n [node]=\"selectedNode()\"\n [nodeKey]=\"selectedNodeKey()\"\n [logsKey]=\"selectedLogsKey()\"\n [originalValues]=\"selectedOriginalValues()\"\n [recalculatedValues]=\"selectedRecalculatedValues()\"\n [filterTermTypes]=\"filterTermTypes()\"></he-node-logs-models>\n }\n }\n }\n}\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\n<ng-template #selectCycle>\n @if (cycles().length > 1) {\n <div class=\"field has-addons is-py-2 is-px-3\">\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 @for (value of cycles(); track value; let cycleIndex = $index) {\n <option [value]=\"cycleIndex\">{{ cycleIndex + 1 }}. {{ defaultLabel(value) }}</option>\n }\n </select>\n </div>\n </div>\n </div>\n }\n</ng-template>\n", styles: ["fa-icon{display:inline-block;width:10px}he-data-table ::ng-deep .table thead tr th:nth-child(2),he-data-table ::ng-deep .table tbody tr td:nth-child(2){max-width:102px;width:102px}\n"] }]
|
|
231
|
+
], template: "@if (isGroupNode()) {\n <div class=\"tabs is-mb-1\">\n <ul>\n @for (term of groupNodeTerms(); track term.name) {\n <li [class.is-active]=\"selectedGroupTerm() === term\">\n <a (click)=\"selectedGroupTerm.set(term)\">{{ term.name }}</a>\n </li>\n }\n </ul>\n </div>\n}\n\n<div class=\"columns is-variable is-align-items-center is-2 is-m-0\">\n <div class=\"column\">\n @if (hasData() && selectedView() === View.table) {\n <button class=\"button is-small is-ghost\" (click)=\"showDownload()\">\n <fa-icon [icon]=\"faDownload\" size=\"lg\" />\n </button>\n }\n </div>\n @if (selectedView() === View.table) {\n <div class=\"column is-narrow\">\n <he-search-extend\n class=\"is-secondary is-small\"\n placeholder=\"Filter terms by name\"\n (search)=\"filterTerm.set($event)\" />\n </div>\n }\n @if (views()?.length > 1) {\n <div class=\"column is-narrow\">\n <div class=\"field has-addons\">\n @for (view of views(); track view) {\n <div class=\"control\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView() === view\"\n (click)=\"selectedView.set(view)\">\n <span class=\"icon is-small\">\n <fa-icon [icon]=\"viewIcon[view]\" aria-hidden=\"true\" />\n </span>\n <span class=\"is-hidden-mobile\">{{ view }}</span>\n </button>\n </div>\n }\n </div>\n </div>\n }\n</div>\n\n@if (isNodeKeyAllowed()) {\n @switch (selectedView()) {\n @case (View.table) {\n <div class=\"is-px-3 is-pb-3\">\n @if (hasData()) {\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 @if (dataKeys().length > 1) {\n <tr class=\"has-text-weight-bold\">\n <th class=\"width-auto\"></th>\n <th [class.is-hidden]=\"isGroupNode()\"></th>\n @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n @if (data()[dataKey]?.length) {\n <th [attr.colspan]=\"data()[dataKey].length\" [class.has-border-right]=\"!dataKeyLast\">\n <span>{{ dataKey | keyToLabel }}</span>\n </th>\n }\n }\n </tr>\n }\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]=\"isGroupNode()\"></th>\n @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n @for (node of data()[dataKey]; track node.value.term.name; let nodeLast = $last) {\n <th\n [attr.title]=\"node.value.term.name\"\n [class.has-border-right]=\"dataKeys().length > 1 && !dataKeyLast && nodeLast\">\n <he-node-link [node]=\"node.value.term\">\n <span\n [innerHtml]=\"\n node.value.term.name | ellipsis: 30 | compound: node.value.term.termType\n \"></span>\n </he-node-link>\n </th>\n }\n }\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]=\"isGroupNode()\">\n <a [href]=\"schemaBaseUrl + '/Cycle#functionalUnit'\" target=\"_blank\">Functional unit</a>\n </th>\n @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n @for (node of data()[dataKey]; track node.value.term.name; let nodeLast = $last) {\n <th\n [attr.title]=\"node.value.term.units\"\n [class.has-border-right]=\"dataKeys().length > 1 && !dataKeyLast && nodeLast\">\n <span [innerHtml]=\"node.value.term.units | compound: node.value.term.termType\"></span>\n <he-terms-units-description class=\"is-inline-block is-ml-2\" [term]=\"node.value.term\" />\n </th>\n }\n }\n </tr>\n </thead>\n <tbody>\n @for (cycle of cycles(); track trackById(cycleIndex, cycle); let cycleIndex = $index) {\n <tr>\n <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n <he-node-link [node]=\"cycleNode(cycle)\">\n <span>{{ cycleIndex + 1 }}. {{ defaultLabel(cycle) }}</span>\n </he-node-link>\n </td>\n <td class=\"has-border-right\" [class.is-hidden]=\"isGroupNode()\">\n <he-cycles-functional-unit-measure [cycle]=\"cycle\" />\n </td>\n @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n @for (node of data()[dataKey]; track node.value.term.name; let nodeLast = $last) {\n <td\n class=\"is-nowrap\"\n [class.has-border-right]=\"dataKeys().length > 1 && !dataKeyLast && nodeLast\">\n @if (node.value.values[cycle['@id']]) {\n <span\n class=\"trigger-popover\"\n [ngbPopover]=\"details\"\n autoClose=\"outside\"\n popoverClass=\"is-narrow is-overflow-visible\"\n placement=\"left\"\n container=\"body\"\n [popoverContext]=\"{ data: node.value.values[cycle['@id']], cycle, key: dataKey }\">\n <span pointer>\n {{\n propertyValue(node.value.values[cycle['@id']].value, node.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]=\"node.value.values[cycle['@id']].node\"\n key=\"value\" />\n </span>\n } @else {\n <span>-</span>\n }\n </td>\n }\n }\n </tr>\n }\n </tbody>\n </table>\n </he-data-table>\n <he-blank-node-state-notice\n [dataState]=\"dataState()\"\n [showDeleted]=\"firstNodeKey() === BlankNodesKey.emissions\" />\n } @else {\n <div class=\"is-pt-3 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 @if (showSwitchToRecalculated()) {\n <span>\n Switch to\n <code>recalculated</code>\n version.\n </span>\n }\n </div>\n }\n </div>\n }\n @case (View.chart) {\n @switch (firstNodeKey()) {\n @case (BlankNodesKey.inputs) {\n <he-cycles-result [cycles]=\"cycles()\"></he-cycles-result>\n }\n @case (BlankNodesKey.emissions) {\n <he-cycles-emissions-chart [cycles]=\"cycles()\" />\n }\n }\n }\n @case (View.timeline) {\n <ng-container *ngTemplateOutlet=\"selectCycle\" />\n <div class=\"is-px-3 is-pb-3\">\n <he-cycles-nodes-timeline [recalculatedValues]=\"timelineValues()\" [dataState]=\"dataState()\" />\n <he-blank-node-state-notice [dataState]=\"dataState()\" />\n </div>\n }\n @case (View.logs) {\n <ng-container *ngTemplateOutlet=\"selectCycle\" />\n @if (nodeKeys().length > 1) {\n <div class=\"tabs is-mb-2\">\n <ul>\n @for (nodeKey of nodeKeys(); track nodeKey) {\n <li [class.is-active]=\"selectedNodeKey() === nodeKey\">\n <a (click)=\"selectedNodeKey.set(nodeKey)\">{{ nodeKey | keyToLabel }}</a>\n </li>\n }\n </ul>\n </div>\n }\n @if (selectedNode()) {\n <he-node-logs-models\n [node]=\"selectedNode()\"\n [nodeKey]=\"selectedNodeKey()\"\n [logsKey]=\"selectedLogsKey()\"\n [originalValues]=\"selectedOriginalValues()\"\n [recalculatedValues]=\"selectedRecalculatedValues()\"\n [filterTermTypes]=\"filterTermTypes()\" />\n }\n }\n }\n}\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 class=\"is-overflow-visible\"\n [data]=\"data\"\n [dataState]=\"dataState()\"\n [nodeType]=\"node['@type']\"\n [dataKey]=\"key\" />\n</ng-template>\n\n<ng-template #selectCycle>\n @if (cycles().length > 1) {\n <div class=\"field has-addons is-py-2 is-px-3\">\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 @for (value of cycles(); track value; let cycleIndex = $index) {\n <option [value]=\"cycleIndex\">{{ cycleIndex + 1 }}. {{ defaultLabel(value) }}</option>\n }\n </select>\n </div>\n </div>\n </div>\n }\n</ng-template>\n", styles: ["fa-icon{display:inline-block;width:10px}he-data-table ::ng-deep .table thead tr th:nth-child(2),he-data-table ::ng-deep .table tbody tr td:nth-child(2){max-width:102px;width:102px}\n"] }]
|
|
232
232
|
}], ctorParameters: () => [] });
|
|
233
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cycles-nodes.component.js","sourceRoot":"","sources":["../../../../src/cycles/cycles-nodes/cycles-nodes.component.ts","../../../../src/cycles/cycles-nodes/cycles-nodes.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, computed, effect, inject, input, signal } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { FormsModule } from '@angular/forms';\nimport { DataState } from '@hestia-earth/api';\nimport {\n  ICycleJSONLD,\n  Input as HestiaInput,\n  Product,\n  NodeType,\n  Animal,\n  Emission,\n  Practice,\n  TermTermType,\n  EmissionMethodTier,\n  Term,\n  Transformation,\n  BlankNodesKey,\n  blankNodesType\n} from '@hestia-earth/schema';\nimport orderBy from 'lodash.orderby';\nimport { propertyValue } from '@hestia-earth/utils/dist/term';\nimport { unique } from '@hestia-earth/utils';\nimport { FaIconComponent } from '@fortawesome/angular-fontawesome';\nimport { NgbTooltip, NgbPopover, NgbModal } from '@ng-bootstrap/ng-bootstrap';\nimport { IconDefinition } from '@fortawesome/fontawesome-svg-core';\n\nimport { filterBlankNode, methodTierOrder } from '../../common/node-utils';\nimport { logsKey } from '../cycles.model';\nimport { HeNodeStoreService } from '../../node/node-store.service';\nimport { groupNodesByTerm, IGroupedKeys, grouppedKeys } from '../../common/node-utils';\nimport { schemaBaseUrl, defaultLabel } from '../../common/utils';\nimport { PrecisionPipe } from '../../common/precision.pipe';\nimport { KeyToLabelPipe } from '../../common/key-to-label.pipe';\nimport { EllipsisPipe } from '../../common/ellipsis.pipe';\nimport { DefaultPipe } from '../../common/default.pipe';\nimport { CompoundPipe } from '../../common/compound.pipe';\nimport { NodeValueDetailsComponent } from '../../node/node-value-details/node-value-details.component';\nimport { NodeCsvExportConfirmComponent } from '../../node/node-csv-export-confirm/node-csv-export-confirm.component';\nimport { NodeLogsModelsComponent } from '../../node/node-logs-models/node-logs-models.component';\nimport { CyclesNodesTimelineComponent } from '../cycles-nodes-timeline/cycles-nodes-timeline.component';\nimport { NgTemplateOutlet } from '@angular/common';\nimport { CyclesEmissionsChartComponent } from '../cycles-emissions-chart/cycles-emissions-chart.component';\nimport { CyclesResultComponent } from '../cycles-result/cycles-result.component';\nimport { BlankNodeStateNoticeComponent } from '../../common/blank-node-state-notice/blank-node-state-notice.component';\nimport { BlankNodeStateComponent } from '../../common/blank-node-state/blank-node-state.component';\nimport { CyclesFunctionalUnitMeasureComponent } from '../cycles-functional-unit-measure/cycles-functional-unit-measure.component';\nimport { TermsUnitsDescriptionComponent } from '../../terms/terms-units-description/terms-units-description.component';\nimport { NodeLinkComponent } from '../../node/node-link/node-link.component';\nimport { DataTableComponent } from '../../common/data-table/data-table.component';\nimport { SearchExtendComponent } from '../../common/search-extend/search-extend.component';\nimport { faCalculator, faChartBar, faDownload, faList, faListAlt } from '@fortawesome/free-solid-svg-icons';\n\nenum View {\n  table = 'Table view',\n  chart = 'Chart view',\n  timeline = 'Operations Timeline',\n  logs = 'Recalculations logs'\n}\n\nconst timelineTermType = [TermTermType.operation];\n\ninterface ICycleConfig {\n  nodeKey: BlankNodesKey;\n}\n\ninterface ICycleConfigAnimal extends ICycleConfig {\n  nodeKey: BlankNodesKey.animals;\n}\n\ninterface ICycleConfigEmission extends ICycleConfig {\n  nodeKey: BlankNodesKey.emissions;\n}\n\ninterface ICycleConfigInput extends ICycleConfig {\n  nodeKey: BlankNodesKey.inputs;\n}\n\ninterface ICycleConfigPractice extends ICycleConfig {\n  nodeKey: BlankNodesKey.practices;\n}\n\nconst viewIcon: {\n  [view in View]: IconDefinition;\n} = {\n  [View.chart]: faChartBar,\n  [View.logs]: faCalculator,\n  [View.table]: faList,\n  [View.timeline]: faListAlt\n};\n\nconst nodeKeyViews: {\n  [type in BlankNodesKey]?: View[];\n} = {\n  [BlankNodesKey.animals]: [View.table, View.logs],\n  [BlankNodesKey.emissions]: [View.table, View.chart, View.logs],\n  [BlankNodesKey.inputs]: [View.table, View.chart, View.logs],\n  [BlankNodesKey.products]: [View.table, View.chart, View.logs],\n  [BlankNodesKey.practices]: [View.table, View.timeline, View.logs]\n};\n\nconst nodeKeyFilterTermTypes: {\n  [type in BlankNodesKey]?: TermTermType[];\n} = {\n  [BlankNodesKey.emissions]: [TermTermType.emission]\n};\n\nconst filterValuesTimeline = (values: blankNodesType[]) =>\n  values.filter(p => timelineTermType.includes(p.term?.termType));\n\ntype groupedEmissions = {\n  [methodTier in EmissionMethodTier]: IGroupedKeys<Emission>[];\n};\n\nexport enum CycleNodesKeyGroup {\n  animals = 'animals',\n  transformations = 'transformations'\n}\n\ntype groupedNode = Animal | Transformation;\n\nconst cycleGroupNode =\n  (cycle?: ICycleJSONLD) =>\n  (node?: groupedNode): groupedNode & { name: string; '@id': string } =>\n    node\n      ? {\n          ...node,\n          name: cycle?.name,\n          '@id': cycle?.['@id'] // set @id so we can fetch the logs from the Cycle\n        }\n      : null;\n\nconst cycleGroupNodes = (groupKey: CycleNodesKeyGroup, cycle?: ICycleJSONLD): groupedNode[] =>\n  (cycle?.[groupKey] || []).map(cycleGroupNode(cycle)).filter(Boolean);\n\nconst filterGroupNodesByTerm = (groupKey: CycleNodesKeyGroup, cycles: ICycleJSONLD[], term?: Term): groupedNode[] =>\n  cycles\n    .map(cycle =>\n      cycleGroupNode(cycle)(\n        ((cycle?.[groupKey] as groupedNode[]) || []).filter(Boolean).find(node => node.term?.name === term?.name)\n      )\n    )\n    .filter(Boolean);\n\n@Component({\n  selector: 'he-cycles-nodes',\n  templateUrl: './cycles-nodes.component.html',\n  styleUrls: ['./cycles-nodes.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [\n    NgbTooltip,\n    FaIconComponent,\n    SearchExtendComponent,\n    DataTableComponent,\n    NodeLinkComponent,\n    TermsUnitsDescriptionComponent,\n    CyclesFunctionalUnitMeasureComponent,\n    NgbPopover,\n    BlankNodeStateComponent,\n    BlankNodeStateNoticeComponent,\n    CyclesResultComponent,\n    CyclesEmissionsChartComponent,\n    NgTemplateOutlet,\n    CyclesNodesTimelineComponent,\n    NodeLogsModelsComponent,\n    NodeValueDetailsComponent,\n    FormsModule,\n    CompoundPipe,\n    DefaultPipe,\n    EllipsisPipe,\n    KeyToLabelPipe,\n    PrecisionPipe\n  ]\n})\nexport class CyclesNodesComponent {\n  private readonly modalService = inject(NgbModal);\n  private nodeStoreService = inject(HeNodeStoreService);\n\n  protected readonly faDownload = faDownload;\n\n  protected readonly dataState = input(DataState.original);\n  protected readonly nodeKeys = input.required<BlankNodesKey[]>();\n  protected readonly nodeKeyGroup = input<CycleNodesKeyGroup>();\n\n  protected readonly BlankNodesKey = BlankNodesKey;\n  protected readonly firstNodeKey = computed(() => this.nodeKeys()?.[0]);\n  protected readonly selectedNodeKey = signal(undefined as BlankNodesKey);\n\n  protected readonly selectedGroupTerm = signal(undefined as Term);\n\n  protected readonly schemaBaseUrl = schemaBaseUrl();\n  protected readonly propertyValue = propertyValue;\n  protected readonly defaultLabel = defaultLabel;\n\n  protected readonly headerKeys = computed(() => [\n    'cycle.id',\n    'cycle.@id',\n    'transformation.id',\n    'transformation.@id',\n    ...this.nodeKeys().flatMap(key => [`cycle.${key}.`, `transformation.${key}.`])\n  ]);\n\n  protected readonly View = View;\n  protected readonly viewIcon = viewIcon;\n  private readonly showView = computed(\n    () =>\n      ({\n        [View.chart]: this.selectedNodeKey() === BlankNodesKey.emissions ? this.cycles().length > 1 : this.hasData(),\n        [View.logs]: !this.isOriginal(),\n        [View.table]: true,\n        [View.timeline]: this.enableTimeline()\n      }) as { [view in View]: boolean }\n  );\n  protected readonly views = computed(\n    () => nodeKeyViews[this.selectedNodeKey()]?.filter(view => this.showView()[view]) ?? []\n  );\n  protected readonly selectedView = signal(View.table);\n\n  private readonly _allNodes = toSignal(this.nodeStoreService.find$<ICycleJSONLD>(NodeType.Cycle));\n  private readonly originalNodes = computed(() => this._allNodes()?.map(data => data[DataState.original]) || []);\n  private readonly currentNodes = computed(() => this._allNodes()?.map(data => data[this.dataState()]) || []);\n\n  // can be Cycle or Transformation\n  protected readonly originalCycles = computed(() =>\n    this.nodeKeyGroup()\n      ? filterGroupNodesByTerm(this.nodeKeyGroup(), this.originalNodes(), this.selectedGroupTerm())\n      : this.originalNodes()\n  );\n  protected readonly cycles = computed(() =>\n    this.nodeKeyGroup()\n      ? filterGroupNodesByTerm(this.nodeKeyGroup(), this.currentNodes(), this.selectedGroupTerm())\n      : this.currentNodes()\n  );\n\n  private readonly selectedIndex = signal(0);\n  private readonly ogirinalSelectedCycle = computed(() => this.originalCycles()?.[this.selectedIndex()]);\n  private readonly selectedCycle = computed(() => this.cycles()?.[this.selectedIndex()]);\n  protected readonly selectedLogsKey = computed(() => logsKey(this.selectedCycle()));\n  protected readonly selectedOriginalValues = computed(\n    () => this.ogirinalSelectedCycle()?.[this.selectedNodeKey()] || []\n  );\n  protected readonly selectedRecalculatedValues = computed(() => this.selectedCycle()?.[this.selectedNodeKey()] || []);\n  // TODO: handle transformation and non-indexed nodes on CE\n  protected readonly selectedNode = computed(() =>\n    this.selectedCycle()\n      ? {\n          ...this.selectedCycle(),\n          '@type': NodeType.Cycle,\n          type: NodeType.Cycle,\n          dataState: DataState.recalculated\n        }\n      : null\n  );\n\n  private readonly isOriginal = computed(() => this.dataState() === DataState.original);\n  private readonly originalValues = computed(() => (this.isOriginal() ? null : this.originalCycles()) || []);\n\n  private readonly hasRecalculatedNodes = computed(() => this.cycles().some(cycle => !cycle.aggregated));\n  protected readonly showSwitchToRecalculated = computed(() => this.isOriginal() && this.hasRecalculatedNodes());\n\n  protected readonly timelineValues = computed(() =>\n    filterValuesTimeline(this.selectedCycle()?.[this.selectedNodeKey()] || [])\n  );\n  private readonly enableTimeline = computed(() => this.timelineValues().length > 0);\n\n  protected readonly isNodeKeyAllowed = computed(() => nodeKeyViews[this.firstNodeKey()].includes(this.selectedView()));\n\n  private readonly groupedNodes = computed(() =>\n    this.originalNodes().flatMap(cycle => cycleGroupNodes(this.nodeKeyGroup(), cycle))\n  );\n  protected readonly groupNodeTerms = computed(() => unique(this.groupedNodes().map(({ term }) => term)));\n  protected readonly isGroupNode = computed(() => this.nodeKeyGroup() && this.groupedNodes()?.length > 0);\n\n  protected readonly isEmission = computed(() => this.selectedNodeKey() === BlankNodesKey.emissions);\n\n  protected readonly data = computed(() =>\n    this.isEmission()\n      ? this.groupEmissions()\n      : Object.fromEntries(this.nodeKeys().map(nodeKey => [nodeKey, this.groupNodesByKey({ nodeKey })]))\n  );\n  protected readonly dataKeys = computed(() => Object.keys(this.data()));\n\n  protected readonly hasData = computed(() => Object.values(this.data()).flat().length > 0);\n\n  protected readonly filterTermTypes = computed(() => nodeKeyFilterTermTypes[this.selectedNodeKey()]);\n  protected readonly filterTerm = signal('');\n\n  protected readonly csvFilename = computed(() => `${this.nodeKeys().join('-')}.csv`);\n\n  constructor() {\n    effect(\n      () => {\n        // make sure logs does not remain displayed\n        if ((this.isOriginal() && this.selectedView() === View.logs) || !this.views().includes(this.selectedView())) {\n          this.selectedView.set(View.table);\n        }\n      },\n      { allowSignalWrites: true }\n    );\n\n    effect(\n      () => {\n        if (!this.selectedNodeKey() || !this.nodeKeys().includes(this.selectedNodeKey())) {\n          this.selectedNodeKey.set(this.firstNodeKey());\n          this.selectedView.set(View.table);\n        }\n      },\n      { allowSignalWrites: true }\n    );\n\n    effect(\n      () => {\n        if (!this.selectedGroupTerm() || !this.groupNodeTerms().includes(this.selectedGroupTerm())) {\n          this.selectedGroupTerm.set(this.groupNodeTerms()[0]);\n        }\n      },\n      { allowSignalWrites: true }\n    );\n  }\n\n  private groupNodesByKey<\n    T extends ICycleConfig,\n    R = T extends ICycleConfigAnimal\n      ? Animal\n      : T extends ICycleConfigEmission\n        ? Emission\n        : T extends ICycleConfigInput\n          ? HestiaInput\n          : T extends ICycleConfigPractice\n            ? Practice\n            : Product\n  >({ nodeKey }: T): IGroupedKeys<R>[] {\n    const nodesPerCycle = groupNodesByTerm<ICycleJSONLD | groupedNode, R>(\n      this.cycles(),\n      nodeKey,\n      this.originalValues(),\n      filterBlankNode(this.filterTerm())\n    );\n    return orderBy(grouppedKeys(nodesPerCycle), ['key'], ['asc']);\n  }\n\n  private groupEmissions() {\n    const emissionsPerCycle = groupNodesByTerm<ICycleJSONLD | groupedNode, Emission>(\n      this.cycles(),\n      BlankNodesKey.emissions,\n      this.originalValues(),\n      filterBlankNode(this.filterTerm())\n    );\n    const values: IGroupedKeys<Emission>[] = orderBy(grouppedKeys(emissionsPerCycle), ['key'], ['asc']);\n    const methodTiers = orderBy(\n      unique(values.map(({ value }) => value.methodTier)).map(methodTier => ({\n        methodTier,\n        methodTierOrder: methodTierOrder(methodTier)\n      })),\n      ['methodTierOrder'],\n      ['asc']\n    );\n    return Object.fromEntries(\n      methodTiers.map(({ methodTier }) => [methodTier, values.filter(({ value }) => value.methodTier === methodTier)])\n    ) as groupedEmissions;\n  }\n\n  protected trackById(_index: number, item: ICycleJSONLD | groupedNode) {\n    return item['@id'];\n  }\n\n  protected selectIndex({ target: { value } }) {\n    this.selectedIndex.set(+value);\n  }\n\n  protected cycleNode(cycle: ICycleJSONLD | groupedNode) {\n    return 'term' in cycle ? cycle.term : cycle;\n  }\n\n  protected showDownload() {\n    const instance = this.modalService.open(NodeCsvExportConfirmComponent);\n    const component = instance.componentInstance as NodeCsvExportConfirmComponent;\n    component.nodes.set(this.cycles());\n    component.filename.set(this.csvFilename());\n    component.isUpload.set(false);\n    component.headerKeys.set(this.headerKeys());\n  }\n}\n","@if (isGroupNode()) {\n  <div class=\"tabs is-mb-1\">\n    <ul>\n      @for (term of groupNodeTerms(); track term.name) {\n        <li [class.is-active]=\"selectedGroupTerm() === term\">\n          <a (click)=\"selectedGroupTerm.set(term)\">{{ term.name }}</a>\n        </li>\n      }\n    </ul>\n  </div>\n}\n\n<div class=\"columns is-variable is-align-items-center is-2 is-m-0\">\n  <div class=\"column\">\n    @if (hasData() && selectedView() === View.table) {\n      <button class=\"button is-small is-ghost\" (click)=\"showDownload()\">\n        <fa-icon [icon]=\"faDownload\" size=\"lg\"></fa-icon>\n      </button>\n    }\n  </div>\n  @if (selectedView() === View.table) {\n    <div class=\"column is-narrow\">\n      <he-search-extend\n        class=\"is-secondary is-small\"\n        placeholder=\"Filter terms by name\"\n        (search)=\"filterTerm.set($event)\"></he-search-extend>\n    </div>\n  }\n  @if (hasData() && views()?.length > 1) {\n    <div class=\"column is-narrow\">\n      <div class=\"field has-addons\">\n        @for (view of views(); track view) {\n          <div class=\"control\">\n            <button\n              class=\"button is-small\"\n              [class.is-selected]=\"selectedView() === view\"\n              (click)=\"selectedView.set(view)\">\n              <span class=\"icon is-small\">\n                <fa-icon [icon]=\"viewIcon[view]\" aria-hidden=\"true\"></fa-icon>\n              </span>\n              <span class=\"is-hidden-mobile\">{{ view }}</span>\n            </button>\n          </div>\n        }\n      </div>\n    </div>\n  }\n</div>\n\n@if (isNodeKeyAllowed()) {\n  @switch (selectedView()) {\n    @case (View.table) {\n      <div class=\"is-px-3 is-pb-3\">\n        @if (hasData()) {\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                @if (dataKeys().length > 1) {\n                  <tr class=\"has-text-weight-bold\">\n                    <th class=\"width-auto\"></th>\n                    <th [class.is-hidden]=\"isGroupNode()\"></th>\n                    @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n                      @if (data()[dataKey]?.length) {\n                        <th [attr.colspan]=\"data()[dataKey].length\" [class.has-border-right]=\"!dataKeyLast\">\n                          <span>{{ dataKey | keyToLabel }}</span>\n                        </th>\n                      }\n                    }\n                  </tr>\n                }\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]=\"isGroupNode()\"></th>\n                  @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n                    @for (node of data()[dataKey]; track node.value.term.name; let nodeLast = $last) {\n                      <th\n                        [attr.title]=\"node.value.term.name\"\n                        [class.has-border-right]=\"dataKeys().length > 1 && !dataKeyLast && nodeLast\">\n                        <he-node-link [node]=\"node.value.term\">\n                          <span\n                            [innerHtml]=\"\n                              node.value.term.name | ellipsis: 30 | compound: node.value.term.termType\n                            \"></span>\n                        </he-node-link>\n                      </th>\n                    }\n                  }\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]=\"isGroupNode()\">\n                    <a [href]=\"schemaBaseUrl + '/Cycle#functionalUnit'\" target=\"_blank\">Functional unit</a>\n                  </th>\n                  @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n                    @for (node of data()[dataKey]; track node.value.term.name; let nodeLast = $last) {\n                      <th\n                        [attr.title]=\"node.value.term.units\"\n                        [class.has-border-right]=\"dataKeys().length > 1 && !dataKeyLast && nodeLast\">\n                        <span [innerHtml]=\"node.value.term.units | compound: node.value.term.termType\"></span>\n                        <he-terms-units-description\n                          class=\"is-inline-block is-ml-2\"\n                          [term]=\"node.value.term\"></he-terms-units-description>\n                      </th>\n                    }\n                  }\n                </tr>\n              </thead>\n              <tbody>\n                @for (cycle of cycles(); track trackById(cycleIndex, cycle); let cycleIndex = $index) {\n                  <tr>\n                    <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n                      <he-node-link [node]=\"cycleNode(cycle)\">\n                        <span>{{ cycleIndex + 1 }}. {{ defaultLabel(cycle) }}</span>\n                      </he-node-link>\n                    </td>\n                    <td class=\"has-border-right\" [class.is-hidden]=\"isGroupNode()\">\n                      <he-cycles-functional-unit-measure [cycle]=\"cycle\"></he-cycles-functional-unit-measure>\n                    </td>\n                    @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n                      @for (node of data()[dataKey]; track node.value.term.name; let nodeLast = $last) {\n                        <td\n                          class=\"is-nowrap\"\n                          [class.has-border-right]=\"dataKeys().length > 1 && !dataKeyLast && nodeLast\">\n                          @if (node.value.values[cycle['@id']]) {\n                            <span\n                              class=\"trigger-popover\"\n                              [ngbPopover]=\"details\"\n                              autoClose=\"outside\"\n                              popoverClass=\"is-narrow\"\n                              placement=\"left\"\n                              container=\"body\"\n                              [popoverContext]=\"{ data: node.value.values[cycle['@id']], cycle, key: dataKey }\">\n                              <span pointer>\n                                {{\n                                  propertyValue(node.value.values[cycle['@id']].value, node.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]=\"node.value.values[cycle['@id']].node\"\n                                key=\"value\"></he-blank-node-state>\n                            </span>\n                          } @else {\n                            <span>-</span>\n                          }\n                        </td>\n                      }\n                    }\n                  </tr>\n                }\n              </tbody>\n            </table>\n          </he-data-table>\n          <he-blank-node-state-notice\n            [dataState]=\"dataState()\"\n            [showDeleted]=\"firstNodeKey() === BlankNodesKey.emissions\"></he-blank-node-state-notice>\n        } @else {\n          <div class=\"is-pt-3 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            @if (showSwitchToRecalculated()) {\n              <span>\n                Switch to\n                <code>recalculated</code>\n                version.\n              </span>\n            }\n          </div>\n        }\n      </div>\n    }\n    @case (View.chart) {\n      @switch (firstNodeKey()) {\n        @case (BlankNodesKey.inputs) {\n          <he-cycles-result [cycles]=\"cycles()\"></he-cycles-result>\n        }\n        @case (BlankNodesKey.emissions) {\n          <he-cycles-emissions-chart [cycles]=\"cycles()\"></he-cycles-emissions-chart>\n        }\n      }\n    }\n    @case (View.timeline) {\n      <ng-container *ngTemplateOutlet=\"selectCycle\"></ng-container>\n      <div class=\"is-px-3 is-pb-3\">\n        <he-cycles-nodes-timeline\n          [recalculatedValues]=\"timelineValues()\"\n          [dataState]=\"dataState()\"></he-cycles-nodes-timeline>\n        <he-blank-node-state-notice [dataState]=\"dataState()\"></he-blank-node-state-notice>\n      </div>\n    }\n    @case (View.logs) {\n      <ng-container *ngTemplateOutlet=\"selectCycle\"></ng-container>\n      @if (nodeKeys().length > 1) {\n        <div class=\"tabs is-mb-2\">\n          <ul>\n            @for (nodeKey of nodeKeys(); track nodeKey) {\n              <li [class.is-active]=\"selectedNodeKey() === nodeKey\">\n                <a (click)=\"selectedNodeKey.set(nodeKey)\">{{ nodeKey | keyToLabel }}</a>\n              </li>\n            }\n          </ul>\n        </div>\n      }\n      @if (selectedNode()) {\n        <he-node-logs-models\n          [node]=\"selectedNode()\"\n          [nodeKey]=\"selectedNodeKey()\"\n          [logsKey]=\"selectedLogsKey()\"\n          [originalValues]=\"selectedOriginalValues()\"\n          [recalculatedValues]=\"selectedRecalculatedValues()\"\n          [filterTermTypes]=\"filterTermTypes()\"></he-node-logs-models>\n      }\n    }\n  }\n}\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\n<ng-template #selectCycle>\n  @if (cycles().length > 1) {\n    <div class=\"field has-addons is-py-2 is-px-3\">\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            @for (value of cycles(); track value; let cycleIndex = $index) {\n              <option [value]=\"cycleIndex\">{{ cycleIndex + 1 }}. {{ defaultLabel(value) }}</option>\n            }\n          </select>\n        </div>\n      </div>\n    </div>\n  }\n</ng-template>\n"]}
|
|
233
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cycles-nodes.component.js","sourceRoot":"","sources":["../../../../src/cycles/cycles-nodes/cycles-nodes.component.ts","../../../../src/cycles/cycles-nodes/cycles-nodes.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, computed, effect, inject, input, signal } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { FormsModule } from '@angular/forms';\nimport { DataState } from '@hestia-earth/api';\nimport {\n  ICycleJSONLD,\n  Input as HestiaInput,\n  Product,\n  NodeType,\n  Animal,\n  Emission,\n  Practice,\n  TermTermType,\n  EmissionMethodTier,\n  Term,\n  Transformation,\n  BlankNodesKey,\n  blankNodesType\n} from '@hestia-earth/schema';\nimport orderBy from 'lodash.orderby';\nimport { propertyValue } from '@hestia-earth/utils/dist/term';\nimport { unique } from '@hestia-earth/utils';\nimport { FaIconComponent } from '@fortawesome/angular-fontawesome';\nimport { NgbTooltip, NgbPopover, NgbModal } from '@ng-bootstrap/ng-bootstrap';\nimport { IconDefinition } from '@fortawesome/fontawesome-svg-core';\n\nimport { filterBlankNode, methodTierOrder } from '../../common/node-utils';\nimport { logsKey } from '../cycles.model';\nimport { HeNodeStoreService } from '../../node/node-store.service';\nimport { groupNodesByTerm, IGroupedKeys, grouppedKeys } from '../../common/node-utils';\nimport { schemaBaseUrl, defaultLabel } from '../../common/utils';\nimport { PrecisionPipe } from '../../common/precision.pipe';\nimport { KeyToLabelPipe } from '../../common/key-to-label.pipe';\nimport { EllipsisPipe } from '../../common/ellipsis.pipe';\nimport { DefaultPipe } from '../../common/default.pipe';\nimport { CompoundPipe } from '../../common/compound.pipe';\nimport { NodeValueDetailsComponent } from '../../node/node-value-details/node-value-details.component';\nimport { NodeCsvExportConfirmComponent } from '../../node/node-csv-export-confirm/node-csv-export-confirm.component';\nimport { NodeLogsModelsComponent } from '../../node/node-logs-models/node-logs-models.component';\nimport { CyclesNodesTimelineComponent } from '../cycles-nodes-timeline/cycles-nodes-timeline.component';\nimport { NgTemplateOutlet } from '@angular/common';\nimport { CyclesEmissionsChartComponent } from '../cycles-emissions-chart/cycles-emissions-chart.component';\nimport { CyclesResultComponent } from '../cycles-result/cycles-result.component';\nimport { BlankNodeStateNoticeComponent } from '../../common/blank-node-state-notice/blank-node-state-notice.component';\nimport { BlankNodeStateComponent } from '../../common/blank-node-state/blank-node-state.component';\nimport { CyclesFunctionalUnitMeasureComponent } from '../cycles-functional-unit-measure/cycles-functional-unit-measure.component';\nimport { TermsUnitsDescriptionComponent } from '../../terms/terms-units-description/terms-units-description.component';\nimport { NodeLinkComponent } from '../../node/node-link/node-link.component';\nimport { DataTableComponent } from '../../common/data-table/data-table.component';\nimport { SearchExtendComponent } from '../../common/search-extend/search-extend.component';\nimport { faCalculator, faChartBar, faDownload, faList, faListAlt } from '@fortawesome/free-solid-svg-icons';\n\nenum View {\n  table = 'Table view',\n  chart = 'Chart view',\n  timeline = 'Operations Timeline',\n  logs = 'Recalculations logs'\n}\n\nconst timelineTermType = [TermTermType.operation];\n\ninterface ICycleConfig {\n  nodeKey: BlankNodesKey;\n}\n\ninterface ICycleConfigAnimal extends ICycleConfig {\n  nodeKey: BlankNodesKey.animals;\n}\n\ninterface ICycleConfigEmission extends ICycleConfig {\n  nodeKey: BlankNodesKey.emissions;\n}\n\ninterface ICycleConfigInput extends ICycleConfig {\n  nodeKey: BlankNodesKey.inputs;\n}\n\ninterface ICycleConfigPractice extends ICycleConfig {\n  nodeKey: BlankNodesKey.practices;\n}\n\nconst viewIcon: {\n  [view in View]: IconDefinition;\n} = {\n  [View.chart]: faChartBar,\n  [View.logs]: faCalculator,\n  [View.table]: faList,\n  [View.timeline]: faListAlt\n};\n\nconst nodeKeyViews: {\n  [type in BlankNodesKey]?: View[];\n} = {\n  [BlankNodesKey.animals]: [View.table, View.logs],\n  [BlankNodesKey.emissions]: [View.table, View.chart, View.logs],\n  [BlankNodesKey.inputs]: [View.table, View.chart, View.logs],\n  [BlankNodesKey.products]: [View.table, View.chart, View.logs],\n  [BlankNodesKey.practices]: [View.table, View.timeline, View.logs]\n};\n\nconst nodeKeyFilterTermTypes: {\n  [type in BlankNodesKey]?: TermTermType[];\n} = {\n  [BlankNodesKey.emissions]: [TermTermType.emission]\n};\n\nconst filterValuesTimeline = (values: blankNodesType[]) =>\n  values.filter(p => timelineTermType.includes(p.term?.termType));\n\ntype groupedEmissions = {\n  [methodTier in EmissionMethodTier]: IGroupedKeys<Emission>[];\n};\n\nexport enum CycleNodesKeyGroup {\n  animals = 'animals',\n  transformations = 'transformations'\n}\n\ntype groupedNode = Animal | Transformation;\n\nconst cycleGroupNode =\n  (cycle?: ICycleJSONLD) =>\n  (node?: groupedNode): groupedNode & { name: string; '@id': string } =>\n    node\n      ? {\n          ...node,\n          name: cycle?.name,\n          '@id': cycle?.['@id'] // set @id so we can fetch the logs from the Cycle\n        }\n      : null;\n\nconst cycleGroupNodes = (groupKey: CycleNodesKeyGroup, cycle?: ICycleJSONLD): groupedNode[] =>\n  (cycle?.[groupKey] || []).map(cycleGroupNode(cycle)).filter(Boolean);\n\nconst filterGroupNodesByTerm = (groupKey: CycleNodesKeyGroup, cycles: ICycleJSONLD[], term?: Term): groupedNode[] =>\n  cycles\n    .map(cycle =>\n      cycleGroupNode(cycle)(\n        ((cycle?.[groupKey] as groupedNode[]) || []).filter(Boolean).find(node => node.term?.name === term?.name)\n      )\n    )\n    .filter(Boolean);\n\n@Component({\n  selector: 'he-cycles-nodes',\n  templateUrl: './cycles-nodes.component.html',\n  styleUrls: ['./cycles-nodes.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [\n    NgbTooltip,\n    FaIconComponent,\n    SearchExtendComponent,\n    DataTableComponent,\n    NodeLinkComponent,\n    TermsUnitsDescriptionComponent,\n    CyclesFunctionalUnitMeasureComponent,\n    NgbPopover,\n    BlankNodeStateComponent,\n    BlankNodeStateNoticeComponent,\n    CyclesResultComponent,\n    CyclesEmissionsChartComponent,\n    NgTemplateOutlet,\n    CyclesNodesTimelineComponent,\n    NodeLogsModelsComponent,\n    NodeValueDetailsComponent,\n    FormsModule,\n    CompoundPipe,\n    DefaultPipe,\n    EllipsisPipe,\n    KeyToLabelPipe,\n    PrecisionPipe\n  ]\n})\nexport class CyclesNodesComponent {\n  private readonly modalService = inject(NgbModal);\n  private readonly nodeStoreService = inject(HeNodeStoreService);\n\n  protected readonly faDownload = faDownload;\n\n  protected readonly dataState = input(DataState.original);\n  protected readonly nodeKeys = input.required<BlankNodesKey[]>();\n  protected readonly nodeKeyGroup = input<CycleNodesKeyGroup>();\n\n  protected readonly BlankNodesKey = BlankNodesKey;\n  protected readonly firstNodeKey = computed(() => this.nodeKeys()?.[0]);\n  protected readonly selectedNodeKey = signal(undefined as BlankNodesKey);\n\n  protected readonly selectedGroupTerm = signal(undefined as Term);\n\n  protected readonly schemaBaseUrl = schemaBaseUrl();\n  protected readonly propertyValue = propertyValue;\n  protected readonly defaultLabel = defaultLabel;\n\n  protected readonly headerKeys = computed(() => [\n    'cycle.id',\n    'cycle.@id',\n    'transformation.id',\n    'transformation.@id',\n    ...this.nodeKeys().flatMap(key => [`cycle.${key}.`, `transformation.${key}.`])\n  ]);\n\n  protected readonly View = View;\n  protected readonly viewIcon = viewIcon;\n  private readonly showView = computed(\n    () =>\n      ({\n        [View.chart]: this.selectedNodeKey() === BlankNodesKey.emissions ? this.cycles().length > 1 : this.hasData(),\n        [View.logs]: !this.isOriginal() && this.hasRecalculatedNodes(),\n        [View.table]: true,\n        [View.timeline]: this.enableTimeline()\n      }) as { [view in View]: boolean }\n  );\n  protected readonly views = computed(\n    () => nodeKeyViews[this.selectedNodeKey()]?.filter(view => this.showView()[view]) ?? []\n  );\n  protected readonly selectedView = signal(View.table);\n\n  private readonly _allNodes = toSignal(this.nodeStoreService.find$<ICycleJSONLD>(NodeType.Cycle));\n  private readonly originalNodes = computed(() => this._allNodes()?.map(data => data[DataState.original]) || []);\n  private readonly currentNodes = computed(() => this._allNodes()?.map(data => data[this.dataState()]) || []);\n\n  // can be Cycle or Transformation\n  protected readonly originalCycles = computed(() =>\n    this.nodeKeyGroup()\n      ? filterGroupNodesByTerm(this.nodeKeyGroup(), this.originalNodes(), this.selectedGroupTerm())\n      : this.originalNodes()\n  );\n  protected readonly cycles = computed(() =>\n    this.nodeKeyGroup()\n      ? filterGroupNodesByTerm(this.nodeKeyGroup(), this.currentNodes(), this.selectedGroupTerm())\n      : this.currentNodes()\n  );\n\n  private readonly selectedIndex = signal(0);\n  private readonly ogirinalSelectedCycle = computed(() => this.originalCycles()?.[this.selectedIndex()]);\n  private readonly selectedCycle = computed(() => this.cycles()?.[this.selectedIndex()]);\n  protected readonly selectedLogsKey = computed(() => logsKey(this.selectedCycle()));\n  protected readonly selectedOriginalValues = computed(\n    () => this.ogirinalSelectedCycle()?.[this.selectedNodeKey()] || []\n  );\n  protected readonly selectedRecalculatedValues = computed(() => this.selectedCycle()?.[this.selectedNodeKey()] || []);\n  // TODO: handle transformation and non-indexed nodes on CE\n  protected readonly selectedNode = computed(() =>\n    this.selectedCycle()\n      ? {\n          ...this.selectedCycle(),\n          '@type': NodeType.Cycle,\n          type: NodeType.Cycle,\n          dataState: DataState.recalculated\n        }\n      : null\n  );\n\n  private readonly isOriginal = computed(() => this.dataState() === DataState.original);\n  private readonly originalValues = computed(() => (this.isOriginal() ? null : this.originalCycles()) || []);\n\n  private readonly hasRecalculatedNodes = computed(() =>\n    this.cycles().some(({ aggregated }: { aggregated?: any }) => !aggregated)\n  );\n  protected readonly showSwitchToRecalculated = computed(() => this.isOriginal() && this.hasRecalculatedNodes());\n\n  protected readonly timelineValues = computed(() =>\n    filterValuesTimeline(this.selectedCycle()?.[this.selectedNodeKey()] || [])\n  );\n  private readonly enableTimeline = computed(() => this.timelineValues().length > 0);\n\n  protected readonly isNodeKeyAllowed = computed(() => nodeKeyViews[this.firstNodeKey()].includes(this.selectedView()));\n\n  private readonly groupedNodes = computed(() =>\n    this.originalNodes().flatMap(cycle => cycleGroupNodes(this.nodeKeyGroup(), cycle))\n  );\n  protected readonly groupNodeTerms = computed(() => unique(this.groupedNodes().map(({ term }) => term)));\n  protected readonly isGroupNode = computed(() => this.nodeKeyGroup() && this.groupedNodes()?.length > 0);\n\n  protected readonly isEmission = computed(() => this.selectedNodeKey() === BlankNodesKey.emissions);\n\n  protected readonly data = computed(() =>\n    this.isEmission()\n      ? this.groupEmissions()\n      : Object.fromEntries(this.nodeKeys().map(nodeKey => [nodeKey, this.groupNodesByKey({ nodeKey })]))\n  );\n  protected readonly dataKeys = computed(() => Object.keys(this.data()));\n\n  protected readonly hasData = computed(() => Object.values(this.data()).flat().length > 0);\n\n  protected readonly filterTermTypes = computed(() => nodeKeyFilterTermTypes[this.selectedNodeKey()]);\n  protected readonly filterTerm = signal('');\n\n  protected readonly csvFilename = computed(() => `${this.nodeKeys().join('-')}.csv`);\n\n  constructor() {\n    effect(\n      () => {\n        // make sure logs does not remain displayed\n        if ((this.isOriginal() && this.selectedView() === View.logs) || !this.views().includes(this.selectedView())) {\n          this.selectedView.set(View.table);\n        }\n      },\n      { allowSignalWrites: true }\n    );\n\n    effect(\n      () => {\n        if (!this.selectedNodeKey() || !this.nodeKeys().includes(this.selectedNodeKey())) {\n          this.selectedNodeKey.set(this.firstNodeKey());\n          this.selectedView.set(View.table);\n        }\n      },\n      { allowSignalWrites: true }\n    );\n\n    effect(\n      () => {\n        if (!this.selectedGroupTerm() || !this.groupNodeTerms().includes(this.selectedGroupTerm())) {\n          this.selectedGroupTerm.set(this.groupNodeTerms()[0]);\n        }\n      },\n      { allowSignalWrites: true }\n    );\n  }\n\n  private groupNodesByKey<\n    T extends ICycleConfig,\n    R = T extends ICycleConfigAnimal\n      ? Animal\n      : T extends ICycleConfigEmission\n        ? Emission\n        : T extends ICycleConfigInput\n          ? HestiaInput\n          : T extends ICycleConfigPractice\n            ? Practice\n            : Product\n  >({ nodeKey }: T): IGroupedKeys<R>[] {\n    const nodesPerCycle = groupNodesByTerm<ICycleJSONLD | groupedNode, R>(\n      this.cycles(),\n      nodeKey,\n      this.originalValues(),\n      filterBlankNode(this.filterTerm())\n    );\n    return orderBy(grouppedKeys(nodesPerCycle), ['key'], ['asc']);\n  }\n\n  private groupEmissions() {\n    const emissionsPerCycle = groupNodesByTerm<ICycleJSONLD | groupedNode, Emission>(\n      this.cycles(),\n      BlankNodesKey.emissions,\n      this.originalValues(),\n      filterBlankNode(this.filterTerm())\n    );\n    const values: IGroupedKeys<Emission>[] = orderBy(grouppedKeys(emissionsPerCycle), ['key'], ['asc']);\n    const methodTiers = orderBy(\n      unique(values.map(({ value }) => value.methodTier)).map(methodTier => ({\n        methodTier,\n        methodTierOrder: methodTierOrder(methodTier)\n      })),\n      ['methodTierOrder'],\n      ['asc']\n    );\n    return Object.fromEntries(\n      methodTiers.map(({ methodTier }) => [methodTier, values.filter(({ value }) => value.methodTier === methodTier)])\n    ) as groupedEmissions;\n  }\n\n  protected trackById(_index: number, item: ICycleJSONLD | groupedNode) {\n    return item['@id'];\n  }\n\n  protected selectIndex({ target: { value } }) {\n    this.selectedIndex.set(+value);\n  }\n\n  protected cycleNode(cycle: ICycleJSONLD | groupedNode) {\n    return 'term' in cycle ? cycle.term : cycle;\n  }\n\n  protected showDownload() {\n    const instance = this.modalService.open(NodeCsvExportConfirmComponent);\n    const component = instance.componentInstance as NodeCsvExportConfirmComponent;\n    component.nodes.set(this.cycles());\n    component.filename.set(this.csvFilename());\n    component.isUpload.set(false);\n    component.headerKeys.set(this.headerKeys());\n  }\n}\n","@if (isGroupNode()) {\n  <div class=\"tabs is-mb-1\">\n    <ul>\n      @for (term of groupNodeTerms(); track term.name) {\n        <li [class.is-active]=\"selectedGroupTerm() === term\">\n          <a (click)=\"selectedGroupTerm.set(term)\">{{ term.name }}</a>\n        </li>\n      }\n    </ul>\n  </div>\n}\n\n<div class=\"columns is-variable is-align-items-center is-2 is-m-0\">\n  <div class=\"column\">\n    @if (hasData() && selectedView() === View.table) {\n      <button class=\"button is-small is-ghost\" (click)=\"showDownload()\">\n        <fa-icon [icon]=\"faDownload\" size=\"lg\" />\n      </button>\n    }\n  </div>\n  @if (selectedView() === View.table) {\n    <div class=\"column is-narrow\">\n      <he-search-extend\n        class=\"is-secondary is-small\"\n        placeholder=\"Filter terms by name\"\n        (search)=\"filterTerm.set($event)\" />\n    </div>\n  }\n  @if (views()?.length > 1) {\n    <div class=\"column is-narrow\">\n      <div class=\"field has-addons\">\n        @for (view of views(); track view) {\n          <div class=\"control\">\n            <button\n              class=\"button is-small\"\n              [class.is-selected]=\"selectedView() === view\"\n              (click)=\"selectedView.set(view)\">\n              <span class=\"icon is-small\">\n                <fa-icon [icon]=\"viewIcon[view]\" aria-hidden=\"true\" />\n              </span>\n              <span class=\"is-hidden-mobile\">{{ view }}</span>\n            </button>\n          </div>\n        }\n      </div>\n    </div>\n  }\n</div>\n\n@if (isNodeKeyAllowed()) {\n  @switch (selectedView()) {\n    @case (View.table) {\n      <div class=\"is-px-3 is-pb-3\">\n        @if (hasData()) {\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                @if (dataKeys().length > 1) {\n                  <tr class=\"has-text-weight-bold\">\n                    <th class=\"width-auto\"></th>\n                    <th [class.is-hidden]=\"isGroupNode()\"></th>\n                    @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n                      @if (data()[dataKey]?.length) {\n                        <th [attr.colspan]=\"data()[dataKey].length\" [class.has-border-right]=\"!dataKeyLast\">\n                          <span>{{ dataKey | keyToLabel }}</span>\n                        </th>\n                      }\n                    }\n                  </tr>\n                }\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]=\"isGroupNode()\"></th>\n                  @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n                    @for (node of data()[dataKey]; track node.value.term.name; let nodeLast = $last) {\n                      <th\n                        [attr.title]=\"node.value.term.name\"\n                        [class.has-border-right]=\"dataKeys().length > 1 && !dataKeyLast && nodeLast\">\n                        <he-node-link [node]=\"node.value.term\">\n                          <span\n                            [innerHtml]=\"\n                              node.value.term.name | ellipsis: 30 | compound: node.value.term.termType\n                            \"></span>\n                        </he-node-link>\n                      </th>\n                    }\n                  }\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]=\"isGroupNode()\">\n                    <a [href]=\"schemaBaseUrl + '/Cycle#functionalUnit'\" target=\"_blank\">Functional unit</a>\n                  </th>\n                  @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n                    @for (node of data()[dataKey]; track node.value.term.name; let nodeLast = $last) {\n                      <th\n                        [attr.title]=\"node.value.term.units\"\n                        [class.has-border-right]=\"dataKeys().length > 1 && !dataKeyLast && nodeLast\">\n                        <span [innerHtml]=\"node.value.term.units | compound: node.value.term.termType\"></span>\n                        <he-terms-units-description class=\"is-inline-block is-ml-2\" [term]=\"node.value.term\" />\n                      </th>\n                    }\n                  }\n                </tr>\n              </thead>\n              <tbody>\n                @for (cycle of cycles(); track trackById(cycleIndex, cycle); let cycleIndex = $index) {\n                  <tr>\n                    <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n                      <he-node-link [node]=\"cycleNode(cycle)\">\n                        <span>{{ cycleIndex + 1 }}. {{ defaultLabel(cycle) }}</span>\n                      </he-node-link>\n                    </td>\n                    <td class=\"has-border-right\" [class.is-hidden]=\"isGroupNode()\">\n                      <he-cycles-functional-unit-measure [cycle]=\"cycle\" />\n                    </td>\n                    @for (dataKey of dataKeys(); track dataKey; let dataKeyLast = $last) {\n                      @for (node of data()[dataKey]; track node.value.term.name; let nodeLast = $last) {\n                        <td\n                          class=\"is-nowrap\"\n                          [class.has-border-right]=\"dataKeys().length > 1 && !dataKeyLast && nodeLast\">\n                          @if (node.value.values[cycle['@id']]) {\n                            <span\n                              class=\"trigger-popover\"\n                              [ngbPopover]=\"details\"\n                              autoClose=\"outside\"\n                              popoverClass=\"is-narrow is-overflow-visible\"\n                              placement=\"left\"\n                              container=\"body\"\n                              [popoverContext]=\"{ data: node.value.values[cycle['@id']], cycle, key: dataKey }\">\n                              <span pointer>\n                                {{\n                                  propertyValue(node.value.values[cycle['@id']].value, node.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]=\"node.value.values[cycle['@id']].node\"\n                                key=\"value\" />\n                            </span>\n                          } @else {\n                            <span>-</span>\n                          }\n                        </td>\n                      }\n                    }\n                  </tr>\n                }\n              </tbody>\n            </table>\n          </he-data-table>\n          <he-blank-node-state-notice\n            [dataState]=\"dataState()\"\n            [showDeleted]=\"firstNodeKey() === BlankNodesKey.emissions\" />\n        } @else {\n          <div class=\"is-pt-3 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            @if (showSwitchToRecalculated()) {\n              <span>\n                Switch to\n                <code>recalculated</code>\n                version.\n              </span>\n            }\n          </div>\n        }\n      </div>\n    }\n    @case (View.chart) {\n      @switch (firstNodeKey()) {\n        @case (BlankNodesKey.inputs) {\n          <he-cycles-result [cycles]=\"cycles()\"></he-cycles-result>\n        }\n        @case (BlankNodesKey.emissions) {\n          <he-cycles-emissions-chart [cycles]=\"cycles()\" />\n        }\n      }\n    }\n    @case (View.timeline) {\n      <ng-container *ngTemplateOutlet=\"selectCycle\" />\n      <div class=\"is-px-3 is-pb-3\">\n        <he-cycles-nodes-timeline [recalculatedValues]=\"timelineValues()\" [dataState]=\"dataState()\" />\n        <he-blank-node-state-notice [dataState]=\"dataState()\" />\n      </div>\n    }\n    @case (View.logs) {\n      <ng-container *ngTemplateOutlet=\"selectCycle\" />\n      @if (nodeKeys().length > 1) {\n        <div class=\"tabs is-mb-2\">\n          <ul>\n            @for (nodeKey of nodeKeys(); track nodeKey) {\n              <li [class.is-active]=\"selectedNodeKey() === nodeKey\">\n                <a (click)=\"selectedNodeKey.set(nodeKey)\">{{ nodeKey | keyToLabel }}</a>\n              </li>\n            }\n          </ul>\n        </div>\n      }\n      @if (selectedNode()) {\n        <he-node-logs-models\n          [node]=\"selectedNode()\"\n          [nodeKey]=\"selectedNodeKey()\"\n          [logsKey]=\"selectedLogsKey()\"\n          [originalValues]=\"selectedOriginalValues()\"\n          [recalculatedValues]=\"selectedRecalculatedValues()\"\n          [filterTermTypes]=\"filterTermTypes()\" />\n      }\n    }\n  }\n}\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    class=\"is-overflow-visible\"\n    [data]=\"data\"\n    [dataState]=\"dataState()\"\n    [nodeType]=\"node['@type']\"\n    [dataKey]=\"key\" />\n</ng-template>\n\n<ng-template #selectCycle>\n  @if (cycles().length > 1) {\n    <div class=\"field has-addons is-py-2 is-px-3\">\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            @for (value of cycles(); track value; let cycleIndex = $index) {\n              <option [value]=\"cycleIndex\">{{ cycleIndex + 1 }}. {{ defaultLabel(value) }}</option>\n            }\n          </select>\n        </div>\n      </div>\n    </div>\n  }\n</ng-template>\n"]}
|