@hestia-earth/ui-components 0.23.0 → 0.23.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/common/utils.d.ts +1 -0
- package/engine/engine-models-version-link/engine-models-version-link.component.d.ts +2 -1
- package/esm2022/common/utils.mjs +8 -1
- package/esm2022/cycles/cycles-activity/cycles-activity.component.mjs +4 -4
- package/esm2022/cycles/cycles-completeness/cycles-completeness.component.mjs +4 -4
- package/esm2022/cycles/cycles-emissions/cycles-emissions.component.mjs +4 -4
- package/esm2022/cycles/cycles-practices/cycles-practices.component.mjs +4 -4
- package/esm2022/engine/engine-models-version-link/engine-models-version-link.component.mjs +6 -3
- package/esm2022/engine/engine-orchestrator-edit/engine-orchestrator-edit.component.mjs +2 -2
- package/esm2022/files/files-error.model.mjs +11 -10
- package/esm2022/files/files-form/files-form.component.mjs +3 -12
- package/esm2022/files/files-upload-errors/files-upload-errors.component.mjs +6 -10
- package/esm2022/impact-assessments/impact-assessments-products/impact-assessments-products.component.mjs +5 -3
- package/esm2022/node/node-csv-select-headers/node-csv-select-headers.component.mjs +2 -2
- package/esm2022/node/node-logs-models/node-logs-models.component.mjs +6 -5
- package/esm2022/select/select.component.mjs +2 -2
- package/esm2022/sites/sites-measurements/sites-measurements.component.mjs +4 -4
- package/fesm2022/hestia-earth-ui-components.mjs +58 -57
- package/fesm2022/hestia-earth-ui-components.mjs.map +1 -1
- package/files/files-form/files-form.component.d.ts +1 -1
- package/files/files-upload-errors/files-upload-errors.component.d.ts +3 -6
- package/node/node-logs-models/node-logs-models.component.d.ts +1 -0
- package/package.json +4 -4
|
@@ -2,7 +2,7 @@ import { ChangeDetectionStrategy, Component, Input } from '@angular/core';
|
|
|
2
2
|
import { parse } from 'papaparse';
|
|
3
3
|
import { SchemaType, NodeType, UploadLimit, isExpandable, typeToSchemaType } from '@hestia-earth/schema';
|
|
4
4
|
import { maxFileSizeMb, SupportedExtensions } from '@hestia-earth/api';
|
|
5
|
-
import { toCsv } from '@hestia-earth/schema-convert';
|
|
5
|
+
import { toCsv, ErrorKeys as SchemaErrorKeys } from '@hestia-earth/schema-convert';
|
|
6
6
|
import { baseUrl } from '../../common';
|
|
7
7
|
import { schemaBaseUrl } from '../../schema/schema.service';
|
|
8
8
|
import * as i0 from "@angular/core";
|
|
@@ -24,16 +24,11 @@ export var ErrorKeys;
|
|
|
24
24
|
ErrorKeys["InvalidFirstColumn"] = "invalid-first-column";
|
|
25
25
|
ErrorKeys["DuplicatedHeaders"] = "duplicated-headers";
|
|
26
26
|
ErrorKeys["DuplicatedIds"] = "duplicated-ids";
|
|
27
|
-
ErrorKeys["DuplicatedIdFields"] = "duplicated-id-fields";
|
|
28
|
-
ErrorKeys["PropertyInvalidFormat"] = "property-invalid-format";
|
|
29
|
-
ErrorKeys["PropertyNotFound"] = "property-not-found";
|
|
30
27
|
ErrorKeys["PropertyRequired"] = "property-required";
|
|
31
28
|
ErrorKeys["PropertyInternal"] = "property-internal";
|
|
32
|
-
ErrorKeys["SchemaNotFound"] = "schema-not-found";
|
|
33
29
|
ErrorKeys["UploadsLimit"] = "upload-limit";
|
|
34
30
|
ErrorKeys["NestedHeaders"] = "nested-headers";
|
|
35
31
|
ErrorKeys["ReferenceExistingHeaders"] = "reference-existing-headers";
|
|
36
|
-
ErrorKeys["ObjectArrayInvalid"] = "object-array-invalid";
|
|
37
32
|
ErrorKeys["MaxSize"] = "max-size";
|
|
38
33
|
ErrorKeys["MaxRows"] = "max-rows";
|
|
39
34
|
})(ErrorKeys || (ErrorKeys = {}));
|
|
@@ -76,6 +71,7 @@ export class FilesUploadErrorsComponent {
|
|
|
76
71
|
constructor() {
|
|
77
72
|
this.baseUrl = baseUrl();
|
|
78
73
|
this.nodeTypes = acceptedTypes;
|
|
74
|
+
this.SchemaErrorKeys = SchemaErrorKeys;
|
|
79
75
|
this.ErrorKeys = ErrorKeys;
|
|
80
76
|
this.UploadLimit = UploadLimit;
|
|
81
77
|
this.maxFileSizeMb = maxFileSizeMb;
|
|
@@ -119,7 +115,7 @@ export class FilesUploadErrorsComponent {
|
|
|
119
115
|
return [this.schemaUrl, this.error?.schemaKey || this.error?.key].filter(Boolean).join('#');
|
|
120
116
|
}
|
|
121
117
|
get isSchemaError() {
|
|
122
|
-
return [
|
|
118
|
+
return [SchemaErrorKeys.PropertyNotFound, SchemaErrorKeys.SchemaNotFound].includes(this.error?.message);
|
|
123
119
|
}
|
|
124
120
|
get showCsvPreview() {
|
|
125
121
|
return this.headers?.length && this.rows?.length && !this.isSchemaError && !this.isJSONFile;
|
|
@@ -128,14 +124,14 @@ export class FilesUploadErrorsComponent {
|
|
|
128
124
|
return this.error?.node && this.isJSONFile;
|
|
129
125
|
}
|
|
130
126
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.8", ngImport: i0, type: FilesUploadErrorsComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
131
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.8", type: FilesUploadErrorsComponent, selector: "he-files-upload-errors", inputs: { file: "file", error: "error" }, ngImport: i0, template: "<ng-container *ngIf=\"!!error\">\n <div class=\"has-text-danger\" [ngSwitch]=\"error?.message\">\n <div *ngSwitchCase=\"ErrorKeys.InvalidJSON\">\n <p>The uploaded content does not appear to contain valid JSON data:</p>\n\n <div class=\"is-mt-2\">\n <pre><code>{{error.error}}</code></pre>\n </div>\n\n <p>\n Please validate your JSON content before uploading it using an\n <a href=\"https://jsoneditoronline.org\" target=\"_blank\">online tool</a>\n for example.\n </p>\n </div>\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.NoData\">\n <p>No Nodes could be found in your Upload. Please verify the column headers match the Hestia schema.</p>\n\n <ng-container *ngTemplateOutlet=\"defaultError\"></ng-container>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.PropertyNotFound\">\n <div class=\"notification is-danger\" role=\"alert\">\n <p>\n <fa-icon class=\"pr-2\" icon=\"exclamation-triangle\"></fa-icon>\n <span>You have column headers that do not match the Hestia schema.</span>\n <span class=\"px-1\">Please refer to the</span>\n <a [href]=\"schemaUrl\" target=\"_blank\">schema</a>\n <span class=\"pl-1\">and then correct the column headers indicated below.</span>\n </p>\n\n <ng-container *ngTemplateOutlet=\"defaultError\"></ng-container>\n </div>\n\n <div class=\"is-mb-2\">\n <pre><code>{{error.key}}</code></pre>\n </div>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.PropertyInvalidFormat\">\n <div class=\"is-mb-2\">\n <span>\n The following value in the column\n <i>{{ error.key }}</i>\n is not in the correct format:\n </span>\n <br />\n <pre class=\"is-mt-2\"><code>{{stringify(error.value)}}</code></pre>\n </div>\n <ng-container [ngSwitch]=\"hasNumberWithCommasError\">\n <ng-container *ngSwitchCase=\"true\">\n <p class=\"is-mt-2\">\n Please format your numbers to remove commas and use a decimal point to separate the whole part of the number\n from the fractional part of the number.\n </p>\n\n <p class=\"is-mt-2 has-text-danger has-text-underline\">Incorrect:</p>\n <pre class=\"is-mt-2\"><code>3,510.1</code></pre>\n\n <p class=\"is-mt-2 has-text-success has-text-underline\">Correct:</p>\n <pre class=\"is-mt-2\"><code>3510.1</code></pre>\n </ng-container>\n\n <ng-container *ngSwitchCase=\"false\">\n <ng-container *ngTemplateOutlet=\"schemaError\"></ng-container>\n\n <p>The error could either be a value in the column, or an error with the column header.</p>\n <p>\n If you are trying to set no data, we accept either\n <code>-</code>\n or an empty cell.\n </p>\n </ng-container>\n </ng-container>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.PropertyRequired\">\n <div class=\"is-mb-2\">\n <span class=\"pr-1\">The following field is required in the</span>\n <a class=\"pr-1\" *ngIf=\"error.schema\" [href]=\"schemaUrl\" target=\"_blank\">{{ error.schema }}</a>\n <span>schema:</span>\n <br />\n <pre class=\"is-mt-2\"><code>{{error.property}}</code></pre>\n </div>\n <p [ngSwitch]=\"error.property\">\n <span *ngSwitchCase=\"'id'\">\n You must assign a unique\n <code>id</code>\n for every record in your data.\n </span>\n <span *ngSwitchCase=\"'type'\">\n Every record in your data must contain a\n <code>type</code>\n .\n </span>\n </p>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.PropertyInternal\">\n <div class=\"is-mb-2\">\n <span class=\"pr-1\">\n The following field is\n <code>internal</code>\n in the\n </span>\n <a [href]=\"schemaKeyUrl\" target=\"_blank\">{{ error.schema }} schema</a>\n <span>:</span>\n <br />\n <pre class=\"is-mt-2\"><code>{{error.schemaKey || error.key}}</code></pre>\n </div>\n <p>\n You cannot upload\n <code>internal</code>\n fields to Hestia.\n </p>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.SchemaNotFound\">\n <div class=\"is-mb-2\">\n <span>The following Node does not exist in the schema:</span>\n <br />\n <pre class=\"is-mt-2\"><code>{{error.key}}</code></pre>\n </div>\n <p>\n <span>The list of accepted values is:</span>\n </p>\n <div class=\"content\">\n <ul>\n <li *ngFor=\"let type of nodeTypes\">\n <i>{{ type }}</i>\n </li>\n </ul>\n </div>\n </div>\n\n <ng-container *ngSwitchCase=\"ErrorKeys.UploadsLimit\">\n <div class=\"is-mb-2\">\n <span>You have reached the upload limit of</span>\n <code class=\"mx-1\">{{ UploadLimit[error.schema] }}</code>\n <span>\n {{ error.schema }} in a single file (\n <b>{{ error.value }}</b>\n have been found).\n </span>\n </div>\n\n <p>\n <span>Please re-upload these data in multiple files with a maximum of</span>\n <code class=\"mx-1\">{{ UploadLimit[error.schema] }}</code>\n <span>{{ error.schema }} in each file.</span>\n </p>\n </ng-container>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.InvalidFirstColumn\">\n <span>\n No data in the Hestia format was detected on the first sheet. Either try selecting a different sheet in the\n upload screen, or check the sheet to ensure the headers follow the schema - see examples\n </span>\n <a class=\"pl-1\" routerLink=\"/upload\" fragment=\"example\">here</a>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.NoHeaders\">\n <p>\n No headers matching the Hestia schema were found on the first row. Please remove all empty rows at the begining\n of the file and try uploading again.\n </p>\n\n <ng-container *ngTemplateOutlet=\"defaultError\"></ng-container>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.InvalidSheetName\">\n <div class=\"is-mb-2\">\n <span class=\"pr-1\">The Excel sheet</span>\n <code>{{ error.value }}</code>\n <span class=\"pl-1\">does not exist.</span>\n <span class=\"pl-1\">Your Excel file contains the following sheets:</span>\n <ul class=\"mt-1 content is-list-style-disc pl-5\">\n <li *ngFor=\"let e of error.error.split(';')\">\n <code>{{ e }}</code>\n </li>\n </ul>\n </div>\n\n <p>\n <span>Please re-upload selecting one of the sheets above.</span>\n </p>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.DuplicatedHeaders\">\n <span>Your upload contains duplicated column headers.</span>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.DuplicatedIds\">\n <span class=\"pr-1\">You have multiple</span>\n <code>{{ error.schema | pluralize }}</code>\n <span class=\"px-1\">with the same</span>\n <a [href]=\"schemaKeyUrl\" target=\"_blank\">{{ error.schemaKey }}</a>\n <span class=\"px-1\">but different data. The</span>\n <a [href]=\"schemaKeyUrl\" target=\"_blank\">{{ error.schemaKey }}</a>\n <span class=\"pl-1\">is:</span>\n <br />\n <pre class=\"is-mt-2\"><code>{{error.value}}</code></pre>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.DuplicatedIdFields\">\n <span class=\"is-pr-1\">You have used both the</span>\n <code>@id</code>\n <span class=\"is-px-1\">and</span>\n <code>id</code>\n <span class=\"is-px-1\">fields in the same data. Please use</span>\n <code>@id</code>\n <span class=\"is-px-1\">when linking to existing nodes on the Platform, and</span>\n <code>id</code>\n <span class=\"is-px-1\">when linking to nodes you are uploading in this file using the same</span>\n <code>id</code>\n <span class=\"is-px-1\">field.</span>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.NestedHeaders\">\n <p>\n To link nodes together, you first need to define the node (e.g.,\n <code>source.id</code>\n ,\n <code>source.bibliography.title</code>\n ) and then create the link (e.g.,\n <code>cycle.defaultSource.id</code>\n ). We do not allow nesting (e.g.,\n <code>cycle.defaultSource.bibliography.title</code>\n ).\n </p>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.ReferenceExistingHeaders\">\n <p>\n You have used the internal\n <code>@id</code>\n field to upload a new Node.\n </p>\n <p>\n The\n <code>@id</code>\n field should only be used to reference existing Node or Terms. Example:\n </p>\n\n <p class=\"is-mt-2 has-text-danger has-text-underline\">Incorrect:</p>\n <pre class=\"is-mt-2\"><code>cycle.@id,cycle.inputs.0.term.@id,source.id</code></pre>\n\n <p class=\"is-mt-2 has-text-success has-text-underline\">Correct:</p>\n <pre class=\"is-mt-2\"><code>cycle.id,cycle.inputs.0.term.@id,source.id</code></pre>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.ObjectArrayInvalid\">\n <p>The values in the column {{ error.key }} must be represented as arrays.</p>\n\n <ng-container *ngTemplateOutlet=\"schemaError\"></ng-container>\n\n <p>Examples:</p>\n <pre class=\"is-mt-2\"><code>{{error.key}}.0.term.name</code></pre>\n <pre class=\"is-mt-2\"><code>{{error.key}}.0.value</code></pre>\n <pre class=\"is-mt-2\"><code>{{error.key}}.1.term.name</code></pre>\n <pre class=\"is-mt-2\"><code>{{error.key}}.1.value</code></pre>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.MaxSize\">\n <span>The maximum file size limit is {{ maxFileSizeMb }}Mb (current file size: {{ error.value }}Mb).</span>\n <span class=\"pl-1\">Please re-upload these data in multiple files.</span>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.MaxRows\">\n <span>\n Your file has {{ error.value | number }} rows. This might be an error caused by adding some data at the bottom\n of a spreadsheet by accident: delete any unused rows and resubmit the file.\n </span>\n </div>\n\n <div class=\"is-mb-2\" *ngIf=\"hasGeoJSONError\">\n <p>If the GeoJSON is too large, please try to simplify it following these steps:</p>\n <ul class=\"is-my-1 content is-list-style-decimal pl-5\">\n <li>\n <span class=\"is-pr-1\">Go to</span>\n <a href=\"https://mapshaper.org/\" target=\"_blank\">https://mapshaper.org/</a>\n </li>\n <li>\n <span>\n Save the boundary in a valid\n <code>.geojson</code>\n file.\n </span>\n </li>\n <li>\n <span>Import it on mapshaper.</span>\n </li>\n <li>\n <span>Click \"Simplify\" in the top right corner in the menu bar.</span>\n </li>\n <li>\n <span>Adjust the percentage to a lower value, like 50% to start with.</span>\n </li>\n <li>\n <span>\n Export to\n <code>GeoJson</code>\n or\n <code>CSV</code>\n .\n </span>\n </li>\n <li>\n <span>Copy the resulting content into your upload file.</span>\n </li>\n </ul>\n <p>If you encounter the error again, try again using a lower percentage at step 5.</p>\n </div>\n\n <div class=\"my-3\" *ngIf=\"columns.length || (error.index && !isSchemaError)\">\n <b>Hint:</b>\n <span class=\"pl-1\">the error might be located</span>\n <span class=\"pl-1\" *ngIf=\"error.index && !isSchemaError\">on row {{ error.index }}</span>\n <ng-container *ngIf=\"columns.length\">\n <span class=\"pl-1\">in {{ 'column' | pluralize: columns.length }}:</span>\n <ul class=\"mt-1 content is-list-style-disc pl-5\">\n <li *ngFor=\"let column of columns\">\n <b class=\"pr-2\">{{ column.column }}</b>\n <code>{{ column.name }}</code>\n </li>\n </ul>\n </ng-container>\n </div>\n\n <div class=\"my-3\" *ngIf=\"error.suggestions?.length\">\n <b>Suggestions:</b>\n <span class=\"pl-1\">did you mean instead:</span>\n <ul class=\"mt-1 content is-list-style-disc pl-5\">\n <li *ngFor=\"let suggestion of error.suggestions\">\n <code>{{ suggestion }}</code>\n </li>\n </ul>\n </div>\n\n <div *ngIf=\"showCsvPreview\" class=\"is-my-2\">\n <b>\n Preview\n <sup>1</sup>\n :\n </b>\n <div class=\"table-container is-mt-2\">\n <table class=\"table is-bordered is-fullwidth is-hoverable is-narrow mb-0\">\n <thead>\n <th *ngIf=\"error.index\"></th>\n <th *ngFor=\"let header of headers\">{{ header }}</th>\n </thead>\n <tbody>\n <ng-container *bindOnce=\"rows\">\n <tr *ngFor=\"let row of rows\">\n <td class=\"has-text-danger\" *ngIf=\"error.index\">\n <span class=\"is-nowrap\">Row {{ error.index }}</span>\n </td>\n <td *ngFor=\"let col of row; let colIndex = index\">\n <span\n [class.has-text-danger]=\"\n error?.message === ErrorKeys.DuplicatedIds && hasDuplicatedError(col, colIndex)\n \">\n {{ col | ellipsis: 50 }}\n </span>\n </td>\n </tr>\n </ng-container>\n </tbody>\n </table>\n </div>\n <p class=\"is-size-7\">\n <i>1. The preview does not necessarily reflect the original Upload's order or content.</i>\n </p>\n </div>\n\n <div *ngIf=\"showJsonPreview\" class=\"is-my-2\">\n <b>Preview:</b>\n <pre class=\"is-mt-2\"><code>{{stringify(error.node)}}</code></pre>\n </div>\n </div>\n</ng-container>\n\n<ng-template #defaultError>\n <p class=\"is-mt-2 has-text-black\">\n <span class=\"is-pr-1\">If you are still stuck, please see</span>\n <a [href]=\"baseUrl + '/upload#example'\" target=\"_blank\">the example uploads and videos</a>\n <span class=\"is-px-1\">\n or try using the Wizard to get the first few rows of data formatted correctly, or use the chat to ask for help.\n </span>\n </p>\n</ng-template>\n\n<ng-template #schemaError>\n <p>\n <span class=\"pr-1\">You can see the accepted field type and some examples</span>\n <a [href]=\"schemaKeyUrl\" target=\"_blank\">here</a>\n <span>.</span>\n </p>\n</ng-template>\n", styles: [":host{display:block}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i1.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { kind: "directive", type: i1.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { kind: "component", type: i2.FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "spin", "pulse", "mask", "styles", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "classes", "transform", "a11yRole"] }, { kind: "directive", type: i3.BindOnceDirective, selector: "[bindOnce]", inputs: ["bindOnce"] }, { kind: "pipe", type: i1.DecimalPipe, name: "number" }, { kind: "pipe", type: i4.EllipsisPipe, name: "ellipsis" }, { kind: "pipe", type: i5.PluralizePipe, name: "pluralize" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
127
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.8", type: FilesUploadErrorsComponent, selector: "he-files-upload-errors", inputs: { file: "file", error: "error" }, ngImport: i0, template: "<ng-container *ngIf=\"!!error\">\n <div class=\"has-text-danger\" [ngSwitch]=\"error?.message\">\n <div *ngSwitchCase=\"ErrorKeys.InvalidJSON\">\n <p>The uploaded content does not appear to contain valid JSON data:</p>\n\n <div class=\"is-mt-2\">\n <pre><code>{{error.error}}</code></pre>\n </div>\n\n <p>\n Please validate your JSON content before uploading it using an\n <a href=\"https://jsoneditoronline.org\" target=\"_blank\">online tool</a>\n for example.\n </p>\n </div>\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.NoData\">\n <p>No Nodes could be found in your Upload. Please verify the column headers match the Hestia schema.</p>\n\n <ng-container *ngTemplateOutlet=\"defaultError\"></ng-container>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"SchemaErrorKeys.PropertyNotFound\">\n <div class=\"notification is-danger\" role=\"alert\">\n <p>\n <fa-icon class=\"pr-2\" icon=\"exclamation-triangle\"></fa-icon>\n <span>You have column headers that do not match the Hestia schema.</span>\n <span class=\"px-1\">Please refer to the</span>\n <a [href]=\"schemaUrl\" target=\"_blank\">schema</a>\n <span class=\"pl-1\">and then correct the column headers indicated below.</span>\n </p>\n\n <ng-container *ngTemplateOutlet=\"defaultError\"></ng-container>\n </div>\n\n <div class=\"is-mb-2\">\n <pre><code>{{error.key}}</code></pre>\n </div>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"SchemaErrorKeys.PropertyInvalidFormat\">\n <div class=\"is-mb-2\">\n <span>\n The following value in the column\n <i>{{ error.key }}</i>\n is not in the correct format:\n </span>\n <br />\n <pre class=\"is-mt-2\"><code>{{stringify(error.value)}}</code></pre>\n </div>\n <ng-container [ngSwitch]=\"hasNumberWithCommasError\">\n <ng-container *ngSwitchCase=\"true\">\n <p class=\"is-mt-2\">\n Please format your numbers to remove commas and use a decimal point to separate the whole part of the number\n from the fractional part of the number.\n </p>\n\n <p class=\"is-mt-2 has-text-danger has-text-underline\">Incorrect:</p>\n <pre class=\"is-mt-2\"><code>3,510.1</code></pre>\n\n <p class=\"is-mt-2 has-text-success has-text-underline\">Correct:</p>\n <pre class=\"is-mt-2\"><code>3510.1</code></pre>\n </ng-container>\n\n <ng-container *ngSwitchCase=\"false\">\n <ng-container *ngTemplateOutlet=\"schemaError\"></ng-container>\n\n <p>The error could either be a value in the column, or an error with the column header.</p>\n <p>\n If you are trying to set no data, we accept either\n <code>-</code>\n or an empty cell.\n </p>\n </ng-container>\n </ng-container>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.PropertyRequired\">\n <div class=\"is-mb-2\">\n <span class=\"pr-1\">The following field is required in the</span>\n <a class=\"pr-1\" *ngIf=\"error.schema\" [href]=\"schemaUrl\" target=\"_blank\">{{ error.schema }}</a>\n <span>schema:</span>\n <br />\n <pre class=\"is-mt-2\"><code>{{error.property}}</code></pre>\n </div>\n <p [ngSwitch]=\"error.property\">\n <span *ngSwitchCase=\"'id'\">\n You must assign a unique\n <code>id</code>\n for every record in your data.\n </span>\n <span *ngSwitchCase=\"'type'\">\n Every record in your data must contain a\n <code>type</code>\n .\n </span>\n </p>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.PropertyInternal\">\n <div class=\"is-mb-2\">\n <span class=\"pr-1\">\n The following field is\n <code>internal</code>\n in the\n </span>\n <a [href]=\"schemaKeyUrl\" target=\"_blank\">{{ error.schema }} schema</a>\n <span>:</span>\n <br />\n <pre class=\"is-mt-2\"><code>{{error.schemaKey || error.key}}</code></pre>\n </div>\n <p>\n You cannot upload\n <code>internal</code>\n fields to Hestia.\n </p>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"SchemaErrorKeys.SchemaNotFound\">\n <div class=\"is-mb-2\">\n <span>The following Node does not exist in the schema:</span>\n <br />\n <pre class=\"is-mt-2\"><code>{{error.key}}</code></pre>\n </div>\n <p>\n <span>The list of accepted values is:</span>\n </p>\n <div class=\"content\">\n <ul>\n <li *ngFor=\"let type of nodeTypes\">\n <i>{{ type }}</i>\n </li>\n </ul>\n </div>\n </div>\n\n <ng-container *ngSwitchCase=\"ErrorKeys.UploadsLimit\">\n <div class=\"is-mb-2\">\n <span>You have reached the upload limit of</span>\n <code class=\"mx-1\">{{ UploadLimit[error.schema] }}</code>\n <span>\n {{ error.schema }} in a single file (\n <b>{{ error.value }}</b>\n have been found).\n </span>\n </div>\n\n <p>\n <span>Please re-upload these data in multiple files with a maximum of</span>\n <code class=\"mx-1\">{{ UploadLimit[error.schema] }}</code>\n <span>{{ error.schema }} in each file.</span>\n </p>\n </ng-container>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.InvalidFirstColumn\">\n <span>\n No data in the Hestia format was detected on the first sheet. Either try selecting a different sheet in the\n upload screen, or check the sheet to ensure the headers follow the schema - see examples\n </span>\n <a class=\"pl-1\" routerLink=\"/upload\" fragment=\"example\">here</a>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.NoHeaders\">\n <p>\n No headers matching the Hestia schema were found on the first row. Please remove all empty rows at the begining\n of the file and try uploading again.\n </p>\n\n <ng-container *ngTemplateOutlet=\"defaultError\"></ng-container>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.InvalidSheetName\">\n <div class=\"is-mb-2\">\n <span class=\"pr-1\">The Excel sheet</span>\n <code>{{ error.value }}</code>\n <span class=\"pl-1\">does not exist.</span>\n <span class=\"pl-1\">Your Excel file contains the following sheets:</span>\n <ul class=\"mt-1 content is-list-style-disc pl-5\">\n <li *ngFor=\"let e of error.error.split(';')\">\n <code>{{ e }}</code>\n </li>\n </ul>\n </div>\n\n <p>\n <span>Please re-upload selecting one of the sheets above.</span>\n </p>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.DuplicatedHeaders\">\n <span>Your upload contains duplicated column headers.</span>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.DuplicatedIds\">\n <span class=\"pr-1\">You have multiple</span>\n <code>{{ error.schema | pluralize }}</code>\n <span class=\"px-1\">with the same</span>\n <a [href]=\"schemaKeyUrl\" target=\"_blank\">{{ error.schemaKey }}</a>\n <span class=\"px-1\">but different data. The</span>\n <a [href]=\"schemaKeyUrl\" target=\"_blank\">{{ error.schemaKey }}</a>\n <span class=\"pl-1\">is:</span>\n <br />\n <pre class=\"is-mt-2\"><code>{{error.value}}</code></pre>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"SchemaErrorKeys.DuplicatedIdFields\">\n <span class=\"is-pr-1\">You have used both the</span>\n <code>@id</code>\n <span class=\"is-px-1\">and</span>\n <code>id</code>\n <span class=\"is-px-1\">fields in the same data. Please use</span>\n <code>@id</code>\n <span class=\"is-px-1\">when linking to existing nodes on the Platform, and</span>\n <code>id</code>\n <span class=\"is-px-1\">when linking to nodes you are uploading in this file using the same</span>\n <code>id</code>\n <span class=\"is-px-1\">field.</span>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.NestedHeaders\">\n <p>\n To link nodes together, you first need to define the node (e.g.,\n <code>source.id</code>\n ,\n <code>source.bibliography.title</code>\n ) and then create the link (e.g.,\n <code>cycle.defaultSource.id</code>\n ). We do not allow nesting (e.g.,\n <code>cycle.defaultSource.bibliography.title</code>\n ).\n </p>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.ReferenceExistingHeaders\">\n <p>\n You have used the internal\n <code>@id</code>\n field to upload a new Node.\n </p>\n <p>\n The\n <code>@id</code>\n field should only be used to reference existing Node or Terms. Example:\n </p>\n\n <p class=\"is-mt-2 has-text-danger has-text-underline\">Incorrect:</p>\n <pre class=\"is-mt-2\"><code>cycle.@id,cycle.inputs.0.term.@id,source.id</code></pre>\n\n <p class=\"is-mt-2 has-text-success has-text-underline\">Correct:</p>\n <pre class=\"is-mt-2\"><code>cycle.id,cycle.inputs.0.term.@id,source.id</code></pre>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"SchemaErrorKeys.ObjectArrayInvalid\">\n <p>The values in the column {{ error.key }} must be represented as arrays.</p>\n\n <ng-container *ngTemplateOutlet=\"schemaError\"></ng-container>\n\n <p>Examples:</p>\n <pre class=\"is-mt-2\"><code>{{error.key}}.0.term.name</code></pre>\n <pre class=\"is-mt-2\"><code>{{error.key}}.0.value</code></pre>\n <pre class=\"is-mt-2\"><code>{{error.key}}.1.term.name</code></pre>\n <pre class=\"is-mt-2\"><code>{{error.key}}.1.value</code></pre>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.MaxSize\">\n <span>The maximum file size limit is {{ maxFileSizeMb }}Mb (current file size: {{ error.value }}Mb).</span>\n <span class=\"pl-1\">Please re-upload these data in multiple files.</span>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.MaxRows\">\n <span>\n Your file has {{ error.value | number }} rows. This might be an error caused by adding some data at the bottom\n of a spreadsheet by accident: delete any unused rows and resubmit the file.\n </span>\n </div>\n\n <div class=\"is-mb-2\" *ngIf=\"hasGeoJSONError\">\n <p>If the GeoJSON is too large, please try to simplify it following these steps:</p>\n <ul class=\"is-my-1 content is-list-style-decimal pl-5\">\n <li>\n <span class=\"is-pr-1\">Go to</span>\n <a href=\"https://mapshaper.org/\" target=\"_blank\">https://mapshaper.org/</a>\n </li>\n <li>\n <span>\n Save the boundary in a valid\n <code>.geojson</code>\n file.\n </span>\n </li>\n <li>\n <span>Import it on mapshaper.</span>\n </li>\n <li>\n <span>Click \"Simplify\" in the top right corner in the menu bar.</span>\n </li>\n <li>\n <span>Adjust the percentage to a lower value, like 50% to start with.</span>\n </li>\n <li>\n <span>\n Export to\n <code>GeoJson</code>\n or\n <code>CSV</code>\n .\n </span>\n </li>\n <li>\n <span>Copy the resulting content into your upload file.</span>\n </li>\n </ul>\n <p>If you encounter the error again, try again using a lower percentage at step 5.</p>\n </div>\n\n <div class=\"my-3\" *ngIf=\"columns.length || (error.index && !isSchemaError)\">\n <b>Hint:</b>\n <span class=\"pl-1\">the error might be located</span>\n <span class=\"pl-1\" *ngIf=\"error.index && !isSchemaError\">on row {{ error.index }}</span>\n <ng-container *ngIf=\"columns.length\">\n <span class=\"pl-1\">in {{ 'column' | pluralize: columns.length }}:</span>\n <ul class=\"mt-1 content is-list-style-disc pl-5\">\n <li *ngFor=\"let column of columns\">\n <b class=\"pr-2\">{{ column.column }}</b>\n <code>{{ column.name }}</code>\n </li>\n </ul>\n </ng-container>\n </div>\n\n <div class=\"my-3\" *ngIf=\"error.suggestions?.length\">\n <b>Suggestions:</b>\n <span class=\"pl-1\">did you mean instead:</span>\n <ul class=\"mt-1 content is-list-style-disc pl-5\">\n <li *ngFor=\"let suggestion of error.suggestions\">\n <code>{{ suggestion }}</code>\n </li>\n </ul>\n </div>\n\n <div *ngIf=\"showCsvPreview\" class=\"is-my-2\">\n <b>\n Preview\n <sup>1</sup>\n :\n </b>\n <div class=\"table-container is-mt-2\">\n <table class=\"table is-bordered is-fullwidth is-hoverable is-narrow mb-0\">\n <thead>\n <th *ngIf=\"error.index\"></th>\n <th *ngFor=\"let header of headers\">{{ header }}</th>\n </thead>\n <tbody>\n <ng-container *bindOnce=\"rows\">\n <tr *ngFor=\"let row of rows\">\n <td class=\"has-text-danger\" *ngIf=\"error.index\">\n <span class=\"is-nowrap\">Row {{ error.index }}</span>\n </td>\n <td *ngFor=\"let col of row; let colIndex = index\">\n <span\n [class.has-text-danger]=\"\n error?.message === ErrorKeys.DuplicatedIds && hasDuplicatedError(col, colIndex)\n \">\n {{ col | ellipsis: 50 }}\n </span>\n </td>\n </tr>\n </ng-container>\n </tbody>\n </table>\n </div>\n <p class=\"is-size-7\">\n <i>1. The preview does not necessarily reflect the original Upload's order or content.</i>\n </p>\n </div>\n\n <div *ngIf=\"showJsonPreview\" class=\"is-my-2\">\n <b>Preview:</b>\n <pre class=\"is-mt-2\"><code>{{stringify(error.node)}}</code></pre>\n </div>\n </div>\n</ng-container>\n\n<ng-template #defaultError>\n <p class=\"is-mt-2 has-text-black\">\n <span class=\"is-pr-1\">If you are still stuck, please see</span>\n <a [href]=\"baseUrl + '/upload#example'\" target=\"_blank\">the example uploads and videos</a>\n <span class=\"is-px-1\">\n or try using the Wizard to get the first few rows of data formatted correctly, or use the chat to ask for help.\n </span>\n </p>\n</ng-template>\n\n<ng-template #schemaError>\n <p>\n <span class=\"pr-1\">You can see the accepted field type and some examples</span>\n <a [href]=\"schemaKeyUrl\" target=\"_blank\">here</a>\n <span>.</span>\n </p>\n</ng-template>\n", styles: [":host{display:block}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i1.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { kind: "directive", type: i1.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { kind: "component", type: i2.FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "spin", "pulse", "mask", "styles", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "classes", "transform", "a11yRole"] }, { kind: "directive", type: i3.BindOnceDirective, selector: "[bindOnce]", inputs: ["bindOnce"] }, { kind: "pipe", type: i1.DecimalPipe, name: "number" }, { kind: "pipe", type: i4.EllipsisPipe, name: "ellipsis" }, { kind: "pipe", type: i5.PluralizePipe, name: "pluralize" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
132
128
|
}
|
|
133
129
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.8", ngImport: i0, type: FilesUploadErrorsComponent, decorators: [{
|
|
134
130
|
type: Component,
|
|
135
|
-
args: [{ selector: 'he-files-upload-errors', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"!!error\">\n <div class=\"has-text-danger\" [ngSwitch]=\"error?.message\">\n <div *ngSwitchCase=\"ErrorKeys.InvalidJSON\">\n <p>The uploaded content does not appear to contain valid JSON data:</p>\n\n <div class=\"is-mt-2\">\n <pre><code>{{error.error}}</code></pre>\n </div>\n\n <p>\n Please validate your JSON content before uploading it using an\n <a href=\"https://jsoneditoronline.org\" target=\"_blank\">online tool</a>\n for example.\n </p>\n </div>\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.NoData\">\n <p>No Nodes could be found in your Upload. Please verify the column headers match the Hestia schema.</p>\n\n <ng-container *ngTemplateOutlet=\"defaultError\"></ng-container>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.PropertyNotFound\">\n <div class=\"notification is-danger\" role=\"alert\">\n <p>\n <fa-icon class=\"pr-2\" icon=\"exclamation-triangle\"></fa-icon>\n <span>You have column headers that do not match the Hestia schema.</span>\n <span class=\"px-1\">Please refer to the</span>\n <a [href]=\"schemaUrl\" target=\"_blank\">schema</a>\n <span class=\"pl-1\">and then correct the column headers indicated below.</span>\n </p>\n\n <ng-container *ngTemplateOutlet=\"defaultError\"></ng-container>\n </div>\n\n <div class=\"is-mb-2\">\n <pre><code>{{error.key}}</code></pre>\n </div>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.PropertyInvalidFormat\">\n <div class=\"is-mb-2\">\n <span>\n The following value in the column\n <i>{{ error.key }}</i>\n is not in the correct format:\n </span>\n <br />\n <pre class=\"is-mt-2\"><code>{{stringify(error.value)}}</code></pre>\n </div>\n <ng-container [ngSwitch]=\"hasNumberWithCommasError\">\n <ng-container *ngSwitchCase=\"true\">\n <p class=\"is-mt-2\">\n Please format your numbers to remove commas and use a decimal point to separate the whole part of the number\n from the fractional part of the number.\n </p>\n\n <p class=\"is-mt-2 has-text-danger has-text-underline\">Incorrect:</p>\n <pre class=\"is-mt-2\"><code>3,510.1</code></pre>\n\n <p class=\"is-mt-2 has-text-success has-text-underline\">Correct:</p>\n <pre class=\"is-mt-2\"><code>3510.1</code></pre>\n </ng-container>\n\n <ng-container *ngSwitchCase=\"false\">\n <ng-container *ngTemplateOutlet=\"schemaError\"></ng-container>\n\n <p>The error could either be a value in the column, or an error with the column header.</p>\n <p>\n If you are trying to set no data, we accept either\n <code>-</code>\n or an empty cell.\n </p>\n </ng-container>\n </ng-container>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.PropertyRequired\">\n <div class=\"is-mb-2\">\n <span class=\"pr-1\">The following field is required in the</span>\n <a class=\"pr-1\" *ngIf=\"error.schema\" [href]=\"schemaUrl\" target=\"_blank\">{{ error.schema }}</a>\n <span>schema:</span>\n <br />\n <pre class=\"is-mt-2\"><code>{{error.property}}</code></pre>\n </div>\n <p [ngSwitch]=\"error.property\">\n <span *ngSwitchCase=\"'id'\">\n You must assign a unique\n <code>id</code>\n for every record in your data.\n </span>\n <span *ngSwitchCase=\"'type'\">\n Every record in your data must contain a\n <code>type</code>\n .\n </span>\n </p>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.PropertyInternal\">\n <div class=\"is-mb-2\">\n <span class=\"pr-1\">\n The following field is\n <code>internal</code>\n in the\n </span>\n <a [href]=\"schemaKeyUrl\" target=\"_blank\">{{ error.schema }} schema</a>\n <span>:</span>\n <br />\n <pre class=\"is-mt-2\"><code>{{error.schemaKey || error.key}}</code></pre>\n </div>\n <p>\n You cannot upload\n <code>internal</code>\n fields to Hestia.\n </p>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.SchemaNotFound\">\n <div class=\"is-mb-2\">\n <span>The following Node does not exist in the schema:</span>\n <br />\n <pre class=\"is-mt-2\"><code>{{error.key}}</code></pre>\n </div>\n <p>\n <span>The list of accepted values is:</span>\n </p>\n <div class=\"content\">\n <ul>\n <li *ngFor=\"let type of nodeTypes\">\n <i>{{ type }}</i>\n </li>\n </ul>\n </div>\n </div>\n\n <ng-container *ngSwitchCase=\"ErrorKeys.UploadsLimit\">\n <div class=\"is-mb-2\">\n <span>You have reached the upload limit of</span>\n <code class=\"mx-1\">{{ UploadLimit[error.schema] }}</code>\n <span>\n {{ error.schema }} in a single file (\n <b>{{ error.value }}</b>\n have been found).\n </span>\n </div>\n\n <p>\n <span>Please re-upload these data in multiple files with a maximum of</span>\n <code class=\"mx-1\">{{ UploadLimit[error.schema] }}</code>\n <span>{{ error.schema }} in each file.</span>\n </p>\n </ng-container>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.InvalidFirstColumn\">\n <span>\n No data in the Hestia format was detected on the first sheet. Either try selecting a different sheet in the\n upload screen, or check the sheet to ensure the headers follow the schema - see examples\n </span>\n <a class=\"pl-1\" routerLink=\"/upload\" fragment=\"example\">here</a>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.NoHeaders\">\n <p>\n No headers matching the Hestia schema were found on the first row. Please remove all empty rows at the begining\n of the file and try uploading again.\n </p>\n\n <ng-container *ngTemplateOutlet=\"defaultError\"></ng-container>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.InvalidSheetName\">\n <div class=\"is-mb-2\">\n <span class=\"pr-1\">The Excel sheet</span>\n <code>{{ error.value }}</code>\n <span class=\"pl-1\">does not exist.</span>\n <span class=\"pl-1\">Your Excel file contains the following sheets:</span>\n <ul class=\"mt-1 content is-list-style-disc pl-5\">\n <li *ngFor=\"let e of error.error.split(';')\">\n <code>{{ e }}</code>\n </li>\n </ul>\n </div>\n\n <p>\n <span>Please re-upload selecting one of the sheets above.</span>\n </p>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.DuplicatedHeaders\">\n <span>Your upload contains duplicated column headers.</span>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.DuplicatedIds\">\n <span class=\"pr-1\">You have multiple</span>\n <code>{{ error.schema | pluralize }}</code>\n <span class=\"px-1\">with the same</span>\n <a [href]=\"schemaKeyUrl\" target=\"_blank\">{{ error.schemaKey }}</a>\n <span class=\"px-1\">but different data. The</span>\n <a [href]=\"schemaKeyUrl\" target=\"_blank\">{{ error.schemaKey }}</a>\n <span class=\"pl-1\">is:</span>\n <br />\n <pre class=\"is-mt-2\"><code>{{error.value}}</code></pre>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.DuplicatedIdFields\">\n <span class=\"is-pr-1\">You have used both the</span>\n <code>@id</code>\n <span class=\"is-px-1\">and</span>\n <code>id</code>\n <span class=\"is-px-1\">fields in the same data. Please use</span>\n <code>@id</code>\n <span class=\"is-px-1\">when linking to existing nodes on the Platform, and</span>\n <code>id</code>\n <span class=\"is-px-1\">when linking to nodes you are uploading in this file using the same</span>\n <code>id</code>\n <span class=\"is-px-1\">field.</span>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.NestedHeaders\">\n <p>\n To link nodes together, you first need to define the node (e.g.,\n <code>source.id</code>\n ,\n <code>source.bibliography.title</code>\n ) and then create the link (e.g.,\n <code>cycle.defaultSource.id</code>\n ). We do not allow nesting (e.g.,\n <code>cycle.defaultSource.bibliography.title</code>\n ).\n </p>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.ReferenceExistingHeaders\">\n <p>\n You have used the internal\n <code>@id</code>\n field to upload a new Node.\n </p>\n <p>\n The\n <code>@id</code>\n field should only be used to reference existing Node or Terms. Example:\n </p>\n\n <p class=\"is-mt-2 has-text-danger has-text-underline\">Incorrect:</p>\n <pre class=\"is-mt-2\"><code>cycle.@id,cycle.inputs.0.term.@id,source.id</code></pre>\n\n <p class=\"is-mt-2 has-text-success has-text-underline\">Correct:</p>\n <pre class=\"is-mt-2\"><code>cycle.id,cycle.inputs.0.term.@id,source.id</code></pre>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.ObjectArrayInvalid\">\n <p>The values in the column {{ error.key }} must be represented as arrays.</p>\n\n <ng-container *ngTemplateOutlet=\"schemaError\"></ng-container>\n\n <p>Examples:</p>\n <pre class=\"is-mt-2\"><code>{{error.key}}.0.term.name</code></pre>\n <pre class=\"is-mt-2\"><code>{{error.key}}.0.value</code></pre>\n <pre class=\"is-mt-2\"><code>{{error.key}}.1.term.name</code></pre>\n <pre class=\"is-mt-2\"><code>{{error.key}}.1.value</code></pre>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.MaxSize\">\n <span>The maximum file size limit is {{ maxFileSizeMb }}Mb (current file size: {{ error.value }}Mb).</span>\n <span class=\"pl-1\">Please re-upload these data in multiple files.</span>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.MaxRows\">\n <span>\n Your file has {{ error.value | number }} rows. This might be an error caused by adding some data at the bottom\n of a spreadsheet by accident: delete any unused rows and resubmit the file.\n </span>\n </div>\n\n <div class=\"is-mb-2\" *ngIf=\"hasGeoJSONError\">\n <p>If the GeoJSON is too large, please try to simplify it following these steps:</p>\n <ul class=\"is-my-1 content is-list-style-decimal pl-5\">\n <li>\n <span class=\"is-pr-1\">Go to</span>\n <a href=\"https://mapshaper.org/\" target=\"_blank\">https://mapshaper.org/</a>\n </li>\n <li>\n <span>\n Save the boundary in a valid\n <code>.geojson</code>\n file.\n </span>\n </li>\n <li>\n <span>Import it on mapshaper.</span>\n </li>\n <li>\n <span>Click \"Simplify\" in the top right corner in the menu bar.</span>\n </li>\n <li>\n <span>Adjust the percentage to a lower value, like 50% to start with.</span>\n </li>\n <li>\n <span>\n Export to\n <code>GeoJson</code>\n or\n <code>CSV</code>\n .\n </span>\n </li>\n <li>\n <span>Copy the resulting content into your upload file.</span>\n </li>\n </ul>\n <p>If you encounter the error again, try again using a lower percentage at step 5.</p>\n </div>\n\n <div class=\"my-3\" *ngIf=\"columns.length || (error.index && !isSchemaError)\">\n <b>Hint:</b>\n <span class=\"pl-1\">the error might be located</span>\n <span class=\"pl-1\" *ngIf=\"error.index && !isSchemaError\">on row {{ error.index }}</span>\n <ng-container *ngIf=\"columns.length\">\n <span class=\"pl-1\">in {{ 'column' | pluralize: columns.length }}:</span>\n <ul class=\"mt-1 content is-list-style-disc pl-5\">\n <li *ngFor=\"let column of columns\">\n <b class=\"pr-2\">{{ column.column }}</b>\n <code>{{ column.name }}</code>\n </li>\n </ul>\n </ng-container>\n </div>\n\n <div class=\"my-3\" *ngIf=\"error.suggestions?.length\">\n <b>Suggestions:</b>\n <span class=\"pl-1\">did you mean instead:</span>\n <ul class=\"mt-1 content is-list-style-disc pl-5\">\n <li *ngFor=\"let suggestion of error.suggestions\">\n <code>{{ suggestion }}</code>\n </li>\n </ul>\n </div>\n\n <div *ngIf=\"showCsvPreview\" class=\"is-my-2\">\n <b>\n Preview\n <sup>1</sup>\n :\n </b>\n <div class=\"table-container is-mt-2\">\n <table class=\"table is-bordered is-fullwidth is-hoverable is-narrow mb-0\">\n <thead>\n <th *ngIf=\"error.index\"></th>\n <th *ngFor=\"let header of headers\">{{ header }}</th>\n </thead>\n <tbody>\n <ng-container *bindOnce=\"rows\">\n <tr *ngFor=\"let row of rows\">\n <td class=\"has-text-danger\" *ngIf=\"error.index\">\n <span class=\"is-nowrap\">Row {{ error.index }}</span>\n </td>\n <td *ngFor=\"let col of row; let colIndex = index\">\n <span\n [class.has-text-danger]=\"\n error?.message === ErrorKeys.DuplicatedIds && hasDuplicatedError(col, colIndex)\n \">\n {{ col | ellipsis: 50 }}\n </span>\n </td>\n </tr>\n </ng-container>\n </tbody>\n </table>\n </div>\n <p class=\"is-size-7\">\n <i>1. The preview does not necessarily reflect the original Upload's order or content.</i>\n </p>\n </div>\n\n <div *ngIf=\"showJsonPreview\" class=\"is-my-2\">\n <b>Preview:</b>\n <pre class=\"is-mt-2\"><code>{{stringify(error.node)}}</code></pre>\n </div>\n </div>\n</ng-container>\n\n<ng-template #defaultError>\n <p class=\"is-mt-2 has-text-black\">\n <span class=\"is-pr-1\">If you are still stuck, please see</span>\n <a [href]=\"baseUrl + '/upload#example'\" target=\"_blank\">the example uploads and videos</a>\n <span class=\"is-px-1\">\n or try using the Wizard to get the first few rows of data formatted correctly, or use the chat to ask for help.\n </span>\n </p>\n</ng-template>\n\n<ng-template #schemaError>\n <p>\n <span class=\"pr-1\">You can see the accepted field type and some examples</span>\n <a [href]=\"schemaKeyUrl\" target=\"_blank\">here</a>\n <span>.</span>\n </p>\n</ng-template>\n", styles: [":host{display:block}\n"] }]
|
|
131
|
+
args: [{ selector: 'he-files-upload-errors', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"!!error\">\n <div class=\"has-text-danger\" [ngSwitch]=\"error?.message\">\n <div *ngSwitchCase=\"ErrorKeys.InvalidJSON\">\n <p>The uploaded content does not appear to contain valid JSON data:</p>\n\n <div class=\"is-mt-2\">\n <pre><code>{{error.error}}</code></pre>\n </div>\n\n <p>\n Please validate your JSON content before uploading it using an\n <a href=\"https://jsoneditoronline.org\" target=\"_blank\">online tool</a>\n for example.\n </p>\n </div>\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.NoData\">\n <p>No Nodes could be found in your Upload. Please verify the column headers match the Hestia schema.</p>\n\n <ng-container *ngTemplateOutlet=\"defaultError\"></ng-container>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"SchemaErrorKeys.PropertyNotFound\">\n <div class=\"notification is-danger\" role=\"alert\">\n <p>\n <fa-icon class=\"pr-2\" icon=\"exclamation-triangle\"></fa-icon>\n <span>You have column headers that do not match the Hestia schema.</span>\n <span class=\"px-1\">Please refer to the</span>\n <a [href]=\"schemaUrl\" target=\"_blank\">schema</a>\n <span class=\"pl-1\">and then correct the column headers indicated below.</span>\n </p>\n\n <ng-container *ngTemplateOutlet=\"defaultError\"></ng-container>\n </div>\n\n <div class=\"is-mb-2\">\n <pre><code>{{error.key}}</code></pre>\n </div>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"SchemaErrorKeys.PropertyInvalidFormat\">\n <div class=\"is-mb-2\">\n <span>\n The following value in the column\n <i>{{ error.key }}</i>\n is not in the correct format:\n </span>\n <br />\n <pre class=\"is-mt-2\"><code>{{stringify(error.value)}}</code></pre>\n </div>\n <ng-container [ngSwitch]=\"hasNumberWithCommasError\">\n <ng-container *ngSwitchCase=\"true\">\n <p class=\"is-mt-2\">\n Please format your numbers to remove commas and use a decimal point to separate the whole part of the number\n from the fractional part of the number.\n </p>\n\n <p class=\"is-mt-2 has-text-danger has-text-underline\">Incorrect:</p>\n <pre class=\"is-mt-2\"><code>3,510.1</code></pre>\n\n <p class=\"is-mt-2 has-text-success has-text-underline\">Correct:</p>\n <pre class=\"is-mt-2\"><code>3510.1</code></pre>\n </ng-container>\n\n <ng-container *ngSwitchCase=\"false\">\n <ng-container *ngTemplateOutlet=\"schemaError\"></ng-container>\n\n <p>The error could either be a value in the column, or an error with the column header.</p>\n <p>\n If you are trying to set no data, we accept either\n <code>-</code>\n or an empty cell.\n </p>\n </ng-container>\n </ng-container>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.PropertyRequired\">\n <div class=\"is-mb-2\">\n <span class=\"pr-1\">The following field is required in the</span>\n <a class=\"pr-1\" *ngIf=\"error.schema\" [href]=\"schemaUrl\" target=\"_blank\">{{ error.schema }}</a>\n <span>schema:</span>\n <br />\n <pre class=\"is-mt-2\"><code>{{error.property}}</code></pre>\n </div>\n <p [ngSwitch]=\"error.property\">\n <span *ngSwitchCase=\"'id'\">\n You must assign a unique\n <code>id</code>\n for every record in your data.\n </span>\n <span *ngSwitchCase=\"'type'\">\n Every record in your data must contain a\n <code>type</code>\n .\n </span>\n </p>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.PropertyInternal\">\n <div class=\"is-mb-2\">\n <span class=\"pr-1\">\n The following field is\n <code>internal</code>\n in the\n </span>\n <a [href]=\"schemaKeyUrl\" target=\"_blank\">{{ error.schema }} schema</a>\n <span>:</span>\n <br />\n <pre class=\"is-mt-2\"><code>{{error.schemaKey || error.key}}</code></pre>\n </div>\n <p>\n You cannot upload\n <code>internal</code>\n fields to Hestia.\n </p>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"SchemaErrorKeys.SchemaNotFound\">\n <div class=\"is-mb-2\">\n <span>The following Node does not exist in the schema:</span>\n <br />\n <pre class=\"is-mt-2\"><code>{{error.key}}</code></pre>\n </div>\n <p>\n <span>The list of accepted values is:</span>\n </p>\n <div class=\"content\">\n <ul>\n <li *ngFor=\"let type of nodeTypes\">\n <i>{{ type }}</i>\n </li>\n </ul>\n </div>\n </div>\n\n <ng-container *ngSwitchCase=\"ErrorKeys.UploadsLimit\">\n <div class=\"is-mb-2\">\n <span>You have reached the upload limit of</span>\n <code class=\"mx-1\">{{ UploadLimit[error.schema] }}</code>\n <span>\n {{ error.schema }} in a single file (\n <b>{{ error.value }}</b>\n have been found).\n </span>\n </div>\n\n <p>\n <span>Please re-upload these data in multiple files with a maximum of</span>\n <code class=\"mx-1\">{{ UploadLimit[error.schema] }}</code>\n <span>{{ error.schema }} in each file.</span>\n </p>\n </ng-container>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.InvalidFirstColumn\">\n <span>\n No data in the Hestia format was detected on the first sheet. Either try selecting a different sheet in the\n upload screen, or check the sheet to ensure the headers follow the schema - see examples\n </span>\n <a class=\"pl-1\" routerLink=\"/upload\" fragment=\"example\">here</a>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.NoHeaders\">\n <p>\n No headers matching the Hestia schema were found on the first row. Please remove all empty rows at the begining\n of the file and try uploading again.\n </p>\n\n <ng-container *ngTemplateOutlet=\"defaultError\"></ng-container>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.InvalidSheetName\">\n <div class=\"is-mb-2\">\n <span class=\"pr-1\">The Excel sheet</span>\n <code>{{ error.value }}</code>\n <span class=\"pl-1\">does not exist.</span>\n <span class=\"pl-1\">Your Excel file contains the following sheets:</span>\n <ul class=\"mt-1 content is-list-style-disc pl-5\">\n <li *ngFor=\"let e of error.error.split(';')\">\n <code>{{ e }}</code>\n </li>\n </ul>\n </div>\n\n <p>\n <span>Please re-upload selecting one of the sheets above.</span>\n </p>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.DuplicatedHeaders\">\n <span>Your upload contains duplicated column headers.</span>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.DuplicatedIds\">\n <span class=\"pr-1\">You have multiple</span>\n <code>{{ error.schema | pluralize }}</code>\n <span class=\"px-1\">with the same</span>\n <a [href]=\"schemaKeyUrl\" target=\"_blank\">{{ error.schemaKey }}</a>\n <span class=\"px-1\">but different data. The</span>\n <a [href]=\"schemaKeyUrl\" target=\"_blank\">{{ error.schemaKey }}</a>\n <span class=\"pl-1\">is:</span>\n <br />\n <pre class=\"is-mt-2\"><code>{{error.value}}</code></pre>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"SchemaErrorKeys.DuplicatedIdFields\">\n <span class=\"is-pr-1\">You have used both the</span>\n <code>@id</code>\n <span class=\"is-px-1\">and</span>\n <code>id</code>\n <span class=\"is-px-1\">fields in the same data. Please use</span>\n <code>@id</code>\n <span class=\"is-px-1\">when linking to existing nodes on the Platform, and</span>\n <code>id</code>\n <span class=\"is-px-1\">when linking to nodes you are uploading in this file using the same</span>\n <code>id</code>\n <span class=\"is-px-1\">field.</span>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.NestedHeaders\">\n <p>\n To link nodes together, you first need to define the node (e.g.,\n <code>source.id</code>\n ,\n <code>source.bibliography.title</code>\n ) and then create the link (e.g.,\n <code>cycle.defaultSource.id</code>\n ). We do not allow nesting (e.g.,\n <code>cycle.defaultSource.bibliography.title</code>\n ).\n </p>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.ReferenceExistingHeaders\">\n <p>\n You have used the internal\n <code>@id</code>\n field to upload a new Node.\n </p>\n <p>\n The\n <code>@id</code>\n field should only be used to reference existing Node or Terms. Example:\n </p>\n\n <p class=\"is-mt-2 has-text-danger has-text-underline\">Incorrect:</p>\n <pre class=\"is-mt-2\"><code>cycle.@id,cycle.inputs.0.term.@id,source.id</code></pre>\n\n <p class=\"is-mt-2 has-text-success has-text-underline\">Correct:</p>\n <pre class=\"is-mt-2\"><code>cycle.id,cycle.inputs.0.term.@id,source.id</code></pre>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"SchemaErrorKeys.ObjectArrayInvalid\">\n <p>The values in the column {{ error.key }} must be represented as arrays.</p>\n\n <ng-container *ngTemplateOutlet=\"schemaError\"></ng-container>\n\n <p>Examples:</p>\n <pre class=\"is-mt-2\"><code>{{error.key}}.0.term.name</code></pre>\n <pre class=\"is-mt-2\"><code>{{error.key}}.0.value</code></pre>\n <pre class=\"is-mt-2\"><code>{{error.key}}.1.term.name</code></pre>\n <pre class=\"is-mt-2\"><code>{{error.key}}.1.value</code></pre>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.MaxSize\">\n <span>The maximum file size limit is {{ maxFileSizeMb }}Mb (current file size: {{ error.value }}Mb).</span>\n <span class=\"pl-1\">Please re-upload these data in multiple files.</span>\n </div>\n\n <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.MaxRows\">\n <span>\n Your file has {{ error.value | number }} rows. This might be an error caused by adding some data at the bottom\n of a spreadsheet by accident: delete any unused rows and resubmit the file.\n </span>\n </div>\n\n <div class=\"is-mb-2\" *ngIf=\"hasGeoJSONError\">\n <p>If the GeoJSON is too large, please try to simplify it following these steps:</p>\n <ul class=\"is-my-1 content is-list-style-decimal pl-5\">\n <li>\n <span class=\"is-pr-1\">Go to</span>\n <a href=\"https://mapshaper.org/\" target=\"_blank\">https://mapshaper.org/</a>\n </li>\n <li>\n <span>\n Save the boundary in a valid\n <code>.geojson</code>\n file.\n </span>\n </li>\n <li>\n <span>Import it on mapshaper.</span>\n </li>\n <li>\n <span>Click \"Simplify\" in the top right corner in the menu bar.</span>\n </li>\n <li>\n <span>Adjust the percentage to a lower value, like 50% to start with.</span>\n </li>\n <li>\n <span>\n Export to\n <code>GeoJson</code>\n or\n <code>CSV</code>\n .\n </span>\n </li>\n <li>\n <span>Copy the resulting content into your upload file.</span>\n </li>\n </ul>\n <p>If you encounter the error again, try again using a lower percentage at step 5.</p>\n </div>\n\n <div class=\"my-3\" *ngIf=\"columns.length || (error.index && !isSchemaError)\">\n <b>Hint:</b>\n <span class=\"pl-1\">the error might be located</span>\n <span class=\"pl-1\" *ngIf=\"error.index && !isSchemaError\">on row {{ error.index }}</span>\n <ng-container *ngIf=\"columns.length\">\n <span class=\"pl-1\">in {{ 'column' | pluralize: columns.length }}:</span>\n <ul class=\"mt-1 content is-list-style-disc pl-5\">\n <li *ngFor=\"let column of columns\">\n <b class=\"pr-2\">{{ column.column }}</b>\n <code>{{ column.name }}</code>\n </li>\n </ul>\n </ng-container>\n </div>\n\n <div class=\"my-3\" *ngIf=\"error.suggestions?.length\">\n <b>Suggestions:</b>\n <span class=\"pl-1\">did you mean instead:</span>\n <ul class=\"mt-1 content is-list-style-disc pl-5\">\n <li *ngFor=\"let suggestion of error.suggestions\">\n <code>{{ suggestion }}</code>\n </li>\n </ul>\n </div>\n\n <div *ngIf=\"showCsvPreview\" class=\"is-my-2\">\n <b>\n Preview\n <sup>1</sup>\n :\n </b>\n <div class=\"table-container is-mt-2\">\n <table class=\"table is-bordered is-fullwidth is-hoverable is-narrow mb-0\">\n <thead>\n <th *ngIf=\"error.index\"></th>\n <th *ngFor=\"let header of headers\">{{ header }}</th>\n </thead>\n <tbody>\n <ng-container *bindOnce=\"rows\">\n <tr *ngFor=\"let row of rows\">\n <td class=\"has-text-danger\" *ngIf=\"error.index\">\n <span class=\"is-nowrap\">Row {{ error.index }}</span>\n </td>\n <td *ngFor=\"let col of row; let colIndex = index\">\n <span\n [class.has-text-danger]=\"\n error?.message === ErrorKeys.DuplicatedIds && hasDuplicatedError(col, colIndex)\n \">\n {{ col | ellipsis: 50 }}\n </span>\n </td>\n </tr>\n </ng-container>\n </tbody>\n </table>\n </div>\n <p class=\"is-size-7\">\n <i>1. The preview does not necessarily reflect the original Upload's order or content.</i>\n </p>\n </div>\n\n <div *ngIf=\"showJsonPreview\" class=\"is-my-2\">\n <b>Preview:</b>\n <pre class=\"is-mt-2\"><code>{{stringify(error.node)}}</code></pre>\n </div>\n </div>\n</ng-container>\n\n<ng-template #defaultError>\n <p class=\"is-mt-2 has-text-black\">\n <span class=\"is-pr-1\">If you are still stuck, please see</span>\n <a [href]=\"baseUrl + '/upload#example'\" target=\"_blank\">the example uploads and videos</a>\n <span class=\"is-px-1\">\n or try using the Wizard to get the first few rows of data formatted correctly, or use the chat to ask for help.\n </span>\n </p>\n</ng-template>\n\n<ng-template #schemaError>\n <p>\n <span class=\"pr-1\">You can see the accepted field type and some examples</span>\n <a [href]=\"schemaKeyUrl\" target=\"_blank\">here</a>\n <span>.</span>\n </p>\n</ng-template>\n", styles: [":host{display:block}\n"] }]
|
|
136
132
|
}], propDecorators: { file: [{
|
|
137
133
|
type: Input
|
|
138
134
|
}], error: [{
|
|
139
135
|
type: Input
|
|
140
136
|
}] } });
|
|
141
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"files-upload-errors.component.js","sourceRoot":"","sources":["../../../../src/files/files-upload-errors/files-upload-errors.component.ts","../../../../src/files/files-upload-errors/files-upload-errors.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, Input, OnInit } from '@angular/core';\nimport { parse } from 'papaparse';\nimport {\n  SchemaType,\n  NodeType,\n  JSON as HestiaJson,\n  UploadLimit,\n  isExpandable,\n  typeToSchemaType\n} from '@hestia-earth/schema';\nimport { File, maxFileSizeMb, SupportedExtensions } from '@hestia-earth/api';\nimport { toCsv } from '@hestia-earth/schema-convert';\n\nimport { baseUrl } from '../../common';\nimport { schemaBaseUrl } from '../../schema/schema.service';\n\nconst nodeTypeToString = (type: NodeType) => `${type.charAt(0).toLowerCase()}${type.substring(1)}`;\n\nconst acceptedTypes = Object.values(NodeType)\n  .filter(val => val !== NodeType.Term)\n  .map(nodeTypeToString);\n\nexport enum ErrorKeys {\n  NoData = 'no-data',\n  NoHeaders = 'no-headers',\n  InvalidJSON = 'invalid-json',\n  InvalidSheetName = 'invalid-sheet-name',\n  InvalidFirstColumn = 'invalid-first-column',\n  DuplicatedHeaders = 'duplicated-headers',\n  DuplicatedIds = 'duplicated-ids',\n  DuplicatedIdFields = 'duplicated-id-fields',\n  PropertyInvalidFormat = 'property-invalid-format',\n  PropertyNotFound = 'property-not-found',\n  PropertyRequired = 'property-required',\n  PropertyInternal = 'property-internal',\n  SchemaNotFound = 'schema-not-found',\n  UploadsLimit = 'upload-limit',\n  NestedHeaders = 'nested-headers',\n  ReferenceExistingHeaders = 'reference-existing-headers',\n  ObjectArrayInvalid = 'object-array-invalid',\n  MaxSize = 'max-size',\n  MaxRows = 'max-rows'\n}\n\ninterface ICSVHeader {\n  name: string;\n  index: number;\n}\n\ninterface ICSVColumn extends ICSVHeader {\n  column: string;\n}\n\nexport interface IError {\n  message: ErrorKeys;\n  schema?: SchemaType;\n  schemaKey?: string;\n  key?: string;\n  property?: string;\n  value?: any;\n  min?: any;\n  max?: any;\n  error?: string;\n  node?: HestiaJson<SchemaType>;\n  index?: number;\n  headers?: ICSVHeader[];\n  suggestions?: string[];\n}\n\nconst valueToNodes = (value: any, type: SchemaType) =>\n  Array.isArray(value)\n    ? isExpandable(value)\n      ? value.map(val => ({\n          type,\n          ...val\n        }))\n      : null\n    : typeof value === 'object'\n    ? [\n        {\n          type,\n          ...value\n        }\n      ]\n    : null;\n\nconst nodesFromError = (error?: IError, type?: SchemaType) =>\n  error ? valueToNodes(error.node, type) || valueToNodes(error.value, error.schema) || [] : [];\n\nconst errorCsv = nodes => {\n  try {\n    return toCsv(nodes, { includeExising: true });\n  } catch (_err) {\n    return '';\n  }\n};\n\nconst numberToCol = (num: number) => {\n  let total = num + 1; // starts at 0\n  let str = '';\n  while (total > 0) {\n    const q = (total - 1) / 26;\n    const r = (total - 1) % 26;\n    total = Math.floor(q);\n    str = `${String.fromCharCode(65 + r)}${str}`;\n  }\n  return str;\n};\n\n@Component({\n  selector: 'he-files-upload-errors',\n  templateUrl: './files-upload-errors.component.html',\n  styleUrls: ['./files-upload-errors.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class FilesUploadErrorsComponent implements OnInit {\n  @Input()\n  private file?: Pick<File, 'filename' | 'schemaVersion'>;\n  @Input()\n  public error: IError;\n\n  public baseUrl = baseUrl();\n  public nodeTypes = acceptedTypes;\n  public ErrorKeys = ErrorKeys;\n  public UploadLimit = UploadLimit;\n  public maxFileSizeMb = maxFileSizeMb;\n  public columns: ICSVColumn[] = [];\n  public headers: string[] = [];\n  public rows: string[][] = [];\n\n  ngOnInit() {\n    this.columns = (this.error?.headers || []).map(header => ({\n      ...header,\n      column: numberToCol(header.index)\n    }));\n    const schemaType = this.error?.key ? typeToSchemaType(this.error.key.split('.')[0]) : null;\n    const nodeType = Object.values(SchemaType).includes(schemaType) ? schemaType : this.error?.schema;\n    const nodes = nodesFromError(this.error, nodeType);\n    const csv = errorCsv(nodes).trim();\n    const {\n      data: [headers, ...rows]\n    } = csv ? parse(csv) : { data: [[]] };\n    this.headers = headers;\n    this.rows = rows;\n  }\n\n  private get isJSONFile() {\n    return this.file?.filename?.endsWith(SupportedExtensions.json);\n  }\n\n  public stringify(value) {\n    return JSON.stringify(value, null, 2);\n  }\n\n  public hasDuplicatedError(value: string, colIndex: number) {\n    const firstVal = this.rows[0][colIndex];\n    return value !== firstVal;\n  }\n\n  public get hasNumberWithCommasError() {\n    return this.error?.error === 'failed to parse number' && (`${this.error?.value}` || '').includes(',');\n  }\n\n  public get hasGeoJSONError() {\n    return this.error?.error?.includes('Unable to parse GeoJSON value');\n  }\n\n  public get schemaUrl() {\n    return [schemaBaseUrl(this.file?.schemaVersion), this.error?.schema].filter(Boolean).join('/');\n  }\n\n  public get schemaKeyUrl() {\n    return [this.schemaUrl, this.error?.schemaKey || this.error?.key].filter(Boolean).join('#');\n  }\n\n  public get isSchemaError() {\n    return [ErrorKeys.PropertyNotFound, ErrorKeys.SchemaNotFound].includes(this.error?.message);\n  }\n\n  public get showCsvPreview() {\n    return this.headers?.length && this.rows?.length && !this.isSchemaError && !this.isJSONFile;\n  }\n\n  public get showJsonPreview() {\n    return this.error?.node && this.isJSONFile;\n  }\n}\n","<ng-container *ngIf=\"!!error\">\n  <div class=\"has-text-danger\" [ngSwitch]=\"error?.message\">\n    <div *ngSwitchCase=\"ErrorKeys.InvalidJSON\">\n      <p>The uploaded content does not appear to contain valid JSON data:</p>\n\n      <div class=\"is-mt-2\">\n        <pre><code>{{error.error}}</code></pre>\n      </div>\n\n      <p>\n        Please validate your JSON content before uploading it using an\n        <a href=\"https://jsoneditoronline.org\" target=\"_blank\">online tool</a>\n        for example.\n      </p>\n    </div>\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.NoData\">\n      <p>No Nodes could be found in your Upload. Please verify the column headers match the Hestia schema.</p>\n\n      <ng-container *ngTemplateOutlet=\"defaultError\"></ng-container>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.PropertyNotFound\">\n      <div class=\"notification is-danger\" role=\"alert\">\n        <p>\n          <fa-icon class=\"pr-2\" icon=\"exclamation-triangle\"></fa-icon>\n          <span>You have column headers that do not match the Hestia schema.</span>\n          <span class=\"px-1\">Please refer to the</span>\n          <a [href]=\"schemaUrl\" target=\"_blank\">schema</a>\n          <span class=\"pl-1\">and then correct the column headers indicated below.</span>\n        </p>\n\n        <ng-container *ngTemplateOutlet=\"defaultError\"></ng-container>\n      </div>\n\n      <div class=\"is-mb-2\">\n        <pre><code>{{error.key}}</code></pre>\n      </div>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.PropertyInvalidFormat\">\n      <div class=\"is-mb-2\">\n        <span>\n          The following value in the column\n          <i>{{ error.key }}</i>\n          is not in the correct format:\n        </span>\n        <br />\n        <pre class=\"is-mt-2\"><code>{{stringify(error.value)}}</code></pre>\n      </div>\n      <ng-container [ngSwitch]=\"hasNumberWithCommasError\">\n        <ng-container *ngSwitchCase=\"true\">\n          <p class=\"is-mt-2\">\n            Please format your numbers to remove commas and use a decimal point to separate the whole part of the number\n            from the fractional part of the number.\n          </p>\n\n          <p class=\"is-mt-2 has-text-danger has-text-underline\">Incorrect:</p>\n          <pre class=\"is-mt-2\"><code>3,510.1</code></pre>\n\n          <p class=\"is-mt-2 has-text-success has-text-underline\">Correct:</p>\n          <pre class=\"is-mt-2\"><code>3510.1</code></pre>\n        </ng-container>\n\n        <ng-container *ngSwitchCase=\"false\">\n          <ng-container *ngTemplateOutlet=\"schemaError\"></ng-container>\n\n          <p>The error could either be a value in the column, or an error with the column header.</p>\n          <p>\n            If you are trying to set no data, we accept either\n            <code>-</code>\n            or an empty cell.\n          </p>\n        </ng-container>\n      </ng-container>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.PropertyRequired\">\n      <div class=\"is-mb-2\">\n        <span class=\"pr-1\">The following field is required in the</span>\n        <a class=\"pr-1\" *ngIf=\"error.schema\" [href]=\"schemaUrl\" target=\"_blank\">{{ error.schema }}</a>\n        <span>schema:</span>\n        <br />\n        <pre class=\"is-mt-2\"><code>{{error.property}}</code></pre>\n      </div>\n      <p [ngSwitch]=\"error.property\">\n        <span *ngSwitchCase=\"'id'\">\n          You must assign a unique\n          <code>id</code>\n          for every record in your data.\n        </span>\n        <span *ngSwitchCase=\"'type'\">\n          Every record in your data must contain a\n          <code>type</code>\n          .\n        </span>\n      </p>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.PropertyInternal\">\n      <div class=\"is-mb-2\">\n        <span class=\"pr-1\">\n          The following field is\n          <code>internal</code>\n          in the\n        </span>\n        <a [href]=\"schemaKeyUrl\" target=\"_blank\">{{ error.schema }} schema</a>\n        <span>:</span>\n        <br />\n        <pre class=\"is-mt-2\"><code>{{error.schemaKey || error.key}}</code></pre>\n      </div>\n      <p>\n        You cannot upload\n        <code>internal</code>\n        fields to Hestia.\n      </p>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.SchemaNotFound\">\n      <div class=\"is-mb-2\">\n        <span>The following Node does not exist in the schema:</span>\n        <br />\n        <pre class=\"is-mt-2\"><code>{{error.key}}</code></pre>\n      </div>\n      <p>\n        <span>The list of accepted values is:</span>\n      </p>\n      <div class=\"content\">\n        <ul>\n          <li *ngFor=\"let type of nodeTypes\">\n            <i>{{ type }}</i>\n          </li>\n        </ul>\n      </div>\n    </div>\n\n    <ng-container *ngSwitchCase=\"ErrorKeys.UploadsLimit\">\n      <div class=\"is-mb-2\">\n        <span>You have reached the upload limit of</span>\n        <code class=\"mx-1\">{{ UploadLimit[error.schema] }}</code>\n        <span>\n          {{ error.schema }} in a single file (\n          <b>{{ error.value }}</b>\n          have been found).\n        </span>\n      </div>\n\n      <p>\n        <span>Please re-upload these data in multiple files with a maximum of</span>\n        <code class=\"mx-1\">{{ UploadLimit[error.schema] }}</code>\n        <span>{{ error.schema }} in each file.</span>\n      </p>\n    </ng-container>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.InvalidFirstColumn\">\n      <span>\n        No data in the Hestia format was detected on the first sheet. Either try selecting a different sheet in the\n        upload screen, or check the sheet to ensure the headers follow the schema - see examples\n      </span>\n      <a class=\"pl-1\" routerLink=\"/upload\" fragment=\"example\">here</a>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.NoHeaders\">\n      <p>\n        No headers matching the Hestia schema were found on the first row. Please remove all empty rows at the begining\n        of the file and try uploading again.\n      </p>\n\n      <ng-container *ngTemplateOutlet=\"defaultError\"></ng-container>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.InvalidSheetName\">\n      <div class=\"is-mb-2\">\n        <span class=\"pr-1\">The Excel sheet</span>\n        <code>{{ error.value }}</code>\n        <span class=\"pl-1\">does not exist.</span>\n        <span class=\"pl-1\">Your Excel file contains the following sheets:</span>\n        <ul class=\"mt-1 content is-list-style-disc pl-5\">\n          <li *ngFor=\"let e of error.error.split(';')\">\n            <code>{{ e }}</code>\n          </li>\n        </ul>\n      </div>\n\n      <p>\n        <span>Please re-upload selecting one of the sheets above.</span>\n      </p>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.DuplicatedHeaders\">\n      <span>Your upload contains duplicated column headers.</span>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.DuplicatedIds\">\n      <span class=\"pr-1\">You have multiple</span>\n      <code>{{ error.schema | pluralize }}</code>\n      <span class=\"px-1\">with the same</span>\n      <a [href]=\"schemaKeyUrl\" target=\"_blank\">{{ error.schemaKey }}</a>\n      <span class=\"px-1\">but different data. The</span>\n      <a [href]=\"schemaKeyUrl\" target=\"_blank\">{{ error.schemaKey }}</a>\n      <span class=\"pl-1\">is:</span>\n      <br />\n      <pre class=\"is-mt-2\"><code>{{error.value}}</code></pre>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.DuplicatedIdFields\">\n      <span class=\"is-pr-1\">You have used both the</span>\n      <code>@id</code>\n      <span class=\"is-px-1\">and</span>\n      <code>id</code>\n      <span class=\"is-px-1\">fields in the same data. Please use</span>\n      <code>@id</code>\n      <span class=\"is-px-1\">when linking to existing nodes on the Platform, and</span>\n      <code>id</code>\n      <span class=\"is-px-1\">when linking to nodes you are uploading in this file using the same</span>\n      <code>id</code>\n      <span class=\"is-px-1\">field.</span>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.NestedHeaders\">\n      <p>\n        To link nodes together, you first need to define the node (e.g.,\n        <code>source.id</code>\n        ,\n        <code>source.bibliography.title</code>\n        ) and then create the link (e.g.,\n        <code>cycle.defaultSource.id</code>\n        ). We do not allow nesting (e.g.,\n        <code>cycle.defaultSource.bibliography.title</code>\n        ).\n      </p>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.ReferenceExistingHeaders\">\n      <p>\n        You have used the internal\n        <code>@id</code>\n        field to upload a new Node.\n      </p>\n      <p>\n        The\n        <code>@id</code>\n        field should only be used to reference existing Node or Terms. Example:\n      </p>\n\n      <p class=\"is-mt-2 has-text-danger has-text-underline\">Incorrect:</p>\n      <pre class=\"is-mt-2\"><code>cycle.@id,cycle.inputs.0.term.@id,source.id</code></pre>\n\n      <p class=\"is-mt-2 has-text-success has-text-underline\">Correct:</p>\n      <pre class=\"is-mt-2\"><code>cycle.id,cycle.inputs.0.term.@id,source.id</code></pre>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.ObjectArrayInvalid\">\n      <p>The values in the column {{ error.key }} must be represented as arrays.</p>\n\n      <ng-container *ngTemplateOutlet=\"schemaError\"></ng-container>\n\n      <p>Examples:</p>\n      <pre class=\"is-mt-2\"><code>{{error.key}}.0.term.name</code></pre>\n      <pre class=\"is-mt-2\"><code>{{error.key}}.0.value</code></pre>\n      <pre class=\"is-mt-2\"><code>{{error.key}}.1.term.name</code></pre>\n      <pre class=\"is-mt-2\"><code>{{error.key}}.1.value</code></pre>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.MaxSize\">\n      <span>The maximum file size limit is {{ maxFileSizeMb }}Mb (current file size: {{ error.value }}Mb).</span>\n      <span class=\"pl-1\">Please re-upload these data in multiple files.</span>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.MaxRows\">\n      <span>\n        Your file has {{ error.value | number }} rows. This might be an error caused by adding some data at the bottom\n        of a spreadsheet by accident: delete any unused rows and resubmit the file.\n      </span>\n    </div>\n\n    <div class=\"is-mb-2\" *ngIf=\"hasGeoJSONError\">\n      <p>If the GeoJSON is too large, please try to simplify it following these steps:</p>\n      <ul class=\"is-my-1 content is-list-style-decimal pl-5\">\n        <li>\n          <span class=\"is-pr-1\">Go to</span>\n          <a href=\"https://mapshaper.org/\" target=\"_blank\">https://mapshaper.org/</a>\n        </li>\n        <li>\n          <span>\n            Save the boundary in a valid\n            <code>.geojson</code>\n            file.\n          </span>\n        </li>\n        <li>\n          <span>Import it on mapshaper.</span>\n        </li>\n        <li>\n          <span>Click \"Simplify\" in the top right corner in the menu bar.</span>\n        </li>\n        <li>\n          <span>Adjust the percentage to a lower value, like 50% to start with.</span>\n        </li>\n        <li>\n          <span>\n            Export to\n            <code>GeoJson</code>\n            or\n            <code>CSV</code>\n            .\n          </span>\n        </li>\n        <li>\n          <span>Copy the resulting content into your upload file.</span>\n        </li>\n      </ul>\n      <p>If you encounter the error again, try again using a lower percentage at step 5.</p>\n    </div>\n\n    <div class=\"my-3\" *ngIf=\"columns.length || (error.index && !isSchemaError)\">\n      <b>Hint:</b>\n      <span class=\"pl-1\">the error might be located</span>\n      <span class=\"pl-1\" *ngIf=\"error.index && !isSchemaError\">on row {{ error.index }}</span>\n      <ng-container *ngIf=\"columns.length\">\n        <span class=\"pl-1\">in {{ 'column' | pluralize: columns.length }}:</span>\n        <ul class=\"mt-1 content is-list-style-disc pl-5\">\n          <li *ngFor=\"let column of columns\">\n            <b class=\"pr-2\">{{ column.column }}</b>\n            <code>{{ column.name }}</code>\n          </li>\n        </ul>\n      </ng-container>\n    </div>\n\n    <div class=\"my-3\" *ngIf=\"error.suggestions?.length\">\n      <b>Suggestions:</b>\n      <span class=\"pl-1\">did you mean instead:</span>\n      <ul class=\"mt-1 content is-list-style-disc pl-5\">\n        <li *ngFor=\"let suggestion of error.suggestions\">\n          <code>{{ suggestion }}</code>\n        </li>\n      </ul>\n    </div>\n\n    <div *ngIf=\"showCsvPreview\" class=\"is-my-2\">\n      <b>\n        Preview\n        <sup>1</sup>\n        :\n      </b>\n      <div class=\"table-container is-mt-2\">\n        <table class=\"table is-bordered is-fullwidth is-hoverable is-narrow mb-0\">\n          <thead>\n            <th *ngIf=\"error.index\"></th>\n            <th *ngFor=\"let header of headers\">{{ header }}</th>\n          </thead>\n          <tbody>\n            <ng-container *bindOnce=\"rows\">\n              <tr *ngFor=\"let row of rows\">\n                <td class=\"has-text-danger\" *ngIf=\"error.index\">\n                  <span class=\"is-nowrap\">Row {{ error.index }}</span>\n                </td>\n                <td *ngFor=\"let col of row; let colIndex = index\">\n                  <span\n                    [class.has-text-danger]=\"\n                      error?.message === ErrorKeys.DuplicatedIds && hasDuplicatedError(col, colIndex)\n                    \">\n                    {{ col | ellipsis: 50 }}\n                  </span>\n                </td>\n              </tr>\n            </ng-container>\n          </tbody>\n        </table>\n      </div>\n      <p class=\"is-size-7\">\n        <i>1. The preview does not necessarily reflect the original Upload's order or content.</i>\n      </p>\n    </div>\n\n    <div *ngIf=\"showJsonPreview\" class=\"is-my-2\">\n      <b>Preview:</b>\n      <pre class=\"is-mt-2\"><code>{{stringify(error.node)}}</code></pre>\n    </div>\n  </div>\n</ng-container>\n\n<ng-template #defaultError>\n  <p class=\"is-mt-2 has-text-black\">\n    <span class=\"is-pr-1\">If you are still stuck, please see</span>\n    <a [href]=\"baseUrl + '/upload#example'\" target=\"_blank\">the example uploads and videos</a>\n    <span class=\"is-px-1\">\n      or try using the Wizard to get the first few rows of data formatted correctly, or use the chat to ask for help.\n    </span>\n  </p>\n</ng-template>\n\n<ng-template #schemaError>\n  <p>\n    <span class=\"pr-1\">You can see the accepted field type and some examples</span>\n    <a [href]=\"schemaKeyUrl\" target=\"_blank\">here</a>\n    <span>.</span>\n  </p>\n</ng-template>\n"]}
|
|
137
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"files-upload-errors.component.js","sourceRoot":"","sources":["../../../../src/files/files-upload-errors/files-upload-errors.component.ts","../../../../src/files/files-upload-errors/files-upload-errors.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, Input, OnInit } from '@angular/core';\nimport { parse } from 'papaparse';\nimport {\n  SchemaType,\n  NodeType,\n  JSON as HestiaJson,\n  UploadLimit,\n  isExpandable,\n  typeToSchemaType\n} from '@hestia-earth/schema';\nimport { File, maxFileSizeMb, SupportedExtensions } from '@hestia-earth/api';\nimport { toCsv, ErrorKeys as SchemaErrorKeys } from '@hestia-earth/schema-convert';\n\nimport { baseUrl } from '../../common';\nimport { schemaBaseUrl } from '../../schema/schema.service';\n\nconst nodeTypeToString = (type: NodeType) => `${type.charAt(0).toLowerCase()}${type.substring(1)}`;\n\nconst acceptedTypes = Object.values(NodeType)\n  .filter(val => val !== NodeType.Term)\n  .map(nodeTypeToString);\n\nexport enum ErrorKeys {\n  NoData = 'no-data',\n  NoHeaders = 'no-headers',\n  InvalidJSON = 'invalid-json',\n  InvalidSheetName = 'invalid-sheet-name',\n  InvalidFirstColumn = 'invalid-first-column',\n  DuplicatedHeaders = 'duplicated-headers',\n  DuplicatedIds = 'duplicated-ids',\n  PropertyRequired = 'property-required',\n  PropertyInternal = 'property-internal',\n  UploadsLimit = 'upload-limit',\n  NestedHeaders = 'nested-headers',\n  ReferenceExistingHeaders = 'reference-existing-headers',\n  MaxSize = 'max-size',\n  MaxRows = 'max-rows'\n}\n\ninterface ICSVHeader {\n  name: string;\n  index: number;\n}\n\ninterface ICSVColumn extends ICSVHeader {\n  column: string;\n}\n\nexport interface IError {\n  message: SchemaErrorKeys | ErrorKeys;\n  schema?: SchemaType;\n  schemaKey?: string;\n  key?: string;\n  property?: string;\n  value?: any;\n  min?: any;\n  max?: any;\n  error?: string;\n  node?: HestiaJson<SchemaType>;\n  index?: number;\n  headers?: ICSVHeader[];\n  suggestions?: string[];\n}\n\nconst valueToNodes = (value: any, type: SchemaType) =>\n  Array.isArray(value)\n    ? isExpandable(value)\n      ? value.map(val => ({\n          type,\n          ...val\n        }))\n      : null\n    : typeof value === 'object'\n    ? [\n        {\n          type,\n          ...value\n        }\n      ]\n    : null;\n\nconst nodesFromError = (error?: IError, type?: SchemaType) =>\n  error ? valueToNodes(error.node, type) || valueToNodes(error.value, error.schema) || [] : [];\n\nconst errorCsv = nodes => {\n  try {\n    return toCsv(nodes, { includeExising: true });\n  } catch (_err) {\n    return '';\n  }\n};\n\nconst numberToCol = (num: number) => {\n  let total = num + 1; // starts at 0\n  let str = '';\n  while (total > 0) {\n    const q = (total - 1) / 26;\n    const r = (total - 1) % 26;\n    total = Math.floor(q);\n    str = `${String.fromCharCode(65 + r)}${str}`;\n  }\n  return str;\n};\n\n@Component({\n  selector: 'he-files-upload-errors',\n  templateUrl: './files-upload-errors.component.html',\n  styleUrls: ['./files-upload-errors.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class FilesUploadErrorsComponent implements OnInit {\n  @Input()\n  private file?: Pick<File, 'filename' | 'schemaVersion'>;\n  @Input()\n  public error: IError;\n\n  public baseUrl = baseUrl();\n  public nodeTypes = acceptedTypes;\n  public SchemaErrorKeys = SchemaErrorKeys;\n  public ErrorKeys = ErrorKeys;\n  public UploadLimit = UploadLimit;\n  public maxFileSizeMb = maxFileSizeMb;\n  public columns: ICSVColumn[] = [];\n  public headers: string[] = [];\n  public rows: string[][] = [];\n\n  ngOnInit() {\n    this.columns = (this.error?.headers || []).map(header => ({\n      ...header,\n      column: numberToCol(header.index)\n    }));\n    const schemaType = this.error?.key ? typeToSchemaType(this.error.key.split('.')[0]) : null;\n    const nodeType = Object.values(SchemaType).includes(schemaType) ? schemaType : this.error?.schema;\n    const nodes = nodesFromError(this.error, nodeType);\n    const csv = errorCsv(nodes).trim();\n    const {\n      data: [headers, ...rows]\n    } = csv ? parse(csv) : { data: [[]] };\n    this.headers = headers;\n    this.rows = rows;\n  }\n\n  private get isJSONFile() {\n    return this.file?.filename?.endsWith(SupportedExtensions.json);\n  }\n\n  public stringify(value) {\n    return JSON.stringify(value, null, 2);\n  }\n\n  public hasDuplicatedError(value: string, colIndex: number) {\n    const firstVal = this.rows[0][colIndex];\n    return value !== firstVal;\n  }\n\n  public get hasNumberWithCommasError() {\n    return this.error?.error === 'failed to parse number' && (`${this.error?.value}` || '').includes(',');\n  }\n\n  public get hasGeoJSONError() {\n    return this.error?.error?.includes('Unable to parse GeoJSON value');\n  }\n\n  public get schemaUrl() {\n    return [schemaBaseUrl(this.file?.schemaVersion), this.error?.schema].filter(Boolean).join('/');\n  }\n\n  public get schemaKeyUrl() {\n    return [this.schemaUrl, this.error?.schemaKey || this.error?.key].filter(Boolean).join('#');\n  }\n\n  public get isSchemaError() {\n    return [SchemaErrorKeys.PropertyNotFound, SchemaErrorKeys.SchemaNotFound].includes(\n      this.error?.message as SchemaErrorKeys\n    );\n  }\n\n  public get showCsvPreview() {\n    return this.headers?.length && this.rows?.length && !this.isSchemaError && !this.isJSONFile;\n  }\n\n  public get showJsonPreview() {\n    return this.error?.node && this.isJSONFile;\n  }\n}\n","<ng-container *ngIf=\"!!error\">\n  <div class=\"has-text-danger\" [ngSwitch]=\"error?.message\">\n    <div *ngSwitchCase=\"ErrorKeys.InvalidJSON\">\n      <p>The uploaded content does not appear to contain valid JSON data:</p>\n\n      <div class=\"is-mt-2\">\n        <pre><code>{{error.error}}</code></pre>\n      </div>\n\n      <p>\n        Please validate your JSON content before uploading it using an\n        <a href=\"https://jsoneditoronline.org\" target=\"_blank\">online tool</a>\n        for example.\n      </p>\n    </div>\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.NoData\">\n      <p>No Nodes could be found in your Upload. Please verify the column headers match the Hestia schema.</p>\n\n      <ng-container *ngTemplateOutlet=\"defaultError\"></ng-container>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"SchemaErrorKeys.PropertyNotFound\">\n      <div class=\"notification is-danger\" role=\"alert\">\n        <p>\n          <fa-icon class=\"pr-2\" icon=\"exclamation-triangle\"></fa-icon>\n          <span>You have column headers that do not match the Hestia schema.</span>\n          <span class=\"px-1\">Please refer to the</span>\n          <a [href]=\"schemaUrl\" target=\"_blank\">schema</a>\n          <span class=\"pl-1\">and then correct the column headers indicated below.</span>\n        </p>\n\n        <ng-container *ngTemplateOutlet=\"defaultError\"></ng-container>\n      </div>\n\n      <div class=\"is-mb-2\">\n        <pre><code>{{error.key}}</code></pre>\n      </div>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"SchemaErrorKeys.PropertyInvalidFormat\">\n      <div class=\"is-mb-2\">\n        <span>\n          The following value in the column\n          <i>{{ error.key }}</i>\n          is not in the correct format:\n        </span>\n        <br />\n        <pre class=\"is-mt-2\"><code>{{stringify(error.value)}}</code></pre>\n      </div>\n      <ng-container [ngSwitch]=\"hasNumberWithCommasError\">\n        <ng-container *ngSwitchCase=\"true\">\n          <p class=\"is-mt-2\">\n            Please format your numbers to remove commas and use a decimal point to separate the whole part of the number\n            from the fractional part of the number.\n          </p>\n\n          <p class=\"is-mt-2 has-text-danger has-text-underline\">Incorrect:</p>\n          <pre class=\"is-mt-2\"><code>3,510.1</code></pre>\n\n          <p class=\"is-mt-2 has-text-success has-text-underline\">Correct:</p>\n          <pre class=\"is-mt-2\"><code>3510.1</code></pre>\n        </ng-container>\n\n        <ng-container *ngSwitchCase=\"false\">\n          <ng-container *ngTemplateOutlet=\"schemaError\"></ng-container>\n\n          <p>The error could either be a value in the column, or an error with the column header.</p>\n          <p>\n            If you are trying to set no data, we accept either\n            <code>-</code>\n            or an empty cell.\n          </p>\n        </ng-container>\n      </ng-container>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.PropertyRequired\">\n      <div class=\"is-mb-2\">\n        <span class=\"pr-1\">The following field is required in the</span>\n        <a class=\"pr-1\" *ngIf=\"error.schema\" [href]=\"schemaUrl\" target=\"_blank\">{{ error.schema }}</a>\n        <span>schema:</span>\n        <br />\n        <pre class=\"is-mt-2\"><code>{{error.property}}</code></pre>\n      </div>\n      <p [ngSwitch]=\"error.property\">\n        <span *ngSwitchCase=\"'id'\">\n          You must assign a unique\n          <code>id</code>\n          for every record in your data.\n        </span>\n        <span *ngSwitchCase=\"'type'\">\n          Every record in your data must contain a\n          <code>type</code>\n          .\n        </span>\n      </p>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.PropertyInternal\">\n      <div class=\"is-mb-2\">\n        <span class=\"pr-1\">\n          The following field is\n          <code>internal</code>\n          in the\n        </span>\n        <a [href]=\"schemaKeyUrl\" target=\"_blank\">{{ error.schema }} schema</a>\n        <span>:</span>\n        <br />\n        <pre class=\"is-mt-2\"><code>{{error.schemaKey || error.key}}</code></pre>\n      </div>\n      <p>\n        You cannot upload\n        <code>internal</code>\n        fields to Hestia.\n      </p>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"SchemaErrorKeys.SchemaNotFound\">\n      <div class=\"is-mb-2\">\n        <span>The following Node does not exist in the schema:</span>\n        <br />\n        <pre class=\"is-mt-2\"><code>{{error.key}}</code></pre>\n      </div>\n      <p>\n        <span>The list of accepted values is:</span>\n      </p>\n      <div class=\"content\">\n        <ul>\n          <li *ngFor=\"let type of nodeTypes\">\n            <i>{{ type }}</i>\n          </li>\n        </ul>\n      </div>\n    </div>\n\n    <ng-container *ngSwitchCase=\"ErrorKeys.UploadsLimit\">\n      <div class=\"is-mb-2\">\n        <span>You have reached the upload limit of</span>\n        <code class=\"mx-1\">{{ UploadLimit[error.schema] }}</code>\n        <span>\n          {{ error.schema }} in a single file (\n          <b>{{ error.value }}</b>\n          have been found).\n        </span>\n      </div>\n\n      <p>\n        <span>Please re-upload these data in multiple files with a maximum of</span>\n        <code class=\"mx-1\">{{ UploadLimit[error.schema] }}</code>\n        <span>{{ error.schema }} in each file.</span>\n      </p>\n    </ng-container>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.InvalidFirstColumn\">\n      <span>\n        No data in the Hestia format was detected on the first sheet. Either try selecting a different sheet in the\n        upload screen, or check the sheet to ensure the headers follow the schema - see examples\n      </span>\n      <a class=\"pl-1\" routerLink=\"/upload\" fragment=\"example\">here</a>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.NoHeaders\">\n      <p>\n        No headers matching the Hestia schema were found on the first row. Please remove all empty rows at the begining\n        of the file and try uploading again.\n      </p>\n\n      <ng-container *ngTemplateOutlet=\"defaultError\"></ng-container>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.InvalidSheetName\">\n      <div class=\"is-mb-2\">\n        <span class=\"pr-1\">The Excel sheet</span>\n        <code>{{ error.value }}</code>\n        <span class=\"pl-1\">does not exist.</span>\n        <span class=\"pl-1\">Your Excel file contains the following sheets:</span>\n        <ul class=\"mt-1 content is-list-style-disc pl-5\">\n          <li *ngFor=\"let e of error.error.split(';')\">\n            <code>{{ e }}</code>\n          </li>\n        </ul>\n      </div>\n\n      <p>\n        <span>Please re-upload selecting one of the sheets above.</span>\n      </p>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.DuplicatedHeaders\">\n      <span>Your upload contains duplicated column headers.</span>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.DuplicatedIds\">\n      <span class=\"pr-1\">You have multiple</span>\n      <code>{{ error.schema | pluralize }}</code>\n      <span class=\"px-1\">with the same</span>\n      <a [href]=\"schemaKeyUrl\" target=\"_blank\">{{ error.schemaKey }}</a>\n      <span class=\"px-1\">but different data. The</span>\n      <a [href]=\"schemaKeyUrl\" target=\"_blank\">{{ error.schemaKey }}</a>\n      <span class=\"pl-1\">is:</span>\n      <br />\n      <pre class=\"is-mt-2\"><code>{{error.value}}</code></pre>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"SchemaErrorKeys.DuplicatedIdFields\">\n      <span class=\"is-pr-1\">You have used both the</span>\n      <code>@id</code>\n      <span class=\"is-px-1\">and</span>\n      <code>id</code>\n      <span class=\"is-px-1\">fields in the same data. Please use</span>\n      <code>@id</code>\n      <span class=\"is-px-1\">when linking to existing nodes on the Platform, and</span>\n      <code>id</code>\n      <span class=\"is-px-1\">when linking to nodes you are uploading in this file using the same</span>\n      <code>id</code>\n      <span class=\"is-px-1\">field.</span>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.NestedHeaders\">\n      <p>\n        To link nodes together, you first need to define the node (e.g.,\n        <code>source.id</code>\n        ,\n        <code>source.bibliography.title</code>\n        ) and then create the link (e.g.,\n        <code>cycle.defaultSource.id</code>\n        ). We do not allow nesting (e.g.,\n        <code>cycle.defaultSource.bibliography.title</code>\n        ).\n      </p>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.ReferenceExistingHeaders\">\n      <p>\n        You have used the internal\n        <code>@id</code>\n        field to upload a new Node.\n      </p>\n      <p>\n        The\n        <code>@id</code>\n        field should only be used to reference existing Node or Terms. Example:\n      </p>\n\n      <p class=\"is-mt-2 has-text-danger has-text-underline\">Incorrect:</p>\n      <pre class=\"is-mt-2\"><code>cycle.@id,cycle.inputs.0.term.@id,source.id</code></pre>\n\n      <p class=\"is-mt-2 has-text-success has-text-underline\">Correct:</p>\n      <pre class=\"is-mt-2\"><code>cycle.id,cycle.inputs.0.term.@id,source.id</code></pre>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"SchemaErrorKeys.ObjectArrayInvalid\">\n      <p>The values in the column {{ error.key }} must be represented as arrays.</p>\n\n      <ng-container *ngTemplateOutlet=\"schemaError\"></ng-container>\n\n      <p>Examples:</p>\n      <pre class=\"is-mt-2\"><code>{{error.key}}.0.term.name</code></pre>\n      <pre class=\"is-mt-2\"><code>{{error.key}}.0.value</code></pre>\n      <pre class=\"is-mt-2\"><code>{{error.key}}.1.term.name</code></pre>\n      <pre class=\"is-mt-2\"><code>{{error.key}}.1.value</code></pre>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.MaxSize\">\n      <span>The maximum file size limit is {{ maxFileSizeMb }}Mb (current file size: {{ error.value }}Mb).</span>\n      <span class=\"pl-1\">Please re-upload these data in multiple files.</span>\n    </div>\n\n    <div class=\"is-mb-2\" *ngSwitchCase=\"ErrorKeys.MaxRows\">\n      <span>\n        Your file has {{ error.value | number }} rows. This might be an error caused by adding some data at the bottom\n        of a spreadsheet by accident: delete any unused rows and resubmit the file.\n      </span>\n    </div>\n\n    <div class=\"is-mb-2\" *ngIf=\"hasGeoJSONError\">\n      <p>If the GeoJSON is too large, please try to simplify it following these steps:</p>\n      <ul class=\"is-my-1 content is-list-style-decimal pl-5\">\n        <li>\n          <span class=\"is-pr-1\">Go to</span>\n          <a href=\"https://mapshaper.org/\" target=\"_blank\">https://mapshaper.org/</a>\n        </li>\n        <li>\n          <span>\n            Save the boundary in a valid\n            <code>.geojson</code>\n            file.\n          </span>\n        </li>\n        <li>\n          <span>Import it on mapshaper.</span>\n        </li>\n        <li>\n          <span>Click \"Simplify\" in the top right corner in the menu bar.</span>\n        </li>\n        <li>\n          <span>Adjust the percentage to a lower value, like 50% to start with.</span>\n        </li>\n        <li>\n          <span>\n            Export to\n            <code>GeoJson</code>\n            or\n            <code>CSV</code>\n            .\n          </span>\n        </li>\n        <li>\n          <span>Copy the resulting content into your upload file.</span>\n        </li>\n      </ul>\n      <p>If you encounter the error again, try again using a lower percentage at step 5.</p>\n    </div>\n\n    <div class=\"my-3\" *ngIf=\"columns.length || (error.index && !isSchemaError)\">\n      <b>Hint:</b>\n      <span class=\"pl-1\">the error might be located</span>\n      <span class=\"pl-1\" *ngIf=\"error.index && !isSchemaError\">on row {{ error.index }}</span>\n      <ng-container *ngIf=\"columns.length\">\n        <span class=\"pl-1\">in {{ 'column' | pluralize: columns.length }}:</span>\n        <ul class=\"mt-1 content is-list-style-disc pl-5\">\n          <li *ngFor=\"let column of columns\">\n            <b class=\"pr-2\">{{ column.column }}</b>\n            <code>{{ column.name }}</code>\n          </li>\n        </ul>\n      </ng-container>\n    </div>\n\n    <div class=\"my-3\" *ngIf=\"error.suggestions?.length\">\n      <b>Suggestions:</b>\n      <span class=\"pl-1\">did you mean instead:</span>\n      <ul class=\"mt-1 content is-list-style-disc pl-5\">\n        <li *ngFor=\"let suggestion of error.suggestions\">\n          <code>{{ suggestion }}</code>\n        </li>\n      </ul>\n    </div>\n\n    <div *ngIf=\"showCsvPreview\" class=\"is-my-2\">\n      <b>\n        Preview\n        <sup>1</sup>\n        :\n      </b>\n      <div class=\"table-container is-mt-2\">\n        <table class=\"table is-bordered is-fullwidth is-hoverable is-narrow mb-0\">\n          <thead>\n            <th *ngIf=\"error.index\"></th>\n            <th *ngFor=\"let header of headers\">{{ header }}</th>\n          </thead>\n          <tbody>\n            <ng-container *bindOnce=\"rows\">\n              <tr *ngFor=\"let row of rows\">\n                <td class=\"has-text-danger\" *ngIf=\"error.index\">\n                  <span class=\"is-nowrap\">Row {{ error.index }}</span>\n                </td>\n                <td *ngFor=\"let col of row; let colIndex = index\">\n                  <span\n                    [class.has-text-danger]=\"\n                      error?.message === ErrorKeys.DuplicatedIds && hasDuplicatedError(col, colIndex)\n                    \">\n                    {{ col | ellipsis: 50 }}\n                  </span>\n                </td>\n              </tr>\n            </ng-container>\n          </tbody>\n        </table>\n      </div>\n      <p class=\"is-size-7\">\n        <i>1. The preview does not necessarily reflect the original Upload's order or content.</i>\n      </p>\n    </div>\n\n    <div *ngIf=\"showJsonPreview\" class=\"is-my-2\">\n      <b>Preview:</b>\n      <pre class=\"is-mt-2\"><code>{{stringify(error.node)}}</code></pre>\n    </div>\n  </div>\n</ng-container>\n\n<ng-template #defaultError>\n  <p class=\"is-mt-2 has-text-black\">\n    <span class=\"is-pr-1\">If you are still stuck, please see</span>\n    <a [href]=\"baseUrl + '/upload#example'\" target=\"_blank\">the example uploads and videos</a>\n    <span class=\"is-px-1\">\n      or try using the Wizard to get the first few rows of data formatted correctly, or use the chat to ask for help.\n    </span>\n  </p>\n</ng-template>\n\n<ng-template #schemaError>\n  <p>\n    <span class=\"pr-1\">You can see the accepted field type and some examples</span>\n    <a [href]=\"schemaKeyUrl\" target=\"_blank\">here</a>\n    <span>.</span>\n  </p>\n</ng-template>\n"]}
|
|
@@ -53,13 +53,15 @@ export class ImpactAssessmentsProductsComponent {
|
|
|
53
53
|
this.filterTerm = '';
|
|
54
54
|
}
|
|
55
55
|
ngOnChanges(changes) {
|
|
56
|
+
if ('dataState' in changes) {
|
|
57
|
+
this.selectedView = View.table;
|
|
58
|
+
}
|
|
56
59
|
if ('impactAssessments' in changes) {
|
|
57
60
|
this.dataStateValues[this.dataState] = changes.impactAssessments.currentValue.slice();
|
|
58
61
|
this.updateSelectedIndex(this.selectedIndex);
|
|
59
62
|
return this.update();
|
|
60
63
|
}
|
|
61
|
-
if ('dataState' in changes) {
|
|
62
|
-
this.selectedView = View.table;
|
|
64
|
+
else if ('dataState' in changes) {
|
|
63
65
|
return this.updateDataState();
|
|
64
66
|
}
|
|
65
67
|
}
|
|
@@ -167,4 +169,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.8", ngImpor
|
|
|
167
169
|
}], enableFilterMethodModel: [{
|
|
168
170
|
type: Input
|
|
169
171
|
}] } });
|
|
170
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"impact-assessments-products.component.js","sourceRoot":"","sources":["../../../../src/impact-assessments/impact-assessments-products/impact-assessments-products.component.ts","../../../../src/impact-assessments/impact-assessments-products/impact-assessments-products.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAA4B,MAAM,eAAe,CAAC;AAC3E,OAAO,EAAE,SAAS,EAAE,MAAM,mBAAmB,CAAC;AAE9C,OAAO,EAAE,aAAa,EAAE,MAAM,+BAA+B,CAAC;AAC9D,OAAO,OAAO,MAAM,gBAAgB,CAAC;AACrC,OAAO,MAAM,MAAM,eAAe,CAAC;AAGnC,OAAO,EAAE,gBAAgB,EAAgB,YAAY,EAAE,oBAAoB,EAAE,MAAM,yBAAyB,CAAC;AAC7G,OAAO,EAAE,OAAO,EAAE,MAAM,oBAAoB,CAAC;;;;;;;;;;;;;;;;;;;;;;;AAE7C,IAAK,IAKJ;AALD,WAAK,IAAI;IACP,uBAAe,CAAA;IACf,uBAAe,CAAA;IACf,+BAAuB,CAAA;IACvB,qBAAa,CAAA;AACf,CAAC,EALI,IAAI,KAAJ,IAAI,QAKR;AAOD,MAAM,OAAO,kCAAkC;IAgC7C,YAAoB,WAA0B;QAA1B,gBAAW,GAAX,WAAW,CAAe;QA/BtC,oBAAe,GAEnB,EAAE,CAAC;QAKA,sBAAiB,GAA8B,EAAE,CAAC;QAElD,QAAG,GAAqD,SAAS,CAAC;QAMlE,4BAAuB,GAAG,KAAK,CAAC;QAEhC,YAAO,GAAG,KAAK,CAAC;QAChB,kBAAa,GAAG,aAAa,CAAC;QAC9B,YAAO,GAAG,OAAO,EAAE,CAAC;QACpB,iBAAY,GAAG,KAAK,CAAC;QACrB,SAAI,GAAG,IAAI,CAAC;QACZ,iBAAY,GAAG,IAAI,CAAC,KAAK,CAAC;QAE1B,iBAAY,GAAW,EAAE,CAAC;QAE1B,kBAAa,GAAG,CAAC,CAAC;QAElB,eAAU,GAA8B,EAAE,CAAC;QAC3C,eAAU,GAAG,EAAE,CAAC;IAE0B,CAAC;IAElD,WAAW,CAAC,OAAsB;QAChC,IAAI,mBAAmB,IAAI,OAAO,EAAE;YAClC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,SAAU,CAAC,GAAG,OAAO,CAAC,iBAAiB,CAAC,YAAY,CAAC,KAAK,EAAE,CAAC;YACvF,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;YAC7C,OAAO,IAAI,CAAC,MAAM,EAAE,CAAC;SACtB;QACD,IAAI,WAAW,IAAI,OAAO,EAAE;YAC1B,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,KAAK,CAAC;YAC/B,OAAO,IAAI,CAAC,eAAe,EAAE,CAAC;SAC/B;IACH,CAAC;IAEM,SAAS,CAAC,MAAc,EAAE,IAA6B;QAC5D,OAAO,IAAI,CAAC,KAAK,CAAC,CAAC;IACrB,CAAC;IAED,IAAW,UAAU;QACnB,OAAO,IAAI,CAAC,SAAS,KAAK,SAAS,CAAC,QAAQ,CAAC;IAC/C,CAAC;IAED,IAAW,cAAc;QACvB,OAAO,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,IAAI,EAAE,CAAC;IACnF,CAAC;IAEO,WAAW,CAAC,IAAW;QAC7B,OAAO,CAAC,IAAI,CAAC,eAAe,EAAE,MAAM,IAAI,CAAC,IAAI,CAAC,eAAe,IAAI,EAAE,CAAC,CAAC,QAAQ,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC;IAChG,CAAC;IAEO,MAAM;QACZ,IAAI,CAAC,uBAAuB,IAAI,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAC1D,IAAI,CAAC,aAAa,EAAE,CAAC;IACvB,CAAC;IAEO,kBAAkB;QACxB,MAAM,YAAY,GAAG,IAAI,CAAC,iBAAiB;aACxC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC;aACnE,MAAM,CAAC,OAAO,CAAC,CAAC;QACnB,IAAI,CAAC,YAAY,GAAG,MAAM,CAAC,YAAY,EAAE,KAAK,CAAC,CAAC;IAClD,CAAC;IAEM,aAAa;QAClB,oFAAoF;QACpF,sDAAsD;QACtD,MAAM,iBAAiB,GAAG,IAAI,CAAC,uBAAuB,IAAI,IAAI,CAAC,SAAS,KAAK,SAAS,CAAC,QAAQ,CAAC;QAChG,MAAM,4BAA4B,GAAG,gBAAgB,CACnD,IAAI,CAAC,iBAAiB,EACtB,IAAI,CAAC,GAAG,EACR,IAAI,CAAC,cAAc,EACnB,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,iBAAiB,CAAC,CACjD,CAAC;QACF,IAAI,CAAC,UAAU,GAAG,OAAO,CACvB,YAAY,CAAC,4BAA4B,CAAU,EACnD,CAAC,uBAAuB,EAAE,KAAK,CAAC,EAChC,CAAC,KAAK,EAAE,KAAK,CAAC,CACf,CAAC,MAAM,CAAC,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC,CAAC;IACzD,CAAC;IAEO,KAAK,CAAC,aAAa,CAAC,SAAoB;QAC9C,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC;YAC7B,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC,EAAE,MAAM,KAAK,IAAI,CAAC,iBAAiB,EAAE,MAAM;gBACxE,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC;gBACjC,CAAC,CAAC,MAAM,OAAO,CAAC,GAAG,CACf,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAChC,IAAI,CAAC,WAAW,CAAC,GAAG,CAA0B;oBAC5C,GAAG,IAAI;oBACP,SAAS;iBACV,CAAC,CACH,CACF,CAAC;IACV,CAAC;IAEO,KAAK,CAAC,eAAe;QAC3B,MAAM,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,SAAU,CAAC,CAAC;QAC1C,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,SAAU,CAAE,CAAC,KAAK,EAAE,CAAC;QACxE,OAAO,IAAI,CAAC,MAAM,EAAE,CAAC;IACvB,CAAC;IAEM,aAAa,CAAC,OAAO,EAAE,OAAO;QACnC,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACpE,CAAC;IAEM,UAAU,CAAC,MAA+B;QAC/C,OAAO,MAAM,CAAC,IAAI,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAK,MAAc,CAAC,EAAE,CAAC;IAC7G,CAAC;IAEM,UAAU,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,IAAI,EAAO;QACxC,OAAO,IAAI,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,KAAK,EAAE,CAAC,EAAE,IAAI,CAAC;IACtE,CAAC;IAED,IAAW,OAAO;QAChB,4GAA4G;QAC5G,OAAO,IAAI,CAAC,uBAAuB,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC;IAC1F,CAAC;IAED,IAAW,eAAe;QACxB,OAAO,CAAC,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,iBAAiB,CAAC,MAAM,KAAK,CAAC,IAAI,IAAI,CAAC,GAAG,KAAK,sBAAsB,CAAC;IACxG,CAAC;IAEO,UAAU,CAAC,IAAe,EAAE,iBAA0B;QAC5D,OAAO;YACL,CAAC,iBAAiB,IAAI,oBAAoB,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC,IAAI,CAAC;YAC1E,CAAC,IAAI,CAAC,UAAU,IAAI,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,WAAW,EAAE,EAAE,QAAQ,CAAC,IAAI,CAAC,UAAU,CAAC,WAAW,EAAE,CAAC;SAC7F,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;IACnB,CAAC;IAEM,MAAM,CAAC,IAAY;QACxB,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;QACvB,OAAO,IAAI,CAAC,MAAM,EAAE,CAAC;IACvB,CAAC;IAED,qBAAqB;IAEb,mBAAmB,CAAC,KAAa;QACvC,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC,CAAC;QACxB,8BAA8B;QAC9B,UAAU,CAAC,GAAG,EAAE,CAAC,CAAC,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC,CAAC,CAAC;IACjD,CAAC;IAEM,WAAW,CAAC,EAAE,MAAM,EAAE,EAAE,KAAK,EAAE,EAAE;QACtC,OAAO,IAAI,CAAC,mBAAmB,CAAC,CAAC,KAAK,CAAC,CAAC;IAC1C,CAAC;IAEM,KAAK,CAAC,qBAAqB;QAChC,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;QACpB,yCAAyC;QACzC,MAAM,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;QAC7C,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;QACrB,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,IAAI,CAAC;IAChC,CAAC;8GAlKU,kCAAkC;kGAAlC,kCAAkC,6RCvB/C,4oWAqQA;;2FD9Oa,kCAAkC;kBAL9C,SAAS;+BACE,gCAAgC;oGASlC,MAAM;sBADb,KAAK;gBAIC,iBAAiB;sBADvB,KAAK;gBAGC,GAAG;sBADT,KAAK;gBAGC,SAAS;sBADf,KAAK;gBAGC,eAAe;sBADrB,KAAK;gBAGC,uBAAuB;sBAD7B,KAAK","sourcesContent":["import { Component, Input, OnChanges, SimpleChanges } from '@angular/core';\nimport { DataState } from '@hestia-earth/api';\nimport { ICycleJSONLD, IImpactAssessmentJSONLD, Indicator, TermTermType, Term } from '@hestia-earth/schema';\nimport { propertyValue } from '@hestia-earth/utils/dist/term';\nimport orderBy from 'lodash.orderby';\nimport uniqBy from 'lodash.uniqby';\n\nimport { HeNodeService } from '../../node/node.service';\nimport { groupNodesByTerm, IGroupedKeys, grouppedKeys, isMethodModelAllowed } from '../../common/node-utils';\nimport { baseUrl } from '../../common/utils';\n\nenum View {\n  table = 'table',\n  chart = 'chart',\n  breakdown = 'breakdown',\n  logs = 'logs'\n}\n\n@Component({\n  selector: 'he-impact-assessments-products',\n  templateUrl: './impact-assessments-products.component.html',\n  styleUrls: ['./impact-assessments-products.component.scss']\n})\nexport class ImpactAssessmentsProductsComponent implements OnChanges {\n  private dataStateValues: {\n    [dataState in DataState]?: IImpactAssessmentJSONLD[];\n  } = {};\n  @Input()\n  private cycles?: ICycleJSONLD[];\n\n  @Input()\n  public impactAssessments: IImpactAssessmentJSONLD[] = [];\n  @Input()\n  public key: 'endpoints' | 'impacts' | 'emissionsResourceUse' = 'impacts';\n  @Input()\n  public dataState?: DataState;\n  @Input()\n  public filterTermTypes?: TermTermType[];\n  @Input()\n  public enableFilterMethodModel = false;\n\n  public loading = false;\n  public propertyValue = propertyValue;\n  public baseUrl = baseUrl();\n  public showDownload = false;\n  public View = View;\n  public selectedView = View.table;\n\n  public methodModels: Term[] = [];\n  public selectedMethodModel?: Term;\n  public selectedIndex = 0;\n\n  public indicators: IGroupedKeys<Indicator>[] = [];\n  public filterTerm = '';\n\n  constructor(private nodeService: HeNodeService) {}\n\n  ngOnChanges(changes: SimpleChanges) {\n    if ('impactAssessments' in changes) {\n      this.dataStateValues[this.dataState!] = changes.impactAssessments.currentValue.slice();\n      this.updateSelectedIndex(this.selectedIndex);\n      return this.update();\n    }\n    if ('dataState' in changes) {\n      this.selectedView = View.table;\n      return this.updateDataState();\n    }\n  }\n\n  public trackById(_index: number, item: IImpactAssessmentJSONLD) {\n    return item['@id'];\n  }\n\n  public get isOriginal() {\n    return this.dataState === DataState.original;\n  }\n\n  public get originalValues() {\n    return (this.isOriginal ? null : this.dataStateValues[DataState.original]) || [];\n  }\n\n  private termAllowed(term?: Term) {\n    return !this.filterTermTypes?.length || (this.filterTermTypes || []).includes(term?.termType);\n  }\n\n  private update() {\n    this.enableFilterMethodModel && this.updateMethodModels();\n    this.updateImpacts();\n  }\n\n  private updateMethodModels() {\n    const methodModels = this.impactAssessments\n      .flatMap(impact => (impact[this.key] || []).map(v => v.methodModel))\n      .filter(Boolean);\n    this.methodModels = uniqBy(methodModels, '@id');\n  }\n\n  public updateImpacts() {\n    // TODO: when there are no indicators for the default model, the table appears empty\n    // should default back to the first model in this case\n    const filterMethodModel = this.enableFilterMethodModel && this.dataState !== DataState.original;\n    const indicatorPerImpactAssessment = groupNodesByTerm<IImpactAssessmentJSONLD, Indicator>(\n      this.impactAssessments,\n      this.key,\n      this.originalValues,\n      node => this.filterNode(node, filterMethodModel)\n    );\n    this.indicators = orderBy(\n      grouppedKeys(indicatorPerImpactAssessment) as any[],\n      ['value.methodTierOrder', 'key'],\n      ['asc', 'asc']\n    ).filter(({ value }) => this.termAllowed(value?.term));\n  }\n\n  private async loadDataState(dataState: DataState) {\n    this.dataStateValues[dataState] =\n      this.dataStateValues[dataState]?.length === this.impactAssessments?.length\n        ? this.dataStateValues[dataState]\n        : await Promise.all(\n            this.impactAssessments.map(node =>\n              this.nodeService.get<IImpactAssessmentJSONLD>({\n                ...node,\n                dataState\n              })\n            )\n          );\n  }\n\n  private async updateDataState() {\n    await this.loadDataState(this.dataState!);\n    this.impactAssessments = this.dataStateValues[this.dataState!]!.slice();\n    return this.update();\n  }\n\n  public togglePopover(popover, context) {\n    return popover.isOpen() ? popover.close() : popover.open(context);\n  }\n\n  public impactName(impact: IImpactAssessmentJSONLD) {\n    return impact.name || (impact.cycle ? this.cycleLabel(impact.cycle) : impact['@id']) || (impact as any).id;\n  }\n\n  public cycleLabel({ '@id': id, name }: any) {\n    return name || (this.cycles || []).find(v => v['@id'] === id)?.name;\n  }\n\n  public get hasData() {\n    // as we filter by methodModel, we might have an empty table for the default models but not for other models\n    return this.enableFilterMethodModel ? this.methodModels.length : this.indicators.length;\n  }\n\n  public get enableBreakdown() {\n    return !this.isOriginal && this.impactAssessments.length === 1 && this.key !== 'emissionsResourceUse';\n  }\n\n  private filterNode(node: Indicator, filterMethodModel: boolean) {\n    return [\n      !filterMethodModel || isMethodModelAllowed(this.selectedMethodModel)(node),\n      !this.filterTerm || node?.term?.name?.toLowerCase()?.includes(this.filterTerm.toLowerCase())\n    ].every(Boolean);\n  }\n\n  public filter(term: string) {\n    this.filterTerm = term;\n    return this.update();\n  }\n\n  // Recalculation logs\n\n  private updateSelectedIndex(index: number) {\n    this.selectedIndex = -1;\n    // force a refresh of the logs\n    setTimeout(() => (this.selectedIndex = index));\n  }\n\n  public selectIndex({ target: { value } }) {\n    return this.updateSelectedIndex(+value);\n  }\n\n  public async showRecalculationLogs() {\n    this.loading = true;\n    // make sure original data is also loaded\n    await this.loadDataState(DataState.original);\n    this.loading = false;\n    this.selectedView = View.logs;\n  }\n}\n","<div class=\"columns is-variable is-align-items-center is-2 m-0\">\n  <div class=\"column\">\n    <ng-container *ngIf=\"selectedView === View.table && indicators.length\">\n      <button\n        class=\"button is-small is-ghost\"\n        (click)=\"showDownload = true\"\n        ngbTooltip=\"Download as CSV\"\n        placement=\"bottom\">\n        <fa-icon icon=\"download\" size=\"lg\"></fa-icon>\n      </button>\n    </ng-container>\n  </div>\n  <div class=\"column is-narrow\" *ngIf=\"selectedView === View.table\">\n    <he-search-extend\n      class=\"is-secondary is-small\"\n      placeholder=\"Filter indicators by name\"\n      (search)=\"filter($event)\"></he-search-extend>\n  </div>\n  <div class=\"column is-narrow\" *ngIf=\"impactAssessments.length > 1 || enableBreakdown || !isOriginal\">\n    <div class=\"field has-addons\">\n      <div class=\"control\">\n        <button\n          class=\"button is-small\"\n          [class.is-selected]=\"selectedView === View.table\"\n          (click)=\"selectedView = View.table\">\n          <span class=\"icon is-small\">\n            <fa-icon icon=\"list\" aria-hidden=\"true\"></fa-icon>\n          </span>\n          <span class=\"is-hidden-mobile\">Table view</span>\n        </button>\n      </div>\n      <div class=\"control\" *ngIf=\"impactAssessments.length > 1\">\n        <button\n          class=\"button is-small\"\n          [class.is-selected]=\"selectedView === View.chart\"\n          (click)=\"selectedView = View.chart\">\n          <span class=\"icon is-small\">\n            <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n          </span>\n          <span class=\"is-hidden-mobile\">Chart view</span>\n        </button>\n      </div>\n      <div class=\"control\" *ngIf=\"enableBreakdown\">\n        <button\n          class=\"button is-small\"\n          [class.is-selected]=\"selectedView === View.breakdown\"\n          (click)=\"selectedView = View.breakdown\">\n          <span class=\"icon is-small\">\n            <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n          </span>\n          <span class=\"is-hidden-mobile\">Breakdown view</span>\n        </button>\n      </div>\n      <div class=\"control\" *ngIf=\"!isOriginal\">\n        <button\n          class=\"button is-small\"\n          [class.is-selected]=\"selectedView === View.logs\"\n          (click)=\"showRecalculationLogs()\">\n          <span class=\"icon is-small\">\n            <fa-icon icon=\"calculator\" aria-hidden=\"true\"></fa-icon>\n          </span>\n          <span class=\"is-hidden-mobile\">Recalculations logs</span>\n        </button>\n      </div>\n    </div>\n  </div>\n</div>\n\n<div class=\"px-3 pb-3\" [class.is-hidden]=\"selectedView !== View.table\">\n  <ng-container *ngIf=\"hasData; else emptyTable\">\n    <he-data-table class=\"mb-1 is-small\" [small]=\"true\" maxHeight=\"320\">\n      <table class=\"table is-fullwidth is-narrow is-striped\">\n        <thead>\n          <tr class=\"has-text-weight-semibold\">\n            <th class=\"width-auto py-0\">\n              <div class=\"select is-small\" *ngIf=\"enableFilterMethodModel\">\n                <select name=\"selectedMethodModel\" (change)=\"updateImpacts()\" [(ngModel)]=\"selectedMethodModel\">\n                  <option [ngValue]=\"undefined\">Filter Model</option>\n                  <option *ngFor=\"let term of methodModels\" [ngValue]=\"term\">{{ term.name }}</option>\n                </select>\n              </div>\n            </th>\n            <th class=\"has-border-right\"></th>\n            <th class=\"has-border-right\"></th>\n            <th *ngFor=\"let indicator of indicators\" [attr.title]=\"indicator.value.term.name\">\n              <he-node-link [node]=\"indicator.value.term\">\n                <span [innerHtml]=\"indicator.value.term.name | ellipsis: 30 | compound\"></span>\n              </he-node-link>\n            </th>\n          </tr>\n          <tr class=\"is-italic has-text-weight-semibold\">\n            <th class=\"width-auto has-border-right\"></th>\n            <th class=\"has-border-right\">\n              <a [href]=\"baseUrl + '/schema/ImpactAssessment#functionalUnit'\" target=\"_blank\">Functional unit</a>\n            </th>\n            <th class=\"has-border-right\">\n              <a [href]=\"baseUrl + '/schema/ImpactAssessment#product'\" target=\"_blank\">Product</a>\n            </th>\n            <th *ngFor=\"let indicator of indicators\" [attr.title]=\"indicator.value.term.units\">\n              <span [innerHtml]=\"indicator.value.term.units | compound\"></span>\n              <he-terms-units-description\n                class=\"is-inline-block is-ml-2\"\n                [term]=\"indicator.value.term\"></he-terms-units-description>\n            </th>\n          </tr>\n        </thead>\n        <tbody>\n          <ng-container *ngFor=\"let impactAssessment of impactAssessments; trackBy: trackById; let i = index\">\n            <tr>\n              <td class=\"width-auto has-border-right\" [attr.title]=\"impactName(impactAssessment)\">\n                <he-node-link [node]=\"impactAssessment\">\n                  <span>{{ i + 1 }}. {{ impactName(impactAssessment) }}</span>\n                </he-node-link>\n              </td>\n              <td class=\"has-border-right\" [attr.title]=\"impactAssessment.product?.term?.units\">\n                <span>1 {{ impactAssessment.product?.term?.units }}</span>\n              </td>\n              <td class=\"has-border-right\" [attr.title]=\"impactAssessment.product?.term?.name\">\n                <he-node-link *ngIf=\"impactAssessment.product?.term\" [node]=\"impactAssessment.product.term\">\n                  <span>{{ impactAssessment.product.term.name | ellipsis: 30 }}</span>\n                </he-node-link>\n              </td>\n              <td class=\"is-nowrap\" *ngFor=\"let indicator of indicators\">\n                <span\n                  *ngIf=\"indicator.value.values[impactAssessment['@id']]; else emptyValue\"\n                  class=\"trigger-popover\"\n                  [ngbPopover]=\"details\"\n                  [autoClose]=\"'outside'\"\n                  popoverClass=\"is-narrow\"\n                  triggers=\"manual\"\n                  #p=\"ngbPopover\"\n                  placement=\"left\"\n                  container=\"body\"\n                  (click)=\"\n                    togglePopover(p, {\n                      data: indicator.value.values[impactAssessment['@id']],\n                      impactAssessment: impactAssessment,\n                      key: key\n                    })\n                  \">\n                  <span pointer>\n                    {{\n                      propertyValue(indicator.value.values[impactAssessment['@id']].value, indicator.value.term['@id'])\n                        | precision: 3\n                        | default: '-'\n                    }}\n                  </span>\n                  <he-blank-node-state\n                    class=\"ml-1\"\n                    [dataState]=\"dataState\"\n                    [node]=\"indicator.value.values[impactAssessment['@id']].node\"\n                    key=\"value\"\n                    [state]=\"impactAssessment.aggregated ? 'aggregated' : undefined\"></he-blank-node-state>\n                </span>\n              </td>\n            </tr>\n          </ng-container>\n        </tbody>\n      </table>\n    </he-data-table>\n\n    <he-blank-node-state-notice [dataState]=\"dataState\"></he-blank-node-state-notice>\n  </ng-container>\n</div>\n\n<he-impact-assessments-indicator-breakdown-chart\n  *ngIf=\"selectedView === View.breakdown\"\n  [impactAssessment]=\"impactAssessments[0]\"\n  [indicators]=\"impactAssessments[0][key]\"></he-impact-assessments-indicator-breakdown-chart>\n\n<he-impact-assessments-indicators-chart\n  *ngIf=\"selectedView === View.chart && impactAssessments.length > 1\"\n  [key]=\"key\"\n  [impactAssessments]=\"impactAssessments\"\n  [filterTermTypes]=\"filterTermTypes\"></he-impact-assessments-indicators-chart>\n\n<ng-container *ngIf=\"selectedView === View.logs && !isOriginal\">\n  <div class=\"field has-addons pt-2 px-3\" *ngIf=\"impactAssessments.length > 1\">\n    <div class=\"control\">\n      <span class=\"button is-small is-static is-secondary\">Select an Impact Assessment</span>\n    </div>\n    <div class=\"control is-expanded\">\n      <div class=\"select is-small is-fullwidth is-secondary\">\n        <select (change)=\"selectIndex($event)\">\n          <option *ngFor=\"let value of impactAssessments; let i = index\" [value]=\"i\">\n            {{ i + 1 }}. {{ impactName(value) }}\n          </option>\n        </select>\n      </div>\n    </div>\n  </div>\n\n  <he-impact-assessments-products-logs\n    *ngIf=\"selectedIndex >= 0\"\n    [key]=\"key\"\n    [impactAssessment]=\"impactAssessments[selectedIndex]\"\n    [filterTermTypes]=\"filterTermTypes\"\n    [originalValues]=\"originalValues[selectedIndex]?.[key]\"\n    [recalculatedValues]=\"impactAssessments[selectedIndex]?.[key]\"></he-impact-assessments-products-logs>\n</ng-container>\n\n<he-node-csv-export-confirm\n  *ngIf=\"showDownload\"\n  [nodes]=\"impactAssessments\"\n  [filename]=\"'impact-' + key + '.csv'\"\n  [isUpload]=\"false\"\n  [headerKeys]=\"['impactAssessment.id', 'impactAssessment.@id', 'impactAssessment.' + key + '.']\"\n  (closed)=\"showDownload = false\"></he-node-csv-export-confirm>\n\n<ng-template #emptyTable>\n  <div class=\"has-text-centered\">\n    <span>No data available</span>\n    <span class=\"is-pl-1\" [class.is-hidden]=\"!filterTerm\">matching your search criteria</span>\n    <span>.</span>\n    <span [class.is-hidden]=\"!isOriginal\">\n      Switch to\n      <code>recalculated</code>\n      version.\n    </span>\n  </div>\n</ng-template>\n\n<ng-template #emptyValue>\n  <span>-</span>\n</ng-template>\n\n<ng-template #details let-node=\"impactAssessment\" let-data=\"data\" let-key=\"key\">\n  <p *bindOnce=\"node\">\n    <b>\n      <span *ngIf=\"data.cycle\">{{ cycleLabel(node.cycle) }}</span>\n      <span *ngIf=\"!data.cycle\">{{ data.name }}</span>\n    </b>\n  </p>\n  <he-node-value-details\n    [data]=\"data\"\n    [dataState]=\"dataState\"\n    [nodeType]=\"node['@type']\"\n    [dataKey]=\"key\"></he-node-value-details>\n</ng-template>\n\n<ng-template #suggestion let-impact=\"result\" let-t=\"term\">\n  <div class=\"is-block\">\n    <ngb-highlight [result]=\"impact.name || impact.cycle.name\" [term]=\"t\"></ngb-highlight>\n  </div>\n  <div class=\"columns is-flex\">\n    <div class=\"column\" *ngIf=\"impact.country\">\n      <span class=\"pr-1 has-text-underline\">Country:</span>\n      <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.country.name\" [term]=\"t\"></ngb-highlight></span>\n    </div>\n    <div class=\"column\" *ngIf=\"impact.product\">\n      <span class=\"pr-1 has-text-underline\">Product:</span>\n      <span class=\"is-inline-flex\">\n        <ngb-highlight [result]=\"impact.product.term?.name || impact.product.name\" [term]=\"t\"></ngb-highlight>\n      </span>\n    </div>\n    <div class=\"column\" *ngIf=\"impact.endDate\">\n      <span class=\"pr-1 has-text-underline\">Date:</span>\n      <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.endDate\" [term]=\"t\"></ngb-highlight></span>\n    </div>\n  </div>\n</ng-template>\n"]}
|
|
172
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"impact-assessments-products.component.js","sourceRoot":"","sources":["../../../../src/impact-assessments/impact-assessments-products/impact-assessments-products.component.ts","../../../../src/impact-assessments/impact-assessments-products/impact-assessments-products.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAA4B,MAAM,eAAe,CAAC;AAC3E,OAAO,EAAE,SAAS,EAAE,MAAM,mBAAmB,CAAC;AAE9C,OAAO,EAAE,aAAa,EAAE,MAAM,+BAA+B,CAAC;AAC9D,OAAO,OAAO,MAAM,gBAAgB,CAAC;AACrC,OAAO,MAAM,MAAM,eAAe,CAAC;AAGnC,OAAO,EAAE,gBAAgB,EAAgB,YAAY,EAAE,oBAAoB,EAAE,MAAM,yBAAyB,CAAC;AAC7G,OAAO,EAAE,OAAO,EAAE,MAAM,oBAAoB,CAAC;;;;;;;;;;;;;;;;;;;;;;;AAE7C,IAAK,IAKJ;AALD,WAAK,IAAI;IACP,uBAAe,CAAA;IACf,uBAAe,CAAA;IACf,+BAAuB,CAAA;IACvB,qBAAa,CAAA;AACf,CAAC,EALI,IAAI,KAAJ,IAAI,QAKR;AAOD,MAAM,OAAO,kCAAkC;IAgC7C,YAAoB,WAA0B;QAA1B,gBAAW,GAAX,WAAW,CAAe;QA/BtC,oBAAe,GAEnB,EAAE,CAAC;QAKA,sBAAiB,GAA8B,EAAE,CAAC;QAElD,QAAG,GAAqD,SAAS,CAAC;QAMlE,4BAAuB,GAAG,KAAK,CAAC;QAEhC,YAAO,GAAG,KAAK,CAAC;QAChB,kBAAa,GAAG,aAAa,CAAC;QAC9B,YAAO,GAAG,OAAO,EAAE,CAAC;QACpB,iBAAY,GAAG,KAAK,CAAC;QACrB,SAAI,GAAG,IAAI,CAAC;QACZ,iBAAY,GAAG,IAAI,CAAC,KAAK,CAAC;QAE1B,iBAAY,GAAW,EAAE,CAAC;QAE1B,kBAAa,GAAG,CAAC,CAAC;QAElB,eAAU,GAA8B,EAAE,CAAC;QAC3C,eAAU,GAAG,EAAE,CAAC;IAE0B,CAAC;IAElD,WAAW,CAAC,OAAsB;QAChC,IAAI,WAAW,IAAI,OAAO,EAAE;YAC1B,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,KAAK,CAAC;SAChC;QACD,IAAI,mBAAmB,IAAI,OAAO,EAAE;YAClC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,SAAU,CAAC,GAAG,OAAO,CAAC,iBAAiB,CAAC,YAAY,CAAC,KAAK,EAAE,CAAC;YACvF,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;YAC7C,OAAO,IAAI,CAAC,MAAM,EAAE,CAAC;SACtB;aAAM,IAAI,WAAW,IAAI,OAAO,EAAE;YACjC,OAAO,IAAI,CAAC,eAAe,EAAE,CAAC;SAC/B;IACH,CAAC;IAEM,SAAS,CAAC,MAAc,EAAE,IAA6B;QAC5D,OAAO,IAAI,CAAC,KAAK,CAAC,CAAC;IACrB,CAAC;IAED,IAAW,UAAU;QACnB,OAAO,IAAI,CAAC,SAAS,KAAK,SAAS,CAAC,QAAQ,CAAC;IAC/C,CAAC;IAED,IAAW,cAAc;QACvB,OAAO,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,IAAI,EAAE,CAAC;IACnF,CAAC;IAEO,WAAW,CAAC,IAAW;QAC7B,OAAO,CAAC,IAAI,CAAC,eAAe,EAAE,MAAM,IAAI,CAAC,IAAI,CAAC,eAAe,IAAI,EAAE,CAAC,CAAC,QAAQ,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC;IAChG,CAAC;IAEO,MAAM;QACZ,IAAI,CAAC,uBAAuB,IAAI,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAC1D,IAAI,CAAC,aAAa,EAAE,CAAC;IACvB,CAAC;IAEO,kBAAkB;QACxB,MAAM,YAAY,GAAG,IAAI,CAAC,iBAAiB;aACxC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC;aACnE,MAAM,CAAC,OAAO,CAAC,CAAC;QACnB,IAAI,CAAC,YAAY,GAAG,MAAM,CAAC,YAAY,EAAE,KAAK,CAAC,CAAC;IAClD,CAAC;IAEM,aAAa;QAClB,oFAAoF;QACpF,sDAAsD;QACtD,MAAM,iBAAiB,GAAG,IAAI,CAAC,uBAAuB,IAAI,IAAI,CAAC,SAAS,KAAK,SAAS,CAAC,QAAQ,CAAC;QAChG,MAAM,4BAA4B,GAAG,gBAAgB,CACnD,IAAI,CAAC,iBAAiB,EACtB,IAAI,CAAC,GAAG,EACR,IAAI,CAAC,cAAc,EACnB,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,iBAAiB,CAAC,CACjD,CAAC;QACF,IAAI,CAAC,UAAU,GAAG,OAAO,CACvB,YAAY,CAAC,4BAA4B,CAAU,EACnD,CAAC,uBAAuB,EAAE,KAAK,CAAC,EAChC,CAAC,KAAK,EAAE,KAAK,CAAC,CACf,CAAC,MAAM,CAAC,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC,CAAC;IACzD,CAAC;IAEO,KAAK,CAAC,aAAa,CAAC,SAAoB;QAC9C,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC;YAC7B,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC,EAAE,MAAM,KAAK,IAAI,CAAC,iBAAiB,EAAE,MAAM;gBACxE,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC;gBACjC,CAAC,CAAC,MAAM,OAAO,CAAC,GAAG,CACf,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAChC,IAAI,CAAC,WAAW,CAAC,GAAG,CAA0B;oBAC5C,GAAG,IAAI;oBACP,SAAS;iBACV,CAAC,CACH,CACF,CAAC;IACV,CAAC;IAEO,KAAK,CAAC,eAAe;QAC3B,MAAM,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,SAAU,CAAC,CAAC;QAC1C,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,SAAU,CAAE,CAAC,KAAK,EAAE,CAAC;QACxE,OAAO,IAAI,CAAC,MAAM,EAAE,CAAC;IACvB,CAAC;IAEM,aAAa,CAAC,OAAO,EAAE,OAAO;QACnC,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACpE,CAAC;IAEM,UAAU,CAAC,MAA+B;QAC/C,OAAO,MAAM,CAAC,IAAI,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAK,MAAc,CAAC,EAAE,CAAC;IAC7G,CAAC;IAEM,UAAU,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,IAAI,EAAO;QACxC,OAAO,IAAI,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,KAAK,EAAE,CAAC,EAAE,IAAI,CAAC;IACtE,CAAC;IAED,IAAW,OAAO;QAChB,4GAA4G;QAC5G,OAAO,IAAI,CAAC,uBAAuB,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC;IAC1F,CAAC;IAED,IAAW,eAAe;QACxB,OAAO,CAAC,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,iBAAiB,CAAC,MAAM,KAAK,CAAC,IAAI,IAAI,CAAC,GAAG,KAAK,sBAAsB,CAAC;IACxG,CAAC;IAEO,UAAU,CAAC,IAAe,EAAE,iBAA0B;QAC5D,OAAO;YACL,CAAC,iBAAiB,IAAI,oBAAoB,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC,IAAI,CAAC;YAC1E,CAAC,IAAI,CAAC,UAAU,IAAI,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,WAAW,EAAE,EAAE,QAAQ,CAAC,IAAI,CAAC,UAAU,CAAC,WAAW,EAAE,CAAC;SAC7F,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;IACnB,CAAC;IAEM,MAAM,CAAC,IAAY;QACxB,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;QACvB,OAAO,IAAI,CAAC,MAAM,EAAE,CAAC;IACvB,CAAC;IAED,qBAAqB;IAEb,mBAAmB,CAAC,KAAa;QACvC,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC,CAAC;QACxB,8BAA8B;QAC9B,UAAU,CAAC,GAAG,EAAE,CAAC,CAAC,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC,CAAC,CAAC;IACjD,CAAC;IAEM,WAAW,CAAC,EAAE,MAAM,EAAE,EAAE,KAAK,EAAE,EAAE;QACtC,OAAO,IAAI,CAAC,mBAAmB,CAAC,CAAC,KAAK,CAAC,CAAC;IAC1C,CAAC;IAEM,KAAK,CAAC,qBAAqB;QAChC,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;QACpB,yCAAyC;QACzC,MAAM,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;QAC7C,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;QACrB,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,IAAI,CAAC;IAChC,CAAC;8GAnKU,kCAAkC;kGAAlC,kCAAkC,6RCvB/C,4oWAqQA;;2FD9Oa,kCAAkC;kBAL9C,SAAS;+BACE,gCAAgC;oGASlC,MAAM;sBADb,KAAK;gBAIC,iBAAiB;sBADvB,KAAK;gBAGC,GAAG;sBADT,KAAK;gBAGC,SAAS;sBADf,KAAK;gBAGC,eAAe;sBADrB,KAAK;gBAGC,uBAAuB;sBAD7B,KAAK","sourcesContent":["import { Component, Input, OnChanges, SimpleChanges } from '@angular/core';\nimport { DataState } from '@hestia-earth/api';\nimport { ICycleJSONLD, IImpactAssessmentJSONLD, Indicator, TermTermType, Term } from '@hestia-earth/schema';\nimport { propertyValue } from '@hestia-earth/utils/dist/term';\nimport orderBy from 'lodash.orderby';\nimport uniqBy from 'lodash.uniqby';\n\nimport { HeNodeService } from '../../node/node.service';\nimport { groupNodesByTerm, IGroupedKeys, grouppedKeys, isMethodModelAllowed } from '../../common/node-utils';\nimport { baseUrl } from '../../common/utils';\n\nenum View {\n  table = 'table',\n  chart = 'chart',\n  breakdown = 'breakdown',\n  logs = 'logs'\n}\n\n@Component({\n  selector: 'he-impact-assessments-products',\n  templateUrl: './impact-assessments-products.component.html',\n  styleUrls: ['./impact-assessments-products.component.scss']\n})\nexport class ImpactAssessmentsProductsComponent implements OnChanges {\n  private dataStateValues: {\n    [dataState in DataState]?: IImpactAssessmentJSONLD[];\n  } = {};\n  @Input()\n  private cycles?: ICycleJSONLD[];\n\n  @Input()\n  public impactAssessments: IImpactAssessmentJSONLD[] = [];\n  @Input()\n  public key: 'endpoints' | 'impacts' | 'emissionsResourceUse' = 'impacts';\n  @Input()\n  public dataState?: DataState;\n  @Input()\n  public filterTermTypes?: TermTermType[];\n  @Input()\n  public enableFilterMethodModel = false;\n\n  public loading = false;\n  public propertyValue = propertyValue;\n  public baseUrl = baseUrl();\n  public showDownload = false;\n  public View = View;\n  public selectedView = View.table;\n\n  public methodModels: Term[] = [];\n  public selectedMethodModel?: Term;\n  public selectedIndex = 0;\n\n  public indicators: IGroupedKeys<Indicator>[] = [];\n  public filterTerm = '';\n\n  constructor(private nodeService: HeNodeService) {}\n\n  ngOnChanges(changes: SimpleChanges) {\n    if ('dataState' in changes) {\n      this.selectedView = View.table;\n    }\n    if ('impactAssessments' in changes) {\n      this.dataStateValues[this.dataState!] = changes.impactAssessments.currentValue.slice();\n      this.updateSelectedIndex(this.selectedIndex);\n      return this.update();\n    } else if ('dataState' in changes) {\n      return this.updateDataState();\n    }\n  }\n\n  public trackById(_index: number, item: IImpactAssessmentJSONLD) {\n    return item['@id'];\n  }\n\n  public get isOriginal() {\n    return this.dataState === DataState.original;\n  }\n\n  public get originalValues() {\n    return (this.isOriginal ? null : this.dataStateValues[DataState.original]) || [];\n  }\n\n  private termAllowed(term?: Term) {\n    return !this.filterTermTypes?.length || (this.filterTermTypes || []).includes(term?.termType);\n  }\n\n  private update() {\n    this.enableFilterMethodModel && this.updateMethodModels();\n    this.updateImpacts();\n  }\n\n  private updateMethodModels() {\n    const methodModels = this.impactAssessments\n      .flatMap(impact => (impact[this.key] || []).map(v => v.methodModel))\n      .filter(Boolean);\n    this.methodModels = uniqBy(methodModels, '@id');\n  }\n\n  public updateImpacts() {\n    // TODO: when there are no indicators for the default model, the table appears empty\n    // should default back to the first model in this case\n    const filterMethodModel = this.enableFilterMethodModel && this.dataState !== DataState.original;\n    const indicatorPerImpactAssessment = groupNodesByTerm<IImpactAssessmentJSONLD, Indicator>(\n      this.impactAssessments,\n      this.key,\n      this.originalValues,\n      node => this.filterNode(node, filterMethodModel)\n    );\n    this.indicators = orderBy(\n      grouppedKeys(indicatorPerImpactAssessment) as any[],\n      ['value.methodTierOrder', 'key'],\n      ['asc', 'asc']\n    ).filter(({ value }) => this.termAllowed(value?.term));\n  }\n\n  private async loadDataState(dataState: DataState) {\n    this.dataStateValues[dataState] =\n      this.dataStateValues[dataState]?.length === this.impactAssessments?.length\n        ? this.dataStateValues[dataState]\n        : await Promise.all(\n            this.impactAssessments.map(node =>\n              this.nodeService.get<IImpactAssessmentJSONLD>({\n                ...node,\n                dataState\n              })\n            )\n          );\n  }\n\n  private async updateDataState() {\n    await this.loadDataState(this.dataState!);\n    this.impactAssessments = this.dataStateValues[this.dataState!]!.slice();\n    return this.update();\n  }\n\n  public togglePopover(popover, context) {\n    return popover.isOpen() ? popover.close() : popover.open(context);\n  }\n\n  public impactName(impact: IImpactAssessmentJSONLD) {\n    return impact.name || (impact.cycle ? this.cycleLabel(impact.cycle) : impact['@id']) || (impact as any).id;\n  }\n\n  public cycleLabel({ '@id': id, name }: any) {\n    return name || (this.cycles || []).find(v => v['@id'] === id)?.name;\n  }\n\n  public get hasData() {\n    // as we filter by methodModel, we might have an empty table for the default models but not for other models\n    return this.enableFilterMethodModel ? this.methodModels.length : this.indicators.length;\n  }\n\n  public get enableBreakdown() {\n    return !this.isOriginal && this.impactAssessments.length === 1 && this.key !== 'emissionsResourceUse';\n  }\n\n  private filterNode(node: Indicator, filterMethodModel: boolean) {\n    return [\n      !filterMethodModel || isMethodModelAllowed(this.selectedMethodModel)(node),\n      !this.filterTerm || node?.term?.name?.toLowerCase()?.includes(this.filterTerm.toLowerCase())\n    ].every(Boolean);\n  }\n\n  public filter(term: string) {\n    this.filterTerm = term;\n    return this.update();\n  }\n\n  // Recalculation logs\n\n  private updateSelectedIndex(index: number) {\n    this.selectedIndex = -1;\n    // force a refresh of the logs\n    setTimeout(() => (this.selectedIndex = index));\n  }\n\n  public selectIndex({ target: { value } }) {\n    return this.updateSelectedIndex(+value);\n  }\n\n  public async showRecalculationLogs() {\n    this.loading = true;\n    // make sure original data is also loaded\n    await this.loadDataState(DataState.original);\n    this.loading = false;\n    this.selectedView = View.logs;\n  }\n}\n","<div class=\"columns is-variable is-align-items-center is-2 m-0\">\n  <div class=\"column\">\n    <ng-container *ngIf=\"selectedView === View.table && indicators.length\">\n      <button\n        class=\"button is-small is-ghost\"\n        (click)=\"showDownload = true\"\n        ngbTooltip=\"Download as CSV\"\n        placement=\"bottom\">\n        <fa-icon icon=\"download\" size=\"lg\"></fa-icon>\n      </button>\n    </ng-container>\n  </div>\n  <div class=\"column is-narrow\" *ngIf=\"selectedView === View.table\">\n    <he-search-extend\n      class=\"is-secondary is-small\"\n      placeholder=\"Filter indicators by name\"\n      (search)=\"filter($event)\"></he-search-extend>\n  </div>\n  <div class=\"column is-narrow\" *ngIf=\"impactAssessments.length > 1 || enableBreakdown || !isOriginal\">\n    <div class=\"field has-addons\">\n      <div class=\"control\">\n        <button\n          class=\"button is-small\"\n          [class.is-selected]=\"selectedView === View.table\"\n          (click)=\"selectedView = View.table\">\n          <span class=\"icon is-small\">\n            <fa-icon icon=\"list\" aria-hidden=\"true\"></fa-icon>\n          </span>\n          <span class=\"is-hidden-mobile\">Table view</span>\n        </button>\n      </div>\n      <div class=\"control\" *ngIf=\"impactAssessments.length > 1\">\n        <button\n          class=\"button is-small\"\n          [class.is-selected]=\"selectedView === View.chart\"\n          (click)=\"selectedView = View.chart\">\n          <span class=\"icon is-small\">\n            <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n          </span>\n          <span class=\"is-hidden-mobile\">Chart view</span>\n        </button>\n      </div>\n      <div class=\"control\" *ngIf=\"enableBreakdown\">\n        <button\n          class=\"button is-small\"\n          [class.is-selected]=\"selectedView === View.breakdown\"\n          (click)=\"selectedView = View.breakdown\">\n          <span class=\"icon is-small\">\n            <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n          </span>\n          <span class=\"is-hidden-mobile\">Breakdown view</span>\n        </button>\n      </div>\n      <div class=\"control\" *ngIf=\"!isOriginal\">\n        <button\n          class=\"button is-small\"\n          [class.is-selected]=\"selectedView === View.logs\"\n          (click)=\"showRecalculationLogs()\">\n          <span class=\"icon is-small\">\n            <fa-icon icon=\"calculator\" aria-hidden=\"true\"></fa-icon>\n          </span>\n          <span class=\"is-hidden-mobile\">Recalculations logs</span>\n        </button>\n      </div>\n    </div>\n  </div>\n</div>\n\n<div class=\"px-3 pb-3\" [class.is-hidden]=\"selectedView !== View.table\">\n  <ng-container *ngIf=\"hasData; else emptyTable\">\n    <he-data-table class=\"mb-1 is-small\" [small]=\"true\" maxHeight=\"320\">\n      <table class=\"table is-fullwidth is-narrow is-striped\">\n        <thead>\n          <tr class=\"has-text-weight-semibold\">\n            <th class=\"width-auto py-0\">\n              <div class=\"select is-small\" *ngIf=\"enableFilterMethodModel\">\n                <select name=\"selectedMethodModel\" (change)=\"updateImpacts()\" [(ngModel)]=\"selectedMethodModel\">\n                  <option [ngValue]=\"undefined\">Filter Model</option>\n                  <option *ngFor=\"let term of methodModels\" [ngValue]=\"term\">{{ term.name }}</option>\n                </select>\n              </div>\n            </th>\n            <th class=\"has-border-right\"></th>\n            <th class=\"has-border-right\"></th>\n            <th *ngFor=\"let indicator of indicators\" [attr.title]=\"indicator.value.term.name\">\n              <he-node-link [node]=\"indicator.value.term\">\n                <span [innerHtml]=\"indicator.value.term.name | ellipsis: 30 | compound\"></span>\n              </he-node-link>\n            </th>\n          </tr>\n          <tr class=\"is-italic has-text-weight-semibold\">\n            <th class=\"width-auto has-border-right\"></th>\n            <th class=\"has-border-right\">\n              <a [href]=\"baseUrl + '/schema/ImpactAssessment#functionalUnit'\" target=\"_blank\">Functional unit</a>\n            </th>\n            <th class=\"has-border-right\">\n              <a [href]=\"baseUrl + '/schema/ImpactAssessment#product'\" target=\"_blank\">Product</a>\n            </th>\n            <th *ngFor=\"let indicator of indicators\" [attr.title]=\"indicator.value.term.units\">\n              <span [innerHtml]=\"indicator.value.term.units | compound\"></span>\n              <he-terms-units-description\n                class=\"is-inline-block is-ml-2\"\n                [term]=\"indicator.value.term\"></he-terms-units-description>\n            </th>\n          </tr>\n        </thead>\n        <tbody>\n          <ng-container *ngFor=\"let impactAssessment of impactAssessments; trackBy: trackById; let i = index\">\n            <tr>\n              <td class=\"width-auto has-border-right\" [attr.title]=\"impactName(impactAssessment)\">\n                <he-node-link [node]=\"impactAssessment\">\n                  <span>{{ i + 1 }}. {{ impactName(impactAssessment) }}</span>\n                </he-node-link>\n              </td>\n              <td class=\"has-border-right\" [attr.title]=\"impactAssessment.product?.term?.units\">\n                <span>1 {{ impactAssessment.product?.term?.units }}</span>\n              </td>\n              <td class=\"has-border-right\" [attr.title]=\"impactAssessment.product?.term?.name\">\n                <he-node-link *ngIf=\"impactAssessment.product?.term\" [node]=\"impactAssessment.product.term\">\n                  <span>{{ impactAssessment.product.term.name | ellipsis: 30 }}</span>\n                </he-node-link>\n              </td>\n              <td class=\"is-nowrap\" *ngFor=\"let indicator of indicators\">\n                <span\n                  *ngIf=\"indicator.value.values[impactAssessment['@id']]; else emptyValue\"\n                  class=\"trigger-popover\"\n                  [ngbPopover]=\"details\"\n                  [autoClose]=\"'outside'\"\n                  popoverClass=\"is-narrow\"\n                  triggers=\"manual\"\n                  #p=\"ngbPopover\"\n                  placement=\"left\"\n                  container=\"body\"\n                  (click)=\"\n                    togglePopover(p, {\n                      data: indicator.value.values[impactAssessment['@id']],\n                      impactAssessment: impactAssessment,\n                      key: key\n                    })\n                  \">\n                  <span pointer>\n                    {{\n                      propertyValue(indicator.value.values[impactAssessment['@id']].value, indicator.value.term['@id'])\n                        | precision: 3\n                        | default: '-'\n                    }}\n                  </span>\n                  <he-blank-node-state\n                    class=\"ml-1\"\n                    [dataState]=\"dataState\"\n                    [node]=\"indicator.value.values[impactAssessment['@id']].node\"\n                    key=\"value\"\n                    [state]=\"impactAssessment.aggregated ? 'aggregated' : undefined\"></he-blank-node-state>\n                </span>\n              </td>\n            </tr>\n          </ng-container>\n        </tbody>\n      </table>\n    </he-data-table>\n\n    <he-blank-node-state-notice [dataState]=\"dataState\"></he-blank-node-state-notice>\n  </ng-container>\n</div>\n\n<he-impact-assessments-indicator-breakdown-chart\n  *ngIf=\"selectedView === View.breakdown\"\n  [impactAssessment]=\"impactAssessments[0]\"\n  [indicators]=\"impactAssessments[0][key]\"></he-impact-assessments-indicator-breakdown-chart>\n\n<he-impact-assessments-indicators-chart\n  *ngIf=\"selectedView === View.chart && impactAssessments.length > 1\"\n  [key]=\"key\"\n  [impactAssessments]=\"impactAssessments\"\n  [filterTermTypes]=\"filterTermTypes\"></he-impact-assessments-indicators-chart>\n\n<ng-container *ngIf=\"selectedView === View.logs && !isOriginal\">\n  <div class=\"field has-addons pt-2 px-3\" *ngIf=\"impactAssessments.length > 1\">\n    <div class=\"control\">\n      <span class=\"button is-small is-static is-secondary\">Select an Impact Assessment</span>\n    </div>\n    <div class=\"control is-expanded\">\n      <div class=\"select is-small is-fullwidth is-secondary\">\n        <select (change)=\"selectIndex($event)\">\n          <option *ngFor=\"let value of impactAssessments; let i = index\" [value]=\"i\">\n            {{ i + 1 }}. {{ impactName(value) }}\n          </option>\n        </select>\n      </div>\n    </div>\n  </div>\n\n  <he-impact-assessments-products-logs\n    *ngIf=\"selectedIndex >= 0\"\n    [key]=\"key\"\n    [impactAssessment]=\"impactAssessments[selectedIndex]\"\n    [filterTermTypes]=\"filterTermTypes\"\n    [originalValues]=\"originalValues[selectedIndex]?.[key]\"\n    [recalculatedValues]=\"impactAssessments[selectedIndex]?.[key]\"></he-impact-assessments-products-logs>\n</ng-container>\n\n<he-node-csv-export-confirm\n  *ngIf=\"showDownload\"\n  [nodes]=\"impactAssessments\"\n  [filename]=\"'impact-' + key + '.csv'\"\n  [isUpload]=\"false\"\n  [headerKeys]=\"['impactAssessment.id', 'impactAssessment.@id', 'impactAssessment.' + key + '.']\"\n  (closed)=\"showDownload = false\"></he-node-csv-export-confirm>\n\n<ng-template #emptyTable>\n  <div class=\"has-text-centered\">\n    <span>No data available</span>\n    <span class=\"is-pl-1\" [class.is-hidden]=\"!filterTerm\">matching your search criteria</span>\n    <span>.</span>\n    <span [class.is-hidden]=\"!isOriginal\">\n      Switch to\n      <code>recalculated</code>\n      version.\n    </span>\n  </div>\n</ng-template>\n\n<ng-template #emptyValue>\n  <span>-</span>\n</ng-template>\n\n<ng-template #details let-node=\"impactAssessment\" let-data=\"data\" let-key=\"key\">\n  <p *bindOnce=\"node\">\n    <b>\n      <span *ngIf=\"data.cycle\">{{ cycleLabel(node.cycle) }}</span>\n      <span *ngIf=\"!data.cycle\">{{ data.name }}</span>\n    </b>\n  </p>\n  <he-node-value-details\n    [data]=\"data\"\n    [dataState]=\"dataState\"\n    [nodeType]=\"node['@type']\"\n    [dataKey]=\"key\"></he-node-value-details>\n</ng-template>\n\n<ng-template #suggestion let-impact=\"result\" let-t=\"term\">\n  <div class=\"is-block\">\n    <ngb-highlight [result]=\"impact.name || impact.cycle.name\" [term]=\"t\"></ngb-highlight>\n  </div>\n  <div class=\"columns is-flex\">\n    <div class=\"column\" *ngIf=\"impact.country\">\n      <span class=\"pr-1 has-text-underline\">Country:</span>\n      <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.country.name\" [term]=\"t\"></ngb-highlight></span>\n    </div>\n    <div class=\"column\" *ngIf=\"impact.product\">\n      <span class=\"pr-1 has-text-underline\">Product:</span>\n      <span class=\"is-inline-flex\">\n        <ngb-highlight [result]=\"impact.product.term?.name || impact.product.name\" [term]=\"t\"></ngb-highlight>\n      </span>\n    </div>\n    <div class=\"column\" *ngIf=\"impact.endDate\">\n      <span class=\"pr-1 has-text-underline\">Date:</span>\n      <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.endDate\" [term]=\"t\"></ngb-highlight></span>\n    </div>\n  </div>\n</ng-template>\n"]}
|