@hestia-earth/ui-components 0.22.2 → 0.23.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -2,7 +2,7 @@ import { 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 [ErrorKeys.PropertyNotFound, ErrorKeys.SchemaNotFound].includes(this.error?.message);
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"]}
@@ -94,11 +94,11 @@ export class NodeCsvSelectHeadersComponent {
94
94
  });
95
95
  }
96
96
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.8", ngImport: i0, type: NodeCsvSelectHeadersComponent, deps: [{ token: i1.HeSchemaService }], target: i0.ɵɵFactoryTarget.Component }); }
97
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.8", type: NodeCsvSelectHeadersComponent, selector: "he-node-csv-select-headers", inputs: { csv: "csv", keys: "keys", includeDefaultCSV: "includeDefaultCSV" }, outputs: { headersChanged: "headersChanged" }, ngImport: i0, template: "<p class=\"mb-2\">Please select which columns you would like to include:</p>\n\n<p class=\"my-2 is-size-7\">\n <i>You can drag and drop the headers to sort them as they would appear in the CSV file.</i>\n</p>\n\n<div class=\"columns toggle-all mx-4 pb-1 mb-0\">\n <div class=\"column\">\n <label class=\"checkbox ml-1\">\n <input type=\"checkbox\" class=\"selector\" [(ngModel)]=\"allSelected\" (change)=\"toggleAll(); refresh()\" />\n <span class=\"ml-2\">Toggle All</span>\n </label>\n </div>\n\n <div class=\"column is-narrow has-text-right\">\n <div\n class=\"dropdown is-right has-text-left\"\n [class.is-active]=\"showSelectTermFields\"\n (clickOutside)=\"showSelectTermFields = false\">\n <div class=\"dropdown-trigger\" (click)=\"showSelectTermFields = !showSelectTermFields\">\n <button class=\"button is-small\" aria-haspopup=\"true\" aria-controls=\"select-menu\">\n <span>Advanced Filters</span>\n <span class=\"icon is-small\">\n <fa-icon icon=\"filter\" aria-hidden=\"true\"></fa-icon>\n </span>\n </button>\n </div>\n <div class=\"dropdown-menu\" id=\"select-menu\" role=\"menu\">\n <div class=\"dropdown-content\">\n <a class=\"dropdown-item\">\n <label class=\"checkbox ml-1\">\n <input type=\"checkbox\" class=\"selector\" [(ngModel)]=\"showNonIncluded\" (change)=\"refresh()\" />\n <span class=\"ml-2\">\n Include\n <code>internal</code>\n fields\n </span>\n </label>\n </a>\n <div class=\"dropdown-item\">\n <p>Toggle Term Fields</p>\n </div>\n <a class=\"dropdown-item\" *ngFor=\"let field of termFields\">\n <label class=\"checkbox ml-1\">\n <input\n type=\"checkbox\"\n class=\"selector\"\n [(ngModel)]=\"termFieldSelected[field]\"\n (change)=\"toggleTermField(field)\" />\n <span class=\"ml-2\">{{ field }}</span>\n </label>\n </a>\n </div>\n </div>\n </div>\n </div>\n</div>\n\n<div class=\"drag-container\" *ngIf=\"!loading; else loader\">\n <div class=\"card\" *ngFor=\"let group of headers | keys\">\n <header class=\"card-header\" *ngIf=\"group.key\">\n <div class=\"card-header-title\">\n <label class=\"checkbox ml-1\">\n <input\n type=\"checkbox\"\n class=\"selector\"\n [indeterminate]=\"group.value.partialSelected\"\n [(ngModel)]=\"group.value.selected\"\n (change)=\"updateGroup(group.key); refresh()\" />\n <span class=\"ml-2\">{{ group.key }}</span>\n </label>\n </div>\n <span\n class=\"card-header-icon has-text-secondary\"\n aria-label=\"open / close\"\n (click)=\"group.value.open = !group.value.open\"\n pointer>\n <span class=\"icon\">\n <fa-icon icon=\"angle-down\" [class.is-hidden]=\"!group.value.open\"></fa-icon>\n <fa-icon icon=\"angle-left\" [class.is-hidden]=\"group.value.open\"></fa-icon>\n </span>\n </span>\n </header>\n\n <div class=\"card-content p-3\" [class.is-hidden]=\"!group.value.open\">\n <ul cdkDropList (cdkDropListDropped)=\"dropHeader($event, group.key)\">\n <ng-container *ngFor=\"let header of group.value.headers\">\n <ng-container *ngIf=\"header.included || showNonIncluded\">\n <li cdkDrag>\n <label class=\"checkbox ml-2\">\n <input\n type=\"checkbox\"\n class=\"selector\"\n [(ngModel)]=\"header.selected\"\n (change)=\"updateGroupHeader(group.key); refresh()\" />\n <span class=\"ml-2\">{{ header.header }}</span>\n </label>\n </li>\n </ng-container>\n </ng-container>\n </ul>\n </div>\n </div>\n</div>\n\n<ng-template #loader>\n <div class=\"has-text-center py-3\">\n <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n </div>\n</ng-template>\n", styles: [".toggle-all{border-bottom:1px solid #4A4A4A}.drag-container,.table-container{max-height:45vh;overflow-y:auto}.cdk-drag-preview{border-radius:3px;box-shadow:0 5px 5px -3px #0003,0 8px 10px 1px #00000024,0 3px 14px 2px #0000001f;list-style:none}.cdk-drag-placeholder{opacity:0}.cdk-drag-animating{transition:transform .25s cubic-bezier(0,0,.2,1)}\n"], dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.CheckboxControlValueAccessor, selector: "input[type=checkbox][formControlName],input[type=checkbox][formControl],input[type=checkbox][ngModel]" }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: i4.CdkDropList, selector: "[cdkDropList], cdk-drop-list", inputs: ["cdkDropListConnectedTo", "cdkDropListData", "cdkDropListOrientation", "id", "cdkDropListLockAxis", "cdkDropListDisabled", "cdkDropListSortingDisabled", "cdkDropListEnterPredicate", "cdkDropListSortPredicate", "cdkDropListAutoScrollDisabled", "cdkDropListAutoScrollStep"], outputs: ["cdkDropListDropped", "cdkDropListEntered", "cdkDropListExited", "cdkDropListSorted"], exportAs: ["cdkDropList"] }, { kind: "directive", type: i4.CdkDrag, selector: "[cdkDrag]", inputs: ["cdkDragData", "cdkDragLockAxis", "cdkDragRootElement", "cdkDragBoundary", "cdkDragStartDelay", "cdkDragFreeDragPosition", "cdkDragDisabled", "cdkDragConstrainPosition", "cdkDragPreviewClass", "cdkDragPreviewContainer"], outputs: ["cdkDragStarted", "cdkDragReleased", "cdkDragEnded", "cdkDragEntered", "cdkDragExited", "cdkDragDropped", "cdkDragMoved"], exportAs: ["cdkDrag"] }, { kind: "component", type: i5.FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "spin", "pulse", "mask", "styles", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "classes", "transform", "a11yRole"] }, { kind: "directive", type: i6.ClickOutsideDirective, selector: "[clickOutside]", outputs: ["clickOutside"] }, { kind: "pipe", type: i7.KeysPipe, name: "keys" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
97
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.8", type: NodeCsvSelectHeadersComponent, selector: "he-node-csv-select-headers", inputs: { csv: "csv", keys: "keys", includeDefaultCSV: "includeDefaultCSV" }, outputs: { headersChanged: "headersChanged" }, ngImport: i0, template: "<p class=\"mb-2\">Please select which columns you would like to include:</p>\n\n<p class=\"my-2 is-size-7\">\n <i>You can drag and drop the headers to sort them as they would appear in the CSV file.</i>\n</p>\n\n<div class=\"columns toggle-all mx-4 pb-1 mb-0\">\n <div class=\"column\">\n <label class=\"checkbox ml-1\">\n <input type=\"checkbox\" class=\"selector\" [(ngModel)]=\"allSelected\" (change)=\"toggleAll(); refresh()\" />\n <span class=\"ml-2\">Toggle All</span>\n </label>\n </div>\n\n <div class=\"column is-narrow has-text-right\">\n <div\n class=\"dropdown is-right has-text-left\"\n [class.is-active]=\"showSelectTermFields\"\n (clickOutside)=\"showSelectTermFields = false\">\n <div class=\"dropdown-trigger\" (click)=\"showSelectTermFields = !showSelectTermFields\">\n <button class=\"button is-small\" aria-haspopup=\"true\" aria-controls=\"select-menu\">\n <span>Advanced Filters</span>\n <span class=\"icon is-small\">\n <fa-icon icon=\"filter\" aria-hidden=\"true\"></fa-icon>\n </span>\n </button>\n </div>\n <div class=\"dropdown-menu\" id=\"select-menu\" role=\"menu\">\n <div class=\"dropdown-content\">\n <a class=\"dropdown-item\">\n <label class=\"checkbox ml-1\">\n <input type=\"checkbox\" class=\"selector\" [(ngModel)]=\"showNonIncluded\" (change)=\"refresh()\" />\n <span class=\"ml-2\">\n Include\n <code>internal</code>\n fields\n </span>\n </label>\n </a>\n <div class=\"dropdown-item\">\n <p>Toggle Term Fields</p>\n </div>\n <a class=\"dropdown-item\" *ngFor=\"let field of termFields\">\n <label class=\"checkbox ml-1\">\n <input\n type=\"checkbox\"\n class=\"selector\"\n [(ngModel)]=\"termFieldSelected[field]\"\n (change)=\"toggleTermField(field)\" />\n <span class=\"ml-2\">{{ field }}</span>\n </label>\n </a>\n </div>\n </div>\n </div>\n </div>\n</div>\n\n<div class=\"drag-container\" *ngIf=\"!loading; else loader\">\n <div class=\"card\" *ngFor=\"let group of headers | keys\">\n <header class=\"card-header\" *ngIf=\"group.key\">\n <div class=\"card-header-title\">\n <label class=\"checkbox ml-1\">\n <input\n type=\"checkbox\"\n class=\"selector\"\n [indeterminate]=\"group.value.partialSelected\"\n [(ngModel)]=\"group.value.selected\"\n (change)=\"updateGroup(group.key); refresh()\" />\n <span class=\"ml-2\">{{ group.key }}</span>\n </label>\n </div>\n <span\n class=\"card-header-icon has-text-secondary\"\n aria-label=\"open / close\"\n (click)=\"group.value.open = !group.value.open\"\n pointer>\n <span class=\"icon\">\n <fa-icon icon=\"angle-down\" [class.is-hidden]=\"!group.value.open\"></fa-icon>\n <fa-icon icon=\"angle-left\" [class.is-hidden]=\"group.value.open\"></fa-icon>\n </span>\n </span>\n </header>\n\n <div class=\"card-content p-3\" [class.is-hidden]=\"!group.value.open\">\n <ul cdkDropList (cdkDropListDropped)=\"dropHeader($event, group.key)\">\n <ng-container *ngFor=\"let header of group.value.headers\">\n <ng-container *ngIf=\"header.included || showNonIncluded\">\n <li cdkDrag>\n <label class=\"checkbox ml-2\">\n <input\n type=\"checkbox\"\n class=\"selector\"\n [(ngModel)]=\"header.selected\"\n (change)=\"updateGroupHeader(group.key); refresh()\" />\n <span class=\"ml-2\">{{ header.header }}</span>\n </label>\n </li>\n </ng-container>\n </ng-container>\n </ul>\n </div>\n </div>\n</div>\n\n<ng-template #loader>\n <div class=\"has-text-center py-3\">\n <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n </div>\n</ng-template>\n", styles: [".toggle-all{border-bottom:1px solid #4a4a4a}.drag-container,.table-container{max-height:45vh;overflow-y:auto}.cdk-drag-preview{border-radius:3px;box-shadow:0 5px 5px -3px #0003,0 8px 10px 1px #00000024,0 3px 14px 2px #0000001f;list-style:none}.cdk-drag-placeholder{opacity:0}.cdk-drag-animating{transition:transform .25s cubic-bezier(0,0,.2,1)}\n"], dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.CheckboxControlValueAccessor, selector: "input[type=checkbox][formControlName],input[type=checkbox][formControl],input[type=checkbox][ngModel]" }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: i4.CdkDropList, selector: "[cdkDropList], cdk-drop-list", inputs: ["cdkDropListConnectedTo", "cdkDropListData", "cdkDropListOrientation", "id", "cdkDropListLockAxis", "cdkDropListDisabled", "cdkDropListSortingDisabled", "cdkDropListEnterPredicate", "cdkDropListSortPredicate", "cdkDropListAutoScrollDisabled", "cdkDropListAutoScrollStep"], outputs: ["cdkDropListDropped", "cdkDropListEntered", "cdkDropListExited", "cdkDropListSorted"], exportAs: ["cdkDropList"] }, { kind: "directive", type: i4.CdkDrag, selector: "[cdkDrag]", inputs: ["cdkDragData", "cdkDragLockAxis", "cdkDragRootElement", "cdkDragBoundary", "cdkDragStartDelay", "cdkDragFreeDragPosition", "cdkDragDisabled", "cdkDragConstrainPosition", "cdkDragPreviewClass", "cdkDragPreviewContainer"], outputs: ["cdkDragStarted", "cdkDragReleased", "cdkDragEnded", "cdkDragEntered", "cdkDragExited", "cdkDragDropped", "cdkDragMoved"], exportAs: ["cdkDrag"] }, { kind: "component", type: i5.FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "spin", "pulse", "mask", "styles", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "classes", "transform", "a11yRole"] }, { kind: "directive", type: i6.ClickOutsideDirective, selector: "[clickOutside]", outputs: ["clickOutside"] }, { kind: "pipe", type: i7.KeysPipe, name: "keys" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
98
98
  }
99
99
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.8", ngImport: i0, type: NodeCsvSelectHeadersComponent, decorators: [{
100
100
  type: Component,
101
- args: [{ selector: 'he-node-csv-select-headers', changeDetection: ChangeDetectionStrategy.OnPush, template: "<p class=\"mb-2\">Please select which columns you would like to include:</p>\n\n<p class=\"my-2 is-size-7\">\n <i>You can drag and drop the headers to sort them as they would appear in the CSV file.</i>\n</p>\n\n<div class=\"columns toggle-all mx-4 pb-1 mb-0\">\n <div class=\"column\">\n <label class=\"checkbox ml-1\">\n <input type=\"checkbox\" class=\"selector\" [(ngModel)]=\"allSelected\" (change)=\"toggleAll(); refresh()\" />\n <span class=\"ml-2\">Toggle All</span>\n </label>\n </div>\n\n <div class=\"column is-narrow has-text-right\">\n <div\n class=\"dropdown is-right has-text-left\"\n [class.is-active]=\"showSelectTermFields\"\n (clickOutside)=\"showSelectTermFields = false\">\n <div class=\"dropdown-trigger\" (click)=\"showSelectTermFields = !showSelectTermFields\">\n <button class=\"button is-small\" aria-haspopup=\"true\" aria-controls=\"select-menu\">\n <span>Advanced Filters</span>\n <span class=\"icon is-small\">\n <fa-icon icon=\"filter\" aria-hidden=\"true\"></fa-icon>\n </span>\n </button>\n </div>\n <div class=\"dropdown-menu\" id=\"select-menu\" role=\"menu\">\n <div class=\"dropdown-content\">\n <a class=\"dropdown-item\">\n <label class=\"checkbox ml-1\">\n <input type=\"checkbox\" class=\"selector\" [(ngModel)]=\"showNonIncluded\" (change)=\"refresh()\" />\n <span class=\"ml-2\">\n Include\n <code>internal</code>\n fields\n </span>\n </label>\n </a>\n <div class=\"dropdown-item\">\n <p>Toggle Term Fields</p>\n </div>\n <a class=\"dropdown-item\" *ngFor=\"let field of termFields\">\n <label class=\"checkbox ml-1\">\n <input\n type=\"checkbox\"\n class=\"selector\"\n [(ngModel)]=\"termFieldSelected[field]\"\n (change)=\"toggleTermField(field)\" />\n <span class=\"ml-2\">{{ field }}</span>\n </label>\n </a>\n </div>\n </div>\n </div>\n </div>\n</div>\n\n<div class=\"drag-container\" *ngIf=\"!loading; else loader\">\n <div class=\"card\" *ngFor=\"let group of headers | keys\">\n <header class=\"card-header\" *ngIf=\"group.key\">\n <div class=\"card-header-title\">\n <label class=\"checkbox ml-1\">\n <input\n type=\"checkbox\"\n class=\"selector\"\n [indeterminate]=\"group.value.partialSelected\"\n [(ngModel)]=\"group.value.selected\"\n (change)=\"updateGroup(group.key); refresh()\" />\n <span class=\"ml-2\">{{ group.key }}</span>\n </label>\n </div>\n <span\n class=\"card-header-icon has-text-secondary\"\n aria-label=\"open / close\"\n (click)=\"group.value.open = !group.value.open\"\n pointer>\n <span class=\"icon\">\n <fa-icon icon=\"angle-down\" [class.is-hidden]=\"!group.value.open\"></fa-icon>\n <fa-icon icon=\"angle-left\" [class.is-hidden]=\"group.value.open\"></fa-icon>\n </span>\n </span>\n </header>\n\n <div class=\"card-content p-3\" [class.is-hidden]=\"!group.value.open\">\n <ul cdkDropList (cdkDropListDropped)=\"dropHeader($event, group.key)\">\n <ng-container *ngFor=\"let header of group.value.headers\">\n <ng-container *ngIf=\"header.included || showNonIncluded\">\n <li cdkDrag>\n <label class=\"checkbox ml-2\">\n <input\n type=\"checkbox\"\n class=\"selector\"\n [(ngModel)]=\"header.selected\"\n (change)=\"updateGroupHeader(group.key); refresh()\" />\n <span class=\"ml-2\">{{ header.header }}</span>\n </label>\n </li>\n </ng-container>\n </ng-container>\n </ul>\n </div>\n </div>\n</div>\n\n<ng-template #loader>\n <div class=\"has-text-center py-3\">\n <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n </div>\n</ng-template>\n", styles: [".toggle-all{border-bottom:1px solid #4A4A4A}.drag-container,.table-container{max-height:45vh;overflow-y:auto}.cdk-drag-preview{border-radius:3px;box-shadow:0 5px 5px -3px #0003,0 8px 10px 1px #00000024,0 3px 14px 2px #0000001f;list-style:none}.cdk-drag-placeholder{opacity:0}.cdk-drag-animating{transition:transform .25s cubic-bezier(0,0,.2,1)}\n"] }]
101
+ args: [{ selector: 'he-node-csv-select-headers', changeDetection: ChangeDetectionStrategy.OnPush, template: "<p class=\"mb-2\">Please select which columns you would like to include:</p>\n\n<p class=\"my-2 is-size-7\">\n <i>You can drag and drop the headers to sort them as they would appear in the CSV file.</i>\n</p>\n\n<div class=\"columns toggle-all mx-4 pb-1 mb-0\">\n <div class=\"column\">\n <label class=\"checkbox ml-1\">\n <input type=\"checkbox\" class=\"selector\" [(ngModel)]=\"allSelected\" (change)=\"toggleAll(); refresh()\" />\n <span class=\"ml-2\">Toggle All</span>\n </label>\n </div>\n\n <div class=\"column is-narrow has-text-right\">\n <div\n class=\"dropdown is-right has-text-left\"\n [class.is-active]=\"showSelectTermFields\"\n (clickOutside)=\"showSelectTermFields = false\">\n <div class=\"dropdown-trigger\" (click)=\"showSelectTermFields = !showSelectTermFields\">\n <button class=\"button is-small\" aria-haspopup=\"true\" aria-controls=\"select-menu\">\n <span>Advanced Filters</span>\n <span class=\"icon is-small\">\n <fa-icon icon=\"filter\" aria-hidden=\"true\"></fa-icon>\n </span>\n </button>\n </div>\n <div class=\"dropdown-menu\" id=\"select-menu\" role=\"menu\">\n <div class=\"dropdown-content\">\n <a class=\"dropdown-item\">\n <label class=\"checkbox ml-1\">\n <input type=\"checkbox\" class=\"selector\" [(ngModel)]=\"showNonIncluded\" (change)=\"refresh()\" />\n <span class=\"ml-2\">\n Include\n <code>internal</code>\n fields\n </span>\n </label>\n </a>\n <div class=\"dropdown-item\">\n <p>Toggle Term Fields</p>\n </div>\n <a class=\"dropdown-item\" *ngFor=\"let field of termFields\">\n <label class=\"checkbox ml-1\">\n <input\n type=\"checkbox\"\n class=\"selector\"\n [(ngModel)]=\"termFieldSelected[field]\"\n (change)=\"toggleTermField(field)\" />\n <span class=\"ml-2\">{{ field }}</span>\n </label>\n </a>\n </div>\n </div>\n </div>\n </div>\n</div>\n\n<div class=\"drag-container\" *ngIf=\"!loading; else loader\">\n <div class=\"card\" *ngFor=\"let group of headers | keys\">\n <header class=\"card-header\" *ngIf=\"group.key\">\n <div class=\"card-header-title\">\n <label class=\"checkbox ml-1\">\n <input\n type=\"checkbox\"\n class=\"selector\"\n [indeterminate]=\"group.value.partialSelected\"\n [(ngModel)]=\"group.value.selected\"\n (change)=\"updateGroup(group.key); refresh()\" />\n <span class=\"ml-2\">{{ group.key }}</span>\n </label>\n </div>\n <span\n class=\"card-header-icon has-text-secondary\"\n aria-label=\"open / close\"\n (click)=\"group.value.open = !group.value.open\"\n pointer>\n <span class=\"icon\">\n <fa-icon icon=\"angle-down\" [class.is-hidden]=\"!group.value.open\"></fa-icon>\n <fa-icon icon=\"angle-left\" [class.is-hidden]=\"group.value.open\"></fa-icon>\n </span>\n </span>\n </header>\n\n <div class=\"card-content p-3\" [class.is-hidden]=\"!group.value.open\">\n <ul cdkDropList (cdkDropListDropped)=\"dropHeader($event, group.key)\">\n <ng-container *ngFor=\"let header of group.value.headers\">\n <ng-container *ngIf=\"header.included || showNonIncluded\">\n <li cdkDrag>\n <label class=\"checkbox ml-2\">\n <input\n type=\"checkbox\"\n class=\"selector\"\n [(ngModel)]=\"header.selected\"\n (change)=\"updateGroupHeader(group.key); refresh()\" />\n <span class=\"ml-2\">{{ header.header }}</span>\n </label>\n </li>\n </ng-container>\n </ng-container>\n </ul>\n </div>\n </div>\n</div>\n\n<ng-template #loader>\n <div class=\"has-text-center py-3\">\n <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n </div>\n</ng-template>\n", styles: [".toggle-all{border-bottom:1px solid #4a4a4a}.drag-container,.table-container{max-height:45vh;overflow-y:auto}.cdk-drag-preview{border-radius:3px;box-shadow:0 5px 5px -3px #0003,0 8px 10px 1px #00000024,0 3px 14px 2px #0000001f;list-style:none}.cdk-drag-placeholder{opacity:0}.cdk-drag-animating{transition:transform .25s cubic-bezier(0,0,.2,1)}\n"] }]
102
102
  }], ctorParameters: function () { return [{ type: i1.HeSchemaService }]; }, propDecorators: { csv: [{
103
103
  type: Input
104
104
  }], keys: [{
@@ -248,7 +248,7 @@ export class SelectComponent extends ControlValueAccessor {
248
248
  provide: SELECT_TOKEN,
249
249
  useExisting: forwardRef(() => SelectComponent)
250
250
  }
251
- ], queries: [{ propertyName: "nodes", predicate: NODE_ELEMENT_TOKEN, descendants: true }, { propertyName: "options", predicate: SELECT_OPTION_TOKEN, descendants: true }], viewQueries: [{ propertyName: "searchInputEl", first: true, predicate: ["input"], descendants: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<div #origin=\"cdkOverlayOrigin\" cdkOverlayOrigin>\n <ng-container\n *ngTemplateOutlet=\"\n selectTemplate ?? defaultSelectTemplate;\n context: {\n $implicit: this,\n value,\n displayValue,\n labelState,\n icon,\n placeholder,\n searchPlaceholder,\n width,\n valueTemplate\n }\n \"></ng-container>\n</div>\n\n<ng-template\n (backdropClick)=\"close()\"\n (detach)=\"close()\"\n [cdkConnectedOverlayOffsetX]=\"overlayOptions?.offsetX ?? -11\"\n [cdkConnectedOverlayOffsetY]=\"overlayOptions?.offsetY ?? 4\"\n [cdkConnectedOverlayOpen]=\"isOpen\"\n [cdkConnectedOverlayOrigin]=\"origin\"\n cdkConnectedOverlay\n cdkConnectedOverlayBackdropClass=\"cdk-overlay-transparent-backdrop\"\n cdkConnectedOverlayHasBackdrop>\n <div [style.min-width]=\"width\" class=\"select-panel\">\n <ng-content></ng-content>\n </div>\n</ng-template>\n\n<ng-template #defaultSelectTemplate>\n <div class=\"select-host\">\n <div class=\"control | is-flex is-justify-content-space-between is-align-items-center\">\n <ng-container [ngSwitch]=\"labelState\">\n <input\n #input\n (input)=\"onHandleInput($event)\"\n *ngSwitchCase=\"'search'\"\n [placeholder]=\"searchPlaceholder\"\n class=\"select-search | is-transparent\" />\n\n <ng-container *ngSwitchCase=\"'value'\">\n <ng-container\n *ngTemplateOutlet=\"valueTemplate ?? defaultValueTemplate; context: { $implicit: value }\"></ng-container>\n </ng-container>\n </ng-container>\n\n <span class=\"drop-down-icon ml-2\">\n <fa-icon [icon]=\"icon\"></fa-icon>\n </span>\n </div>\n </div>\n</ng-template>\n\n<ng-template #defaultValueTemplate>\n <span [ngClass]=\"{ empty: !value }\" class=\"selected-value\">{{ displayValue || placeholder }}</span>\n</ng-template>\n", styles: ["body.is-os-windows::-webkit-scrollbar,body.is-os-windows *::-webkit-scrollbar{border-radius:10px;height:10px;width:10px;border:none;background:transparent}body.is-os-windows::-webkit-scrollbar-button,body.is-os-windows *::-webkit-scrollbar-button{background:#DDE1E6;color:#dde1e6}body.is-os-windows::-webkit-scrollbar-thumb,body.is-os-windows *::-webkit-scrollbar-thumb{border-radius:10px;background:#DDE1E6}body.is-os-windows::-webkit-scrollbar-thumb:hover,body.is-os-windows *::-webkit-scrollbar-thumb:hover{background:#DDE1E6}body.is-os-windows::-webkit-scrollbar-track,body.is-os-windows *::-webkit-scrollbar-track{border:none;background-color:transparent;background-repeat:no-repeat;background-image:url(https://cdn.hestia.earth/prod/assets/images/scrollbar-thumb.png);background-position:center}body.is-os-windows::-webkit-scrollbar-track:vertical,body.is-os-windows *::-webkit-scrollbar-track:vertical{background-repeat:repeat-y}body.is-os-windows::-webkit-scrollbar-track:horizontal,body.is-os-windows *::-webkit-scrollbar-track:horizontal{background-repeat:repeat-x}body.is-os-windows .is-dark::-webkit-scrollbar-button{background:#4C7194;color:#4c7194}body.is-os-windows .is-dark::-webkit-scrollbar-thumb{background:#4C7194}body.is-os-windows .is-dark::-webkit-scrollbar-thumb:hover{background:#4C7194}::ng-deep:root{--select-border-radius: 6px;--select-min-width: 250px;--select-shadow: rgba(0, 0, 0, .1) 0px 4px 6px -1px, rgba(0, 0, 0, .06) 0px 2px 4px -1px}.select-host{display:flex;justify-content:flex-start;flex-direction:column;padding:5px 10px;box-sizing:border-box;min-width:var(--select-min-width);border-radius:var(--select-border-radius);border:1px solid #F5F5F5;background:hsl(0,0%,100%);box-shadow:var(--select-shadow)}.select-host:hover{cursor:pointer;background-color:var(--color-background, #eceff1)}.select-host.select-panel-open{position:relative;z-index:1001;border-radius:var(--select-border-radius) var(--select-border-radius) 0 0;box-shadow:none}.select-host.is-disabled{opacity:.5;pointer-events:none;-webkit-user-select:none;user-select:none}.selected-value.empty{opacity:.6}.select-search{border:none;outline:none;background:transparent}.select-panel{box-sizing:border-box;border-radius:0 0 var(--select-border-radius, 6px) var(--select-border-radius, 6px);border:1px solid #F5F5F5;background-color:#fff;transform-origin:top;max-height:350px;overflow:auto;box-shadow:var(--select-shadow)}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { 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: "directive", type: i2.CdkConnectedOverlay, selector: "[cdk-connected-overlay], [connected-overlay], [cdkConnectedOverlay]", inputs: ["cdkConnectedOverlayOrigin", "cdkConnectedOverlayPositions", "cdkConnectedOverlayPositionStrategy", "cdkConnectedOverlayOffsetX", "cdkConnectedOverlayOffsetY", "cdkConnectedOverlayWidth", "cdkConnectedOverlayHeight", "cdkConnectedOverlayMinWidth", "cdkConnectedOverlayMinHeight", "cdkConnectedOverlayBackdropClass", "cdkConnectedOverlayPanelClass", "cdkConnectedOverlayViewportMargin", "cdkConnectedOverlayScrollStrategy", "cdkConnectedOverlayOpen", "cdkConnectedOverlayDisableClose", "cdkConnectedOverlayTransformOriginOn", "cdkConnectedOverlayHasBackdrop", "cdkConnectedOverlayLockPosition", "cdkConnectedOverlayFlexibleDimensions", "cdkConnectedOverlayGrowAfterOpen", "cdkConnectedOverlayPush"], outputs: ["backdropClick", "positionChange", "attach", "detach", "overlayKeydown", "overlayOutsideClick"], exportAs: ["cdkConnectedOverlay"] }, { kind: "directive", type: i2.CdkOverlayOrigin, selector: "[cdk-overlay-origin], [overlay-origin], [cdkOverlayOrigin]", exportAs: ["cdkOverlayOrigin"] }, { kind: "component", type: i3.FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "spin", "pulse", "mask", "styles", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "classes", "transform", "a11yRole"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
251
+ ], queries: [{ propertyName: "nodes", predicate: NODE_ELEMENT_TOKEN, descendants: true }, { propertyName: "options", predicate: SELECT_OPTION_TOKEN, descendants: true }], viewQueries: [{ propertyName: "searchInputEl", first: true, predicate: ["input"], descendants: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<div #origin=\"cdkOverlayOrigin\" cdkOverlayOrigin>\n <ng-container\n *ngTemplateOutlet=\"\n selectTemplate ?? defaultSelectTemplate;\n context: {\n $implicit: this,\n value,\n displayValue,\n labelState,\n icon,\n placeholder,\n searchPlaceholder,\n width,\n valueTemplate\n }\n \"></ng-container>\n</div>\n\n<ng-template\n (backdropClick)=\"close()\"\n (detach)=\"close()\"\n [cdkConnectedOverlayOffsetX]=\"overlayOptions?.offsetX ?? -11\"\n [cdkConnectedOverlayOffsetY]=\"overlayOptions?.offsetY ?? 4\"\n [cdkConnectedOverlayOpen]=\"isOpen\"\n [cdkConnectedOverlayOrigin]=\"origin\"\n cdkConnectedOverlay\n cdkConnectedOverlayBackdropClass=\"cdk-overlay-transparent-backdrop\"\n cdkConnectedOverlayHasBackdrop>\n <div [style.min-width]=\"width\" class=\"select-panel\">\n <ng-content></ng-content>\n </div>\n</ng-template>\n\n<ng-template #defaultSelectTemplate>\n <div class=\"select-host\">\n <div class=\"control | is-flex is-justify-content-space-between is-align-items-center\">\n <ng-container [ngSwitch]=\"labelState\">\n <input\n #input\n (input)=\"onHandleInput($event)\"\n *ngSwitchCase=\"'search'\"\n [placeholder]=\"searchPlaceholder\"\n class=\"select-search | is-transparent\" />\n\n <ng-container *ngSwitchCase=\"'value'\">\n <ng-container\n *ngTemplateOutlet=\"valueTemplate ?? defaultValueTemplate; context: { $implicit: value }\"></ng-container>\n </ng-container>\n </ng-container>\n\n <span class=\"drop-down-icon ml-2\">\n <fa-icon [icon]=\"icon\"></fa-icon>\n </span>\n </div>\n </div>\n</ng-template>\n\n<ng-template #defaultValueTemplate>\n <span [ngClass]=\"{ empty: !value }\" class=\"selected-value\">{{ displayValue || placeholder }}</span>\n</ng-template>\n", styles: ["body.is-os-windows::-webkit-scrollbar,body.is-os-windows *::-webkit-scrollbar{border-radius:10px;height:10px;width:10px;border:none;background:transparent}body.is-os-windows::-webkit-scrollbar-button,body.is-os-windows *::-webkit-scrollbar-button{background:#dde1e6;color:#dde1e6}body.is-os-windows::-webkit-scrollbar-thumb,body.is-os-windows *::-webkit-scrollbar-thumb{border-radius:10px;background:#dde1e6}body.is-os-windows::-webkit-scrollbar-thumb:hover,body.is-os-windows *::-webkit-scrollbar-thumb:hover{background:#dde1e6}body.is-os-windows::-webkit-scrollbar-track,body.is-os-windows *::-webkit-scrollbar-track{border:none;background-color:transparent;background-repeat:no-repeat;background-image:url(https://cdn.hestia.earth/prod/assets/images/scrollbar-thumb.png);background-position:center}body.is-os-windows::-webkit-scrollbar-track:vertical,body.is-os-windows *::-webkit-scrollbar-track:vertical{background-repeat:repeat-y}body.is-os-windows::-webkit-scrollbar-track:horizontal,body.is-os-windows *::-webkit-scrollbar-track:horizontal{background-repeat:repeat-x}body.is-os-windows .is-dark::-webkit-scrollbar-button{background:#4c7194;color:#4c7194}body.is-os-windows .is-dark::-webkit-scrollbar-thumb{background:#4c7194}body.is-os-windows .is-dark::-webkit-scrollbar-thumb:hover{background:#4c7194}::ng-deep:root{--select-border-radius: 6px;--select-min-width: 250px;--select-shadow: rgba(0, 0, 0, .1) 0px 4px 6px -1px, rgba(0, 0, 0, .06) 0px 2px 4px -1px}.select-host{display:flex;justify-content:flex-start;flex-direction:column;padding:5px 10px;box-sizing:border-box;min-width:var(--select-min-width);border-radius:var(--select-border-radius);border:1px solid #f5f5f5;background:hsl(0,0%,100%);box-shadow:var(--select-shadow)}.select-host:hover{cursor:pointer;background-color:var(--color-background, #eceff1)}.select-host.select-panel-open{position:relative;z-index:1001;border-radius:var(--select-border-radius) var(--select-border-radius) 0 0;box-shadow:none}.select-host.is-disabled{opacity:.5;pointer-events:none;-webkit-user-select:none;user-select:none}.selected-value.empty{opacity:.6}.select-search{border:none;outline:none;background:transparent}.select-panel{box-sizing:border-box;border-radius:0 0 var(--select-border-radius, 6px) var(--select-border-radius, 6px);border:1px solid #f5f5f5;background-color:#fff;transform-origin:top;max-height:350px;overflow:auto;box-shadow:var(--select-shadow)}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { 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: "directive", type: i2.CdkConnectedOverlay, selector: "[cdk-connected-overlay], [connected-overlay], [cdkConnectedOverlay]", inputs: ["cdkConnectedOverlayOrigin", "cdkConnectedOverlayPositions", "cdkConnectedOverlayPositionStrategy", "cdkConnectedOverlayOffsetX", "cdkConnectedOverlayOffsetY", "cdkConnectedOverlayWidth", "cdkConnectedOverlayHeight", "cdkConnectedOverlayMinWidth", "cdkConnectedOverlayMinHeight", "cdkConnectedOverlayBackdropClass", "cdkConnectedOverlayPanelClass", "cdkConnectedOverlayViewportMargin", "cdkConnectedOverlayScrollStrategy", "cdkConnectedOverlayOpen", "cdkConnectedOverlayDisableClose", "cdkConnectedOverlayTransformOriginOn", "cdkConnectedOverlayHasBackdrop", "cdkConnectedOverlayLockPosition", "cdkConnectedOverlayFlexibleDimensions", "cdkConnectedOverlayGrowAfterOpen", "cdkConnectedOverlayPush"], outputs: ["backdropClick", "positionChange", "attach", "detach", "overlayKeydown", "overlayOutsideClick"], exportAs: ["cdkConnectedOverlay"] }, { kind: "directive", type: i2.CdkOverlayOrigin, selector: "[cdk-overlay-origin], [overlay-origin], [cdkOverlayOrigin]", exportAs: ["cdkOverlayOrigin"] }, { kind: "component", type: i3.FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "spin", "pulse", "mask", "styles", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "classes", "transform", "a11yRole"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
252
252
  }
253
253
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.8", ngImport: i0, type: SelectComponent, decorators: [{
254
254
  type: Component,
@@ -262,7 +262,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.8", ngImpor
262
262
  provide: SELECT_TOKEN,
263
263
  useExisting: forwardRef(() => SelectComponent)
264
264
  }
265
- ], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div #origin=\"cdkOverlayOrigin\" cdkOverlayOrigin>\n <ng-container\n *ngTemplateOutlet=\"\n selectTemplate ?? defaultSelectTemplate;\n context: {\n $implicit: this,\n value,\n displayValue,\n labelState,\n icon,\n placeholder,\n searchPlaceholder,\n width,\n valueTemplate\n }\n \"></ng-container>\n</div>\n\n<ng-template\n (backdropClick)=\"close()\"\n (detach)=\"close()\"\n [cdkConnectedOverlayOffsetX]=\"overlayOptions?.offsetX ?? -11\"\n [cdkConnectedOverlayOffsetY]=\"overlayOptions?.offsetY ?? 4\"\n [cdkConnectedOverlayOpen]=\"isOpen\"\n [cdkConnectedOverlayOrigin]=\"origin\"\n cdkConnectedOverlay\n cdkConnectedOverlayBackdropClass=\"cdk-overlay-transparent-backdrop\"\n cdkConnectedOverlayHasBackdrop>\n <div [style.min-width]=\"width\" class=\"select-panel\">\n <ng-content></ng-content>\n </div>\n</ng-template>\n\n<ng-template #defaultSelectTemplate>\n <div class=\"select-host\">\n <div class=\"control | is-flex is-justify-content-space-between is-align-items-center\">\n <ng-container [ngSwitch]=\"labelState\">\n <input\n #input\n (input)=\"onHandleInput($event)\"\n *ngSwitchCase=\"'search'\"\n [placeholder]=\"searchPlaceholder\"\n class=\"select-search | is-transparent\" />\n\n <ng-container *ngSwitchCase=\"'value'\">\n <ng-container\n *ngTemplateOutlet=\"valueTemplate ?? defaultValueTemplate; context: { $implicit: value }\"></ng-container>\n </ng-container>\n </ng-container>\n\n <span class=\"drop-down-icon ml-2\">\n <fa-icon [icon]=\"icon\"></fa-icon>\n </span>\n </div>\n </div>\n</ng-template>\n\n<ng-template #defaultValueTemplate>\n <span [ngClass]=\"{ empty: !value }\" class=\"selected-value\">{{ displayValue || placeholder }}</span>\n</ng-template>\n", styles: ["body.is-os-windows::-webkit-scrollbar,body.is-os-windows *::-webkit-scrollbar{border-radius:10px;height:10px;width:10px;border:none;background:transparent}body.is-os-windows::-webkit-scrollbar-button,body.is-os-windows *::-webkit-scrollbar-button{background:#DDE1E6;color:#dde1e6}body.is-os-windows::-webkit-scrollbar-thumb,body.is-os-windows *::-webkit-scrollbar-thumb{border-radius:10px;background:#DDE1E6}body.is-os-windows::-webkit-scrollbar-thumb:hover,body.is-os-windows *::-webkit-scrollbar-thumb:hover{background:#DDE1E6}body.is-os-windows::-webkit-scrollbar-track,body.is-os-windows *::-webkit-scrollbar-track{border:none;background-color:transparent;background-repeat:no-repeat;background-image:url(https://cdn.hestia.earth/prod/assets/images/scrollbar-thumb.png);background-position:center}body.is-os-windows::-webkit-scrollbar-track:vertical,body.is-os-windows *::-webkit-scrollbar-track:vertical{background-repeat:repeat-y}body.is-os-windows::-webkit-scrollbar-track:horizontal,body.is-os-windows *::-webkit-scrollbar-track:horizontal{background-repeat:repeat-x}body.is-os-windows .is-dark::-webkit-scrollbar-button{background:#4C7194;color:#4c7194}body.is-os-windows .is-dark::-webkit-scrollbar-thumb{background:#4C7194}body.is-os-windows .is-dark::-webkit-scrollbar-thumb:hover{background:#4C7194}::ng-deep:root{--select-border-radius: 6px;--select-min-width: 250px;--select-shadow: rgba(0, 0, 0, .1) 0px 4px 6px -1px, rgba(0, 0, 0, .06) 0px 2px 4px -1px}.select-host{display:flex;justify-content:flex-start;flex-direction:column;padding:5px 10px;box-sizing:border-box;min-width:var(--select-min-width);border-radius:var(--select-border-radius);border:1px solid #F5F5F5;background:hsl(0,0%,100%);box-shadow:var(--select-shadow)}.select-host:hover{cursor:pointer;background-color:var(--color-background, #eceff1)}.select-host.select-panel-open{position:relative;z-index:1001;border-radius:var(--select-border-radius) var(--select-border-radius) 0 0;box-shadow:none}.select-host.is-disabled{opacity:.5;pointer-events:none;-webkit-user-select:none;user-select:none}.selected-value.empty{opacity:.6}.select-search{border:none;outline:none;background:transparent}.select-panel{box-sizing:border-box;border-radius:0 0 var(--select-border-radius, 6px) var(--select-border-radius, 6px);border:1px solid #F5F5F5;background-color:#fff;transform-origin:top;max-height:350px;overflow:auto;box-shadow:var(--select-shadow)}\n"] }]
265
+ ], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div #origin=\"cdkOverlayOrigin\" cdkOverlayOrigin>\n <ng-container\n *ngTemplateOutlet=\"\n selectTemplate ?? defaultSelectTemplate;\n context: {\n $implicit: this,\n value,\n displayValue,\n labelState,\n icon,\n placeholder,\n searchPlaceholder,\n width,\n valueTemplate\n }\n \"></ng-container>\n</div>\n\n<ng-template\n (backdropClick)=\"close()\"\n (detach)=\"close()\"\n [cdkConnectedOverlayOffsetX]=\"overlayOptions?.offsetX ?? -11\"\n [cdkConnectedOverlayOffsetY]=\"overlayOptions?.offsetY ?? 4\"\n [cdkConnectedOverlayOpen]=\"isOpen\"\n [cdkConnectedOverlayOrigin]=\"origin\"\n cdkConnectedOverlay\n cdkConnectedOverlayBackdropClass=\"cdk-overlay-transparent-backdrop\"\n cdkConnectedOverlayHasBackdrop>\n <div [style.min-width]=\"width\" class=\"select-panel\">\n <ng-content></ng-content>\n </div>\n</ng-template>\n\n<ng-template #defaultSelectTemplate>\n <div class=\"select-host\">\n <div class=\"control | is-flex is-justify-content-space-between is-align-items-center\">\n <ng-container [ngSwitch]=\"labelState\">\n <input\n #input\n (input)=\"onHandleInput($event)\"\n *ngSwitchCase=\"'search'\"\n [placeholder]=\"searchPlaceholder\"\n class=\"select-search | is-transparent\" />\n\n <ng-container *ngSwitchCase=\"'value'\">\n <ng-container\n *ngTemplateOutlet=\"valueTemplate ?? defaultValueTemplate; context: { $implicit: value }\"></ng-container>\n </ng-container>\n </ng-container>\n\n <span class=\"drop-down-icon ml-2\">\n <fa-icon [icon]=\"icon\"></fa-icon>\n </span>\n </div>\n </div>\n</ng-template>\n\n<ng-template #defaultValueTemplate>\n <span [ngClass]=\"{ empty: !value }\" class=\"selected-value\">{{ displayValue || placeholder }}</span>\n</ng-template>\n", styles: ["body.is-os-windows::-webkit-scrollbar,body.is-os-windows *::-webkit-scrollbar{border-radius:10px;height:10px;width:10px;border:none;background:transparent}body.is-os-windows::-webkit-scrollbar-button,body.is-os-windows *::-webkit-scrollbar-button{background:#dde1e6;color:#dde1e6}body.is-os-windows::-webkit-scrollbar-thumb,body.is-os-windows *::-webkit-scrollbar-thumb{border-radius:10px;background:#dde1e6}body.is-os-windows::-webkit-scrollbar-thumb:hover,body.is-os-windows *::-webkit-scrollbar-thumb:hover{background:#dde1e6}body.is-os-windows::-webkit-scrollbar-track,body.is-os-windows *::-webkit-scrollbar-track{border:none;background-color:transparent;background-repeat:no-repeat;background-image:url(https://cdn.hestia.earth/prod/assets/images/scrollbar-thumb.png);background-position:center}body.is-os-windows::-webkit-scrollbar-track:vertical,body.is-os-windows *::-webkit-scrollbar-track:vertical{background-repeat:repeat-y}body.is-os-windows::-webkit-scrollbar-track:horizontal,body.is-os-windows *::-webkit-scrollbar-track:horizontal{background-repeat:repeat-x}body.is-os-windows .is-dark::-webkit-scrollbar-button{background:#4c7194;color:#4c7194}body.is-os-windows .is-dark::-webkit-scrollbar-thumb{background:#4c7194}body.is-os-windows .is-dark::-webkit-scrollbar-thumb:hover{background:#4c7194}::ng-deep:root{--select-border-radius: 6px;--select-min-width: 250px;--select-shadow: rgba(0, 0, 0, .1) 0px 4px 6px -1px, rgba(0, 0, 0, .06) 0px 2px 4px -1px}.select-host{display:flex;justify-content:flex-start;flex-direction:column;padding:5px 10px;box-sizing:border-box;min-width:var(--select-min-width);border-radius:var(--select-border-radius);border:1px solid #f5f5f5;background:hsl(0,0%,100%);box-shadow:var(--select-shadow)}.select-host:hover{cursor:pointer;background-color:var(--color-background, #eceff1)}.select-host.select-panel-open{position:relative;z-index:1001;border-radius:var(--select-border-radius) var(--select-border-radius) 0 0;box-shadow:none}.select-host.is-disabled{opacity:.5;pointer-events:none;-webkit-user-select:none;user-select:none}.selected-value.empty{opacity:.6}.select-search{border:none;outline:none;background:transparent}.select-panel{box-sizing:border-box;border-radius:0 0 var(--select-border-radius, 6px) var(--select-border-radius, 6px);border:1px solid #f5f5f5;background-color:#fff;transform-origin:top;max-height:350px;overflow:auto;box-shadow:var(--select-shadow)}\n"] }]
266
266
  }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }, { type: i0.ElementRef }, { type: i0.DestroyRef }]; }, propDecorators: { placeholder: [{
267
267
  type: Input
268
268
  }], searchPlaceholder: [{