@hestia-earth/ui-components 0.30.3 → 0.30.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/files/files-form/files-form.component.mjs +14 -41
- package/esm2022/files/files-form.model.mjs +12 -16
- package/fesm2022/hestia-earth-ui-components.mjs +19 -49
- package/fesm2022/hestia-earth-ui-components.mjs.map +1 -1
- package/files/files-form/files-form.component.d.ts +4 -7
- package/package.json +1 -1
|
@@ -10,17 +10,16 @@ import { NgbTooltip } from '@ng-bootstrap/ng-bootstrap';
|
|
|
10
10
|
import { isUndefined } from '@hestia-earth/utils';
|
|
11
11
|
import { SvgIconComponent } from 'angular-svg-icon';
|
|
12
12
|
import { MarkdownComponent } from 'ngx-markdown';
|
|
13
|
-
import { formatPropertyError, propertyError, hasError, hasWarning, recursiveProperties, propertyId, groupChanged,
|
|
13
|
+
import { formatPropertyError, propertyError, hasError, hasWarning, recursiveProperties, propertyId, groupChanged, formatter } from '../files-form.model';
|
|
14
14
|
import { errorHasError, errorHasWarning, filterError, formatError } from '../files-error.model';
|
|
15
15
|
import { PopoverConfirmComponent } from '../../common/popover-confirm/popover-confirm.component';
|
|
16
16
|
import { SchemaVersionLinkComponent } from '../../common/schema-version-link/schema-version-link.component';
|
|
17
17
|
import { SchemaInfoComponent } from '../../schema/schema-info/schema-info.component';
|
|
18
|
-
import { HeSchemaService } from '../../schema/schema.service';
|
|
18
|
+
import { HeSchemaService, linkTypeEnabled } from '../../schema/schema.service';
|
|
19
19
|
import { NodeIconComponent } from '../../node/node-icon/node-icon.component';
|
|
20
20
|
import { SitesMapsComponent } from '../../sites/sites-maps/sites-maps.component';
|
|
21
|
-
import { ClipboardComponent } from '../../common';
|
|
21
|
+
import { ClipboardComponent, isExternal } from '../../common';
|
|
22
22
|
import * as i0 from "@angular/core";
|
|
23
|
-
import * as i1 from "@angular/forms";
|
|
24
23
|
const stringify = (value) => JSON.stringify(value);
|
|
25
24
|
export class FilesFormComponent {
|
|
26
25
|
constructor() {
|
|
@@ -41,12 +40,10 @@ export class FilesFormComponent {
|
|
|
41
40
|
this.isOpen = model(true);
|
|
42
41
|
this.node = input.required();
|
|
43
42
|
this.errors = input([]);
|
|
44
|
-
this.errorsEditable = input(false);
|
|
45
43
|
this.schemas = toSignal(this.schemaService.schemas$);
|
|
46
44
|
this.schemaType = computed(() => this.node()?.type || this.node()?.['@type']);
|
|
47
45
|
this.schema = computed(() => this.schemas()?.[this.schemaType()] ?? {});
|
|
48
46
|
this.nodeErorrResolved = output();
|
|
49
|
-
this.nodeErrorAdded = output();
|
|
50
47
|
this.NodeType = NodeType;
|
|
51
48
|
this.SchemaType = SchemaType;
|
|
52
49
|
this.formatPropertyError = formatPropertyError;
|
|
@@ -58,6 +55,14 @@ export class FilesFormComponent {
|
|
|
58
55
|
this.node().region?.name,
|
|
59
56
|
this.node().country?.name
|
|
60
57
|
].some(Boolean));
|
|
58
|
+
this.nodeId = computed(() => this.node()?.id || this.node()?.['@id']);
|
|
59
|
+
this.nodeUrl = computed(() => [
|
|
60
|
+
linkTypeEnabled(this.schemaType()),
|
|
61
|
+
// handle links on Community Edition
|
|
62
|
+
this.node()?.['@id'] || (isExternal() && this.nodeId())
|
|
63
|
+
].every(Boolean)
|
|
64
|
+
? `/${this.schemaType().toLowerCase()}/${this.nodeId()}`
|
|
65
|
+
: '');
|
|
61
66
|
this.filteredErrors = computed(() => this.errors().filter(filterError));
|
|
62
67
|
this.nodeError = computed(() => propertyError(this.filteredErrors())(''));
|
|
63
68
|
this.nodeProperty = computed(() => this.schema()
|
|
@@ -90,38 +95,6 @@ export class FilesFormComponent {
|
|
|
90
95
|
return [fullKey, id].join('_');
|
|
91
96
|
}
|
|
92
97
|
// --- Errors
|
|
93
|
-
addErrorEnabled(property) {
|
|
94
|
-
return this.errorsEditable() && !property.hasError && !property.hasWarning;
|
|
95
|
-
}
|
|
96
|
-
addError(property, { message, level }) {
|
|
97
|
-
if (!message) {
|
|
98
|
-
return;
|
|
99
|
-
}
|
|
100
|
-
const keys = property.fullKey.split('.');
|
|
101
|
-
keys.pop();
|
|
102
|
-
const error = property.isAdded
|
|
103
|
-
? {
|
|
104
|
-
message,
|
|
105
|
-
level: level || 'error',
|
|
106
|
-
dataPath: keyToDataPath(keys.join('.')),
|
|
107
|
-
params: {
|
|
108
|
-
missingProperty: property.key
|
|
109
|
-
}
|
|
110
|
-
}
|
|
111
|
-
: {
|
|
112
|
-
message,
|
|
113
|
-
level: level || 'error',
|
|
114
|
-
dataPath: keyToDataPath(property.fullKey)
|
|
115
|
-
};
|
|
116
|
-
property.newError = { level: 'warning' };
|
|
117
|
-
property.error = error;
|
|
118
|
-
property.hasError = errorHasError(error);
|
|
119
|
-
property.hasWarning = errorHasWarning(error);
|
|
120
|
-
this.nodeErrorAdded.emit(error);
|
|
121
|
-
}
|
|
122
|
-
editError(property) {
|
|
123
|
-
property.newError = property.error;
|
|
124
|
-
}
|
|
125
98
|
resolveError(property) {
|
|
126
99
|
const index = property.error?.index;
|
|
127
100
|
property.error = undefined;
|
|
@@ -131,7 +104,7 @@ export class FilesFormComponent {
|
|
|
131
104
|
return groupChanged(this.properties(), property.fullKey);
|
|
132
105
|
}
|
|
133
106
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.11", ngImport: i0, type: FilesFormComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
134
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.11", type: FilesFormComponent, isStandalone: true, selector: "he-files-form", inputs: { isOpen: { classPropertyName: "isOpen", publicName: "isOpen", isSignal: true, isRequired: false, transformFunction: null }, node: { classPropertyName: "node", publicName: "node", isSignal: true, isRequired: true, transformFunction: null }, errors: { classPropertyName: "errors", publicName: "errors", isSignal: true, isRequired: false, transformFunction: null }, errorsEditable: { classPropertyName: "errorsEditable", publicName: "errorsEditable", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { isOpen: "isOpenChange", nodeErorrResolved: "nodeErorrResolved", nodeErrorAdded: "nodeErrorAdded" }, ngImport: i0, template: "<div class=\"is-flex is-flex-direction-column is-align-items-flex-start is-align-self-stretch | files-form-container\">\n <div\n class=\"is-flex is-flex-direction-row is-justify-content-space-between is-align-items-center is-align-self-stretch is-gap-16 px-4 py-2 has-text-secondary | files-form-header\"\n (click)=\"isOpen.set(!isOpen())\"\n pointer>\n <div class=\"is-flex is-gap-8 is-align-items-center\">\n @if (nodeProperty()) {\n <he-node-icon [type]=\"schemaType()\" />\n <span class=\"has-text-weight-bold has-text-secondary is-size-5\">{{ schemaType() }}</span>\n @if (nodeProperty().hasError || hasError()) {\n <svg-icon name=\"far-times-circle\" class=\"has-text-danger is-flex is-align-items-center\" />\n <span class=\"has-text-danger has-text-weight-bold is-italic\">Error</span>\n } @else if (nodeProperty().hasWarning || hasWarning()) {\n <svg-icon name=\"exclamation-triangle\" class=\"has-text-warning is-flex is-align-items-center\" />\n <span class=\"has-text-warning has-text-weight-bold is-italic\">Warning</span>\n }\n }\n </div>\n <fa-icon [icon]=\"isOpen() ? faMinus : faPlus\" />\n </div>\n\n <div class=\"is-flex is-flex-direction-column is-align-items-flex-start is-align-self-stretch\">\n @if (isOpen()) {\n @if (errorsEditable()) {\n <ng-container *ngTemplateOutlet=\"nodeErrorForm; context: { $implicit: nodeProperty() }\" />\n }\n @if (nodeProperty()?.error) {\n <ng-container\n *ngTemplateOutlet=\"\n propertyError;\n context: { property: nodeProperty(), edit: true, classes: 'is-my-1 is-px-3' }\n \" />\n }\n }\n\n <div class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-self-stretch\">\n @for (property of properties(); track trackByProperty($index, property)) {\n <ng-container *ngTemplateOutlet=\"showProperty; context: { $implicit: property }\" />\n }\n </div>\n\n @if (isOpen()) {\n <ng-container *ngTemplateOutlet=\"propertyMap; context: { $implicit: nodeProperty() }\" />\n }\n </div>\n\n <ng-content></ng-content>\n</div>\n\n<ng-template #propertyKeyTooltip let-property=\"property\">\n <markdown [data]=\"property.schema.description\" />\n</ng-template>\n\n<ng-template #propertyKey let-property=\"property\">\n @if (property.schema?.description) {\n <span\n class=\"property-key trigger-popover\"\n [ngbTooltip]=\"propertyKeyTooltip\"\n [tooltipContext]=\"{ property }\"\n tooltipClass=\"property-tooltip\"\n placement=\"bottom-left\"\n container=\"body\">\n <span>{{ property.key }}</span>\n </span>\n } @else {\n <span>{{ property.key }}</span>\n }\n</ng-template>\n\n<ng-template #showProperty let-property>\n @if ((isOpen() || property.closedVisible) && !property.isHidden) {\n @if (property.properties.length) {\n <div class=\"is-flex is-flex-direction-column is-flex-wrap-wrap w-100 | properties-container\">\n <div\n class=\"is-flex is-flex-direction-row is-justify-content-space-between is-gap-4 has-text-secondary w-100 | properties-container--title\"\n (click)=\"property.isOpen = !property.isOpen\"\n pointer\n [class.is-open]=\"property.isOpen\"\n [class.is-group-error-danger]=\"property.hasError\"\n [class.is-group-error-warning]=\"property.hasWarning\">\n <div class=\"is-flex is-gap-8 has-text-weight-bold\">\n <!-- @if (property.schemaType === SchemaType.Term) {\n <he-node-icon [type]=\"property.schemaType\" />\n } -->\n <ng-container *ngTemplateOutlet=\"propertyKey; context: { property }\" />\n </div>\n <fa-icon [icon]=\"property.isOpen ? faMinus : faPlus\" />\n </div>\n @if (property.isOpen) {\n @if (property.error) {\n <ng-container\n *ngTemplateOutlet=\"\n propertyError;\n context: { property: property, edit: true, classes: 'is-my-1 is-px-3' }\n \" />\n }\n @if (property.isArray) {\n <div class=\"py-2 px-3 w-100\">\n @if (errorsEditable()) {\n <ng-container *ngTemplateOutlet=\"nodeErrorForm; context: { $implicit: property }\" />\n }\n @for (prop2 of property.properties; track trackByProperty($index, prop2)) {\n @if (prop2.key) {\n <div class=\"pt-2 my-4 | property-array-container\" [id]=\"prop2.fullKey + '_' + prop2.id\">\n <div class=\"property-array-number ml-2\">\n <div class=\"is-flex is-justify-content-center is-align-items-center has-addons number-tags\">\n <span class=\"number-tag\">{{ prop2.key }}</span>\n </div>\n </div>\n @if (errorsEditable()) {\n <ng-container *ngTemplateOutlet=\"nodeErrorForm; context: { $implicit: prop2 }\" />\n }\n @if (prop2.error) {\n <ng-container\n *ngTemplateOutlet=\"\n propertyError;\n context: { property: prop2, edit: true, classes: 'is-my-1 is-px-3' }\n \" />\n }\n <div class=\"p-3\">\n <div class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-self-stretch\">\n @for (prop3 of prop2.properties; track trackByProperty($index, prop3)) {\n <ng-container *ngTemplateOutlet=\"showProperty; context: { $implicit: prop3 }\" />\n }\n </div>\n </div>\n </div>\n }\n }\n </div>\n } @else {\n @if (errorsEditable()) {\n <ng-container *ngTemplateOutlet=\"nodeErrorForm; context: { $implicit: property }\" />\n }\n <div class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-self-stretch\">\n @for (prop2 of property.properties; track trackByProperty($index, prop2)) {\n <ng-container *ngTemplateOutlet=\"showProperty; context: { $implicit: prop2 }\" />\n }\n </div>\n }\n }\n </div>\n } @else {\n <div\n class=\"is-flex is-flex-direction-column is-gap-4 | property-container\"\n [class.is-error-danger]=\"property.hasError\"\n [class.is-error-warning]=\"property.hasWarning\">\n <div\n class=\"is-flex is-flex-direction-row is-align-items-center is-justify-content-space-between is-gap-4 | field-container\">\n @if (property.key) {\n <div class=\"is-flex is-align-items-center is-flex-wrap-wrap is-gap-4\">\n <div class=\"is-flex has-text-secondary\">\n <ng-container *ngTemplateOutlet=\"propertyKey; context: { property }\" />\n <span>:</span>\n </div>\n\n <span class=\"is-flex is-size-7 | property-value\">\n {{ property.value }}\n </span>\n\n <div class=\"copy-button\">\n <he-clipboard [value]=\"property.value\" [hideText]=\"true\" [icon]=\"farClone\"></he-clipboard>\n </div>\n </div>\n\n <div class=\"is-flex is-gap-8\">\n @if (addErrorEnabled(property)) {\n <he-popover-confirm\n class=\"is-flex\"\n position=\"left\"\n [content]=\"popupErrorForm\"\n (confirmed)=\"addError(property, $event)\">\n <fa-icon [icon]=\"faComments\" />\n </he-popover-confirm>\n }\n @if (property.externalUrl?.url) {\n <a\n [href]=\"property.externalUrl.url + (property.externalUrl.urlParamValue ? property.value : '')\"\n target=\"_blank\"\n [title]=\"property.externalUrl.title\"\n [ngClass]=\"{ 'is-info': property.key === 'type' }\"\n [attr.disabled]=\"property.externalUrl.urlParamValue && !property.value ? true : null\">\n <fa-icon [icon]=\"property.externalUrl.icon || faExternalLinkAlt\" />\n </a>\n }\n </div>\n }\n </div>\n @if (property.hasError || property.hasWarning) {\n <ng-container *ngTemplateOutlet=\"propertyError; context: { property: property, edit: false }\" />\n }\n </div>\n }\n }\n</ng-template>\n\n<ng-template #propertyError let-property=\"property\" let-edit=\"edit\" let-classes=\"classes\">\n @if (property.error) {\n <div\n class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-items-center is-gap-8 is-size-6 is-m-0 w-100 has-text-grey {{\n classes\n }} | property-error\"\n [class.is-error-danger]=\"property.hasError\"\n [class.is-error-warning]=\"property.hasWarning\">\n @if (property.hasError) {\n <fa-icon class=\"has-text-danger is-py-1\" [icon]=\"farCircleXmark\" size=\"xl\" />\n } @else {\n <svg-icon class=\"has-text-warning is-pt-1\" name=\"exclamation-triangle\" />\n }\n <div [innerHTML]=\"property.error.message\"></div>\n @if (property.error.index >= 0) {\n @if (edit && errorsEditable()) {\n <a (click)=\"editError(property)\">\n <fa-icon class=\"pr-2\" [icon]=\"faEdit\" />\n <span>Edit</span>\n </a>\n }\n @if (property.hasWarning || errorsEditable()) {\n <a (click)=\"resolveError(property)\">\n <fa-icon class=\"pr-2\" [icon]=\"faCheck\" />\n <span>Resolved</span>\n </a>\n }\n }\n </div>\n }\n</ng-template>\n\n<ng-template #propertyMap let-property>\n @if (showMap()) {\n <div class=\"is-flex is-flex-direction-column is-flex-wrap-wrap w-100 | properties-container\">\n <div\n class=\"is-flex is-flex-direction-row is-justify-content-space-between is-gap-4 w-100 | properties-container--title\"\n (click)=\"mapVisible.set(!mapVisible())\"\n pointer\n [class.is-open]=\"mapVisible()\">\n <span class=\"is-size-6\">View on Map</span>\n <fa-icon [icon]=\"mapVisible() ? faMinus : faPlus\" />\n </div>\n @if (mapVisible()) {\n <he-sites-maps [sites]=\"[node()]\" [showNotice]=\"false\" />\n }\n </div>\n }\n</ng-template>\n\n<ng-template #nodeErrorForm let-property>\n @if (property.newError) {\n <div class=\"field has-addons w-100 is-p-2 is-m-0\">\n <div class=\"control\">\n <div class=\"select is-small\">\n <select [(ngModel)]=\"property.newError.level\">\n <option [value]=\"undefined\">Select Level</option>\n <option value=\"error\">Error</option>\n <option value=\"warning\">Warning</option>\n </select>\n </div>\n </div>\n <div class=\"control is-expanded\">\n <textarea\n class=\"textarea is-small\"\n [(ngModel)]=\"property.newError.message\"\n placeholder=\"Enter your message here\"\n rows=\"1\"></textarea>\n </div>\n <div class=\"control\">\n <button\n class=\"button is-small\"\n [disabled]=\"!property.newError.level || !property.newError.message\"\n (click)=\"addError(property, property.newError)\">\n <fa-icon [icon]=\"faPlusCircle\" />\n <span class=\"pl-2\">Add</span>\n </button>\n </div>\n </div>\n }\n</ng-template>\n\n<ng-template #popupErrorForm let-data=\"data\">\n <div class=\"field has-addons\">\n <div class=\"control\">\n <div class=\"select\">\n <select [(ngModel)]=\"data.level\">\n <option [value]=\"undefined\">Select Level</option>\n <option value=\"error\">Error</option>\n <option value=\"warning\">Warning</option>\n </select>\n </div>\n </div>\n <div class=\"control\">\n <input class=\"input\" [(ngModel)]=\"data.message\" placeholder=\"Enter your message here\" />\n </div>\n </div>\n</ng-template>\n", styles: [".notification{color:#0a0a0a!important}.notification.is-success{background-color:#d5f3d8}.notification.is-success *:not(.button):not(.icon):not(.dropdown-item)>fa-icon,.notification.is-success>fa-icon{color:#48c78e}.notification.is-info{background-color:#d3ebed}.notification.is-info *:not(.button):not(.icon):not(.dropdown-item)>fa-icon,.notification.is-info>fa-icon{color:#249da5}.notification.is-warning{background-color:#ffdec0}.notification.is-warning *:not(.button):not(.icon):not(.dropdown-item)>fa-icon,.notification.is-warning>fa-icon{color:#ff881b}.notification.is-danger{background-color:#ffcdd0}.notification.is-danger *:not(.button):not(.icon):not(.dropdown-item)>fa-icon,.notification.is-danger>fa-icon{color:#f14668}he-sites-maps{height:200px}.files-form-container{box-shadow:2px 2px 4px #00000029;background-color:#fff}.files-form-header{padding:10px 12px;background-color:#dbe3ea;border-top-left-radius:3px;border-top-right-radius:3px}.properties-container--title,.property-container{padding:6px 12px}.property-container{flex:none;flex-grow:1;min-width:50%;max-width:100%;border:1px solid #f5f5f5}.property-container.is-error-danger{border-color:#f14668}.property-container.is-error-warning{border-color:#ff881b}.properties-container--title{border-top:1px solid #dbe3ea;border-bottom:1px solid #dbe3ea;background:#f5f7f9}.properties-container .properties-container{border-radius:6px;border:1px solid #dbe3ea;margin:4px 0}.properties-container .properties-container--title{border-top:none}.property-key,.property-value{word-break:break-word}.property-array-container{border-radius:6px;border:1px solid #dbdbdb}.property-array-number .number-tags{border:1px solid #249da5;border-radius:50%;width:20px;height:20px}.property-array-number .number-tags .number-tag{color:#249da5}.copy-button{visibility:hidden}.field-container:hover .copy-button{visibility:visible}.is-group-error-danger{background-color:#ffcdd0}.is-group-error-warning{background-color:#ffdec0}.property-error.is-error-danger{background-color:#ffeced}.field-container+.property-error.is-error-danger{background-color:transparent;color:#e13939!important}.property-error.is-error-warning{background-color:#fff5ec}.field-container+.property-error.is-error-warning{background-color:transparent;color:#ff881b!important}.property-error.is-error-danger svg-icon,.property-error.is-error-danger fa-icon,.property-error.is-error-warning svg-icon,.property-error.is-error-warning fa-icon{width:24px}::ng-deep .property-tooltip{background-color:#fff;color:#0a0a0a;border:1px solid #dbdbdb;z-index:11;max-width:30vw}\n"], dependencies: [{ kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i1.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: ["compareWith"] }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "component", type: FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "mask", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "transform", "a11yRole"] }, { kind: "component", type: MarkdownComponent, selector: "markdown, [markdown]", inputs: ["data", "src", "disableSanitizer", "inline", "clipboard", "clipboardButtonComponent", "clipboardButtonTemplate", "emoji", "katex", "katexOptions", "mermaid", "mermaidOptions", "lineHighlight", "line", "lineOffset", "lineNumbers", "start", "commandLine", "filterOutput", "host", "prompt", "output", "user"], outputs: ["error", "load", "ready"] }, { kind: "directive", type: NgbTooltip, selector: "[ngbTooltip]", inputs: ["animation", "autoClose", "placement", "popperOptions", "triggers", "positionTarget", "container", "disableTooltip", "tooltipClass", "tooltipContext", "openDelay", "closeDelay", "ngbTooltip"], outputs: ["shown", "hidden"], exportAs: ["ngbTooltip"] }, { kind: "component", type: NodeIconComponent, selector: "he-node-icon", inputs: ["type", "size"] }, { kind: "component", type: PopoverConfirmComponent, selector: "he-popover-confirm", inputs: ["message", "content", "position", "popoverClass"], outputs: ["confirmed"] }, { kind: "component", type: SitesMapsComponent, selector: "he-sites-maps", inputs: ["loadPolygons", "sites", "zoom", "showNotice"] }, { kind: "component", type: SvgIconComponent, selector: "svg-icon", inputs: ["src", "name", "stretch", "applyClass", "svgClass", "class", "viewBox", "svgAriaLabel", "svgStyle"] }, { kind: "component", type: ClipboardComponent, selector: "he-clipboard", inputs: ["icon", "value", "disabled", "hideText", "size", "rotate", "clipboardClass"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
107
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.11", type: FilesFormComponent, isStandalone: true, selector: "he-files-form", inputs: { isOpen: { classPropertyName: "isOpen", publicName: "isOpen", isSignal: true, isRequired: false, transformFunction: null }, node: { classPropertyName: "node", publicName: "node", isSignal: true, isRequired: true, transformFunction: null }, errors: { classPropertyName: "errors", publicName: "errors", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { isOpen: "isOpenChange", nodeErorrResolved: "nodeErorrResolved" }, ngImport: i0, template: "<div class=\"is-flex is-flex-direction-column is-align-items-flex-start is-align-self-stretch | files-form-container\">\n <div\n class=\"is-flex is-flex-direction-row is-justify-content-space-between is-align-items-center is-align-self-stretch is-gap-16 px-4 py-2 has-text-secondary | files-form-header\"\n (click)=\"isOpen.set(!isOpen())\"\n pointer>\n <div class=\"is-flex is-gap-8 is-align-items-center\">\n @if (nodeProperty()) {\n <he-node-icon [type]=\"schemaType()\" />\n <span class=\"has-text-weight-bold has-text-secondary is-size-5\">{{ schemaType() }}</span>\n @if (nodeProperty().hasError || hasError()) {\n <svg-icon name=\"far-times-circle\" class=\"has-text-danger is-flex is-align-items-center\" />\n <span class=\"has-text-danger has-text-weight-bold is-italic\">Error</span>\n } @else if (nodeProperty().hasWarning || hasWarning()) {\n <svg-icon name=\"exclamation-triangle\" class=\"has-text-warning is-flex is-align-items-center\" />\n <span class=\"has-text-warning has-text-weight-bold is-italic\">Warning</span>\n }\n }\n </div>\n <div class=\"is-flex is-flex-direction-row is-align-items-center is-gap-16\">\n @if (nodeUrl()) {\n <a class=\"external-link\" [href]=\"nodeUrl()\" target=\"_blank\">\n <span>View</span>\n <span class=\"is-pl-1\">{{ schemaType() }}</span>\n <fa-icon class=\"is-pl-2\" [icon]=\"faExternalLinkAlt\" />\n </a>\n }\n <fa-icon [icon]=\"isOpen() ? faMinus : faPlus\" />\n </div>\n </div>\n\n <div class=\"is-flex is-flex-direction-column is-align-items-flex-start is-align-self-stretch\">\n @if (isOpen()) {\n @if (nodeProperty()?.error) {\n <ng-container\n *ngTemplateOutlet=\"propertyError; context: { property: nodeProperty(), classes: 'is-my-1 is-px-3' }\" />\n }\n }\n\n <div class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-self-stretch\">\n @for (property of properties(); track trackByProperty($index, property)) {\n <ng-container *ngTemplateOutlet=\"showProperty; context: { $implicit: property }\" />\n }\n </div>\n\n @if (isOpen()) {\n <ng-container *ngTemplateOutlet=\"propertyMap; context: { $implicit: nodeProperty() }\" />\n }\n </div>\n\n <ng-content />\n</div>\n\n<ng-template #propertyKeyTooltip let-property=\"property\">\n <markdown [data]=\"property.schema.description\" />\n</ng-template>\n\n<ng-template #propertyKey let-property=\"property\">\n @if (property.schema?.description) {\n <span\n class=\"property-key trigger-popover\"\n [ngbTooltip]=\"propertyKeyTooltip\"\n [tooltipContext]=\"{ property }\"\n tooltipClass=\"property-tooltip\"\n placement=\"bottom-left\"\n container=\"body\">\n <span>{{ property.key }}</span>\n </span>\n } @else {\n <span>{{ property.key }}</span>\n }\n</ng-template>\n\n<ng-template #showProperty let-property>\n @if ((isOpen() || property.closedVisible) && !property.isHidden) {\n @if (property.properties.length) {\n <div class=\"is-flex is-flex-direction-column is-flex-wrap-wrap w-100 | properties-container\">\n <div\n class=\"is-flex is-flex-direction-row is-justify-content-space-between is-gap-4 has-text-secondary w-100 | properties-container--title\"\n (click)=\"property.isOpen = !property.isOpen\"\n pointer\n [class.is-open]=\"property.isOpen\"\n [class.is-group-error-danger]=\"property.hasError\"\n [class.is-group-error-warning]=\"property.hasWarning\">\n <div class=\"is-flex is-gap-8 has-text-weight-bold\">\n <!-- @if (property.schemaType === SchemaType.Term) {\n <he-node-icon [type]=\"property.schemaType\" />\n } -->\n <ng-container *ngTemplateOutlet=\"propertyKey; context: { property }\" />\n </div>\n <fa-icon [icon]=\"property.isOpen ? faMinus : faPlus\" />\n </div>\n @if (property.isOpen) {\n @if (property.error) {\n <ng-container *ngTemplateOutlet=\"propertyError; context: { property, classes: 'is-my-1 is-px-3' }\" />\n }\n @if (property.isArray) {\n <div class=\"py-2 px-3 w-100\">\n @for (prop2 of property.properties; track trackByProperty($index, prop2)) {\n @if (prop2.key) {\n <div class=\"pt-2 my-4 | property-array-container\" [id]=\"prop2.fullKey + '_' + prop2.id\">\n <div class=\"property-array-number ml-2\">\n <div class=\"is-flex is-justify-content-center is-align-items-center has-addons number-tags\">\n <span class=\"number-tag\">{{ prop2.key }}</span>\n </div>\n </div>\n @if (prop2.error) {\n <ng-container\n *ngTemplateOutlet=\"propertyError; context: { property: prop2, classes: 'is-my-1 is-px-3' }\" />\n }\n <div class=\"p-3\">\n <div class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-self-stretch\">\n @for (prop3 of prop2.properties; track trackByProperty($index, prop3)) {\n <ng-container *ngTemplateOutlet=\"showProperty; context: { $implicit: prop3 }\" />\n }\n </div>\n </div>\n </div>\n }\n }\n </div>\n } @else {\n <div class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-self-stretch\">\n @for (prop2 of property.properties; track trackByProperty($index, prop2)) {\n <ng-container *ngTemplateOutlet=\"showProperty; context: { $implicit: prop2 }\" />\n }\n </div>\n }\n }\n </div>\n } @else {\n <div\n class=\"is-flex is-flex-direction-column is-gap-4 | property-container\"\n [class.is-error-danger]=\"property.hasError\"\n [class.is-error-warning]=\"property.hasWarning\">\n <div\n class=\"is-flex is-flex-direction-row is-align-items-center is-justify-content-space-between is-gap-4 | field-container\">\n @if (property.key) {\n <div class=\"is-flex is-align-items-center is-flex-wrap-wrap is-gap-4\">\n <div class=\"is-flex has-text-secondary\">\n <ng-container *ngTemplateOutlet=\"propertyKey; context: { property }\" />\n <span>:</span>\n </div>\n\n <span class=\"is-flex is-size-7 | property-value\">\n {{ property.value }}\n </span>\n\n <div class=\"copy-button\">\n <he-clipboard [value]=\"property.value\" [hideText]=\"true\" [icon]=\"farClone\"></he-clipboard>\n </div>\n </div>\n\n <div class=\"is-flex is-gap-8\">\n @if (property.externalUrl?.url) {\n <a\n [href]=\"property.externalUrl.url + (property.externalUrl.urlParamValue ? property.value : '')\"\n target=\"_blank\"\n [title]=\"property.externalUrl.title\"\n [ngClass]=\"{ 'is-info': property.key === 'type' }\"\n [attr.disabled]=\"property.externalUrl.urlParamValue && !property.value ? true : null\">\n <fa-icon [icon]=\"property.externalUrl.icon || faExternalLinkAlt\" />\n </a>\n }\n </div>\n }\n </div>\n @if (property.hasError || property.hasWarning) {\n <ng-container *ngTemplateOutlet=\"propertyError; context: { property }\" />\n }\n </div>\n }\n }\n</ng-template>\n\n<ng-template #propertyError let-property=\"property\" let-classes=\"classes\">\n @if (property.error) {\n <div\n class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-items-center is-gap-8 is-size-6 is-m-0 w-100 has-text-grey {{\n classes\n }} | property-error\"\n [class.is-error-danger]=\"property.hasError\"\n [class.is-error-warning]=\"property.hasWarning\">\n @if (property.hasError) {\n <fa-icon class=\"has-text-danger is-py-1\" [icon]=\"farCircleXmark\" size=\"xl\" />\n } @else {\n <svg-icon class=\"has-text-warning is-pt-1\" name=\"exclamation-triangle\" />\n }\n <div [innerHTML]=\"property.error.message\"></div>\n @if (property.error.index >= 0) {\n @if (property.hasWarning) {\n <a (click)=\"resolveError(property)\">\n <fa-icon class=\"pr-2\" [icon]=\"faCheck\" />\n <span>Resolved</span>\n </a>\n }\n }\n </div>\n }\n</ng-template>\n\n<ng-template #propertyMap let-property>\n @if (showMap()) {\n <div class=\"is-flex is-flex-direction-column is-flex-wrap-wrap w-100 | properties-container\">\n <div\n class=\"is-flex is-flex-direction-row is-justify-content-space-between is-gap-4 w-100 | properties-container--title\"\n (click)=\"mapVisible.set(!mapVisible())\"\n pointer\n [class.is-open]=\"mapVisible()\">\n <span class=\"is-size-6\">View on Map</span>\n <fa-icon [icon]=\"mapVisible() ? faMinus : faPlus\" />\n </div>\n @if (mapVisible()) {\n <he-sites-maps [sites]=\"[node()]\" [showNotice]=\"false\" />\n }\n </div>\n }\n</ng-template>\n", styles: [".notification{color:#0a0a0a!important}.notification.is-success{background-color:#d5f3d8}.notification.is-success *:not(.button):not(.icon):not(.dropdown-item)>fa-icon,.notification.is-success>fa-icon{color:#48c78e}.notification.is-info{background-color:#d3ebed}.notification.is-info *:not(.button):not(.icon):not(.dropdown-item)>fa-icon,.notification.is-info>fa-icon{color:#249da5}.notification.is-warning{background-color:#ffdec0}.notification.is-warning *:not(.button):not(.icon):not(.dropdown-item)>fa-icon,.notification.is-warning>fa-icon{color:#ff881b}.notification.is-danger{background-color:#ffcdd0}.notification.is-danger *:not(.button):not(.icon):not(.dropdown-item)>fa-icon,.notification.is-danger>fa-icon{color:#f14668}he-sites-maps{height:200px}.external-link{color:#4c7194}.files-form-container{box-shadow:2px 2px 4px #00000029;background-color:#fff}.files-form-header{padding:10px 12px;background-color:#dbe3ea;border-top-left-radius:3px;border-top-right-radius:3px}.properties-container--title,.property-container{padding:6px 12px}.property-container{flex:none;flex-grow:1;min-width:50%;max-width:100%;border:1px solid #f5f5f5}.property-container.is-error-danger{border-color:#f14668}.property-container.is-error-warning{border-color:#ff881b}.properties-container--title{border-top:1px solid #dbe3ea;border-bottom:1px solid #dbe3ea;background:#f5f7f9}.properties-container .properties-container{border-radius:6px;border:1px solid #dbe3ea;margin:4px 0}.properties-container .properties-container--title{border-top:none}.property-key,.property-value{word-break:break-word}.property-array-container{border-radius:6px;border:1px solid #dbdbdb}.property-array-number .number-tags{border:1px solid #249da5;border-radius:50%;width:20px;height:20px}.property-array-number .number-tags .number-tag{color:#249da5}.copy-button{visibility:hidden}.field-container:hover .copy-button{visibility:visible}.is-group-error-danger{background-color:#ffcdd0}.is-group-error-warning{background-color:#ffdec0}.property-error.is-error-danger{background-color:#ffeced}.field-container+.property-error.is-error-danger{background-color:transparent;color:#e13939!important}.property-error.is-error-warning{background-color:#fff5ec}.field-container+.property-error.is-error-warning{background-color:transparent;color:#ff881b!important}.property-error.is-error-danger svg-icon,.property-error.is-error-danger fa-icon,.property-error.is-error-warning svg-icon,.property-error.is-error-warning fa-icon{width:24px}::ng-deep .property-tooltip{background-color:#fff;color:#0a0a0a;border:1px solid #dbdbdb;z-index:11;max-width:30vw}\n"], dependencies: [{ kind: "ngmodule", type: FormsModule }, { kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "component", type: FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "mask", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "transform", "a11yRole"] }, { kind: "component", type: MarkdownComponent, selector: "markdown, [markdown]", inputs: ["data", "src", "disableSanitizer", "inline", "clipboard", "clipboardButtonComponent", "clipboardButtonTemplate", "emoji", "katex", "katexOptions", "mermaid", "mermaidOptions", "lineHighlight", "line", "lineOffset", "lineNumbers", "start", "commandLine", "filterOutput", "host", "prompt", "output", "user"], outputs: ["error", "load", "ready"] }, { kind: "directive", type: NgbTooltip, selector: "[ngbTooltip]", inputs: ["animation", "autoClose", "placement", "popperOptions", "triggers", "positionTarget", "container", "disableTooltip", "tooltipClass", "tooltipContext", "openDelay", "closeDelay", "ngbTooltip"], outputs: ["shown", "hidden"], exportAs: ["ngbTooltip"] }, { kind: "component", type: NodeIconComponent, selector: "he-node-icon", inputs: ["type", "size"] }, { kind: "component", type: SitesMapsComponent, selector: "he-sites-maps", inputs: ["loadPolygons", "sites", "zoom", "showNotice"] }, { kind: "component", type: SvgIconComponent, selector: "svg-icon", inputs: ["src", "name", "stretch", "applyClass", "svgClass", "class", "viewBox", "svgAriaLabel", "svgStyle"] }, { kind: "component", type: ClipboardComponent, selector: "he-clipboard", inputs: ["icon", "value", "disabled", "hideText", "size", "rotate", "clipboardClass"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
135
108
|
}
|
|
136
109
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.11", ngImport: i0, type: FilesFormComponent, decorators: [{
|
|
137
110
|
type: Component,
|
|
@@ -149,6 +122,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.11", ngImpo
|
|
|
149
122
|
SitesMapsComponent,
|
|
150
123
|
SvgIconComponent,
|
|
151
124
|
ClipboardComponent
|
|
152
|
-
], template: "<div class=\"is-flex is-flex-direction-column is-align-items-flex-start is-align-self-stretch | files-form-container\">\n <div\n class=\"is-flex is-flex-direction-row is-justify-content-space-between is-align-items-center is-align-self-stretch is-gap-16 px-4 py-2 has-text-secondary | files-form-header\"\n (click)=\"isOpen.set(!isOpen())\"\n pointer>\n <div class=\"is-flex is-gap-8 is-align-items-center\">\n @if (nodeProperty()) {\n <he-node-icon [type]=\"schemaType()\" />\n <span class=\"has-text-weight-bold has-text-secondary is-size-5\">{{ schemaType() }}</span>\n @if (nodeProperty().hasError || hasError()) {\n <svg-icon name=\"far-times-circle\" class=\"has-text-danger is-flex is-align-items-center\" />\n <span class=\"has-text-danger has-text-weight-bold is-italic\">Error</span>\n } @else if (nodeProperty().hasWarning || hasWarning()) {\n <svg-icon name=\"exclamation-triangle\" class=\"has-text-warning is-flex is-align-items-center\" />\n <span class=\"has-text-warning has-text-weight-bold is-italic\">Warning</span>\n }\n }\n </div>\n <fa-icon [icon]=\"isOpen() ? faMinus : faPlus\" />\n </div>\n\n <div class=\"is-flex is-flex-direction-column is-align-items-flex-start is-align-self-stretch\">\n @if (isOpen()) {\n @if (errorsEditable()) {\n <ng-container *ngTemplateOutlet=\"nodeErrorForm; context: { $implicit: nodeProperty() }\" />\n }\n @if (nodeProperty()?.error) {\n <ng-container\n *ngTemplateOutlet=\"\n propertyError;\n context: { property: nodeProperty(), edit: true, classes: 'is-my-1 is-px-3' }\n \" />\n }\n }\n\n <div class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-self-stretch\">\n @for (property of properties(); track trackByProperty($index, property)) {\n <ng-container *ngTemplateOutlet=\"showProperty; context: { $implicit: property }\" />\n }\n </div>\n\n @if (isOpen()) {\n <ng-container *ngTemplateOutlet=\"propertyMap; context: { $implicit: nodeProperty() }\" />\n }\n </div>\n\n <ng-content></ng-content>\n</div>\n\n<ng-template #propertyKeyTooltip let-property=\"property\">\n <markdown [data]=\"property.schema.description\" />\n</ng-template>\n\n<ng-template #propertyKey let-property=\"property\">\n @if (property.schema?.description) {\n <span\n class=\"property-key trigger-popover\"\n [ngbTooltip]=\"propertyKeyTooltip\"\n [tooltipContext]=\"{ property }\"\n tooltipClass=\"property-tooltip\"\n placement=\"bottom-left\"\n container=\"body\">\n <span>{{ property.key }}</span>\n </span>\n } @else {\n <span>{{ property.key }}</span>\n }\n</ng-template>\n\n<ng-template #showProperty let-property>\n @if ((isOpen() || property.closedVisible) && !property.isHidden) {\n @if (property.properties.length) {\n <div class=\"is-flex is-flex-direction-column is-flex-wrap-wrap w-100 | properties-container\">\n <div\n class=\"is-flex is-flex-direction-row is-justify-content-space-between is-gap-4 has-text-secondary w-100 | properties-container--title\"\n (click)=\"property.isOpen = !property.isOpen\"\n pointer\n [class.is-open]=\"property.isOpen\"\n [class.is-group-error-danger]=\"property.hasError\"\n [class.is-group-error-warning]=\"property.hasWarning\">\n <div class=\"is-flex is-gap-8 has-text-weight-bold\">\n <!-- @if (property.schemaType === SchemaType.Term) {\n <he-node-icon [type]=\"property.schemaType\" />\n } -->\n <ng-container *ngTemplateOutlet=\"propertyKey; context: { property }\" />\n </div>\n <fa-icon [icon]=\"property.isOpen ? faMinus : faPlus\" />\n </div>\n @if (property.isOpen) {\n @if (property.error) {\n <ng-container\n *ngTemplateOutlet=\"\n propertyError;\n context: { property: property, edit: true, classes: 'is-my-1 is-px-3' }\n \" />\n }\n @if (property.isArray) {\n <div class=\"py-2 px-3 w-100\">\n @if (errorsEditable()) {\n <ng-container *ngTemplateOutlet=\"nodeErrorForm; context: { $implicit: property }\" />\n }\n @for (prop2 of property.properties; track trackByProperty($index, prop2)) {\n @if (prop2.key) {\n <div class=\"pt-2 my-4 | property-array-container\" [id]=\"prop2.fullKey + '_' + prop2.id\">\n <div class=\"property-array-number ml-2\">\n <div class=\"is-flex is-justify-content-center is-align-items-center has-addons number-tags\">\n <span class=\"number-tag\">{{ prop2.key }}</span>\n </div>\n </div>\n @if (errorsEditable()) {\n <ng-container *ngTemplateOutlet=\"nodeErrorForm; context: { $implicit: prop2 }\" />\n }\n @if (prop2.error) {\n <ng-container\n *ngTemplateOutlet=\"\n propertyError;\n context: { property: prop2, edit: true, classes: 'is-my-1 is-px-3' }\n \" />\n }\n <div class=\"p-3\">\n <div class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-self-stretch\">\n @for (prop3 of prop2.properties; track trackByProperty($index, prop3)) {\n <ng-container *ngTemplateOutlet=\"showProperty; context: { $implicit: prop3 }\" />\n }\n </div>\n </div>\n </div>\n }\n }\n </div>\n } @else {\n @if (errorsEditable()) {\n <ng-container *ngTemplateOutlet=\"nodeErrorForm; context: { $implicit: property }\" />\n }\n <div class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-self-stretch\">\n @for (prop2 of property.properties; track trackByProperty($index, prop2)) {\n <ng-container *ngTemplateOutlet=\"showProperty; context: { $implicit: prop2 }\" />\n }\n </div>\n }\n }\n </div>\n } @else {\n <div\n class=\"is-flex is-flex-direction-column is-gap-4 | property-container\"\n [class.is-error-danger]=\"property.hasError\"\n [class.is-error-warning]=\"property.hasWarning\">\n <div\n class=\"is-flex is-flex-direction-row is-align-items-center is-justify-content-space-between is-gap-4 | field-container\">\n @if (property.key) {\n <div class=\"is-flex is-align-items-center is-flex-wrap-wrap is-gap-4\">\n <div class=\"is-flex has-text-secondary\">\n <ng-container *ngTemplateOutlet=\"propertyKey; context: { property }\" />\n <span>:</span>\n </div>\n\n <span class=\"is-flex is-size-7 | property-value\">\n {{ property.value }}\n </span>\n\n <div class=\"copy-button\">\n <he-clipboard [value]=\"property.value\" [hideText]=\"true\" [icon]=\"farClone\"></he-clipboard>\n </div>\n </div>\n\n <div class=\"is-flex is-gap-8\">\n @if (addErrorEnabled(property)) {\n <he-popover-confirm\n class=\"is-flex\"\n position=\"left\"\n [content]=\"popupErrorForm\"\n (confirmed)=\"addError(property, $event)\">\n <fa-icon [icon]=\"faComments\" />\n </he-popover-confirm>\n }\n @if (property.externalUrl?.url) {\n <a\n [href]=\"property.externalUrl.url + (property.externalUrl.urlParamValue ? property.value : '')\"\n target=\"_blank\"\n [title]=\"property.externalUrl.title\"\n [ngClass]=\"{ 'is-info': property.key === 'type' }\"\n [attr.disabled]=\"property.externalUrl.urlParamValue && !property.value ? true : null\">\n <fa-icon [icon]=\"property.externalUrl.icon || faExternalLinkAlt\" />\n </a>\n }\n </div>\n }\n </div>\n @if (property.hasError || property.hasWarning) {\n <ng-container *ngTemplateOutlet=\"propertyError; context: { property: property, edit: false }\" />\n }\n </div>\n }\n }\n</ng-template>\n\n<ng-template #propertyError let-property=\"property\" let-edit=\"edit\" let-classes=\"classes\">\n @if (property.error) {\n <div\n class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-items-center is-gap-8 is-size-6 is-m-0 w-100 has-text-grey {{\n classes\n }} | property-error\"\n [class.is-error-danger]=\"property.hasError\"\n [class.is-error-warning]=\"property.hasWarning\">\n @if (property.hasError) {\n <fa-icon class=\"has-text-danger is-py-1\" [icon]=\"farCircleXmark\" size=\"xl\" />\n } @else {\n <svg-icon class=\"has-text-warning is-pt-1\" name=\"exclamation-triangle\" />\n }\n <div [innerHTML]=\"property.error.message\"></div>\n @if (property.error.index >= 0) {\n @if (edit && errorsEditable()) {\n <a (click)=\"editError(property)\">\n <fa-icon class=\"pr-2\" [icon]=\"faEdit\" />\n <span>Edit</span>\n </a>\n }\n @if (property.hasWarning || errorsEditable()) {\n <a (click)=\"resolveError(property)\">\n <fa-icon class=\"pr-2\" [icon]=\"faCheck\" />\n <span>Resolved</span>\n </a>\n }\n }\n </div>\n }\n</ng-template>\n\n<ng-template #propertyMap let-property>\n @if (showMap()) {\n <div class=\"is-flex is-flex-direction-column is-flex-wrap-wrap w-100 | properties-container\">\n <div\n class=\"is-flex is-flex-direction-row is-justify-content-space-between is-gap-4 w-100 | properties-container--title\"\n (click)=\"mapVisible.set(!mapVisible())\"\n pointer\n [class.is-open]=\"mapVisible()\">\n <span class=\"is-size-6\">View on Map</span>\n <fa-icon [icon]=\"mapVisible() ? faMinus : faPlus\" />\n </div>\n @if (mapVisible()) {\n <he-sites-maps [sites]=\"[node()]\" [showNotice]=\"false\" />\n }\n </div>\n }\n</ng-template>\n\n<ng-template #nodeErrorForm let-property>\n @if (property.newError) {\n <div class=\"field has-addons w-100 is-p-2 is-m-0\">\n <div class=\"control\">\n <div class=\"select is-small\">\n <select [(ngModel)]=\"property.newError.level\">\n <option [value]=\"undefined\">Select Level</option>\n <option value=\"error\">Error</option>\n <option value=\"warning\">Warning</option>\n </select>\n </div>\n </div>\n <div class=\"control is-expanded\">\n <textarea\n class=\"textarea is-small\"\n [(ngModel)]=\"property.newError.message\"\n placeholder=\"Enter your message here\"\n rows=\"1\"></textarea>\n </div>\n <div class=\"control\">\n <button\n class=\"button is-small\"\n [disabled]=\"!property.newError.level || !property.newError.message\"\n (click)=\"addError(property, property.newError)\">\n <fa-icon [icon]=\"faPlusCircle\" />\n <span class=\"pl-2\">Add</span>\n </button>\n </div>\n </div>\n }\n</ng-template>\n\n<ng-template #popupErrorForm let-data=\"data\">\n <div class=\"field has-addons\">\n <div class=\"control\">\n <div class=\"select\">\n <select [(ngModel)]=\"data.level\">\n <option [value]=\"undefined\">Select Level</option>\n <option value=\"error\">Error</option>\n <option value=\"warning\">Warning</option>\n </select>\n </div>\n </div>\n <div class=\"control\">\n <input class=\"input\" [(ngModel)]=\"data.message\" placeholder=\"Enter your message here\" />\n </div>\n </div>\n</ng-template>\n", styles: [".notification{color:#0a0a0a!important}.notification.is-success{background-color:#d5f3d8}.notification.is-success *:not(.button):not(.icon):not(.dropdown-item)>fa-icon,.notification.is-success>fa-icon{color:#48c78e}.notification.is-info{background-color:#d3ebed}.notification.is-info *:not(.button):not(.icon):not(.dropdown-item)>fa-icon,.notification.is-info>fa-icon{color:#249da5}.notification.is-warning{background-color:#ffdec0}.notification.is-warning *:not(.button):not(.icon):not(.dropdown-item)>fa-icon,.notification.is-warning>fa-icon{color:#ff881b}.notification.is-danger{background-color:#ffcdd0}.notification.is-danger *:not(.button):not(.icon):not(.dropdown-item)>fa-icon,.notification.is-danger>fa-icon{color:#f14668}he-sites-maps{height:200px}.files-form-container{box-shadow:2px 2px 4px #00000029;background-color:#fff}.files-form-header{padding:10px 12px;background-color:#dbe3ea;border-top-left-radius:3px;border-top-right-radius:3px}.properties-container--title,.property-container{padding:6px 12px}.property-container{flex:none;flex-grow:1;min-width:50%;max-width:100%;border:1px solid #f5f5f5}.property-container.is-error-danger{border-color:#f14668}.property-container.is-error-warning{border-color:#ff881b}.properties-container--title{border-top:1px solid #dbe3ea;border-bottom:1px solid #dbe3ea;background:#f5f7f9}.properties-container .properties-container{border-radius:6px;border:1px solid #dbe3ea;margin:4px 0}.properties-container .properties-container--title{border-top:none}.property-key,.property-value{word-break:break-word}.property-array-container{border-radius:6px;border:1px solid #dbdbdb}.property-array-number .number-tags{border:1px solid #249da5;border-radius:50%;width:20px;height:20px}.property-array-number .number-tags .number-tag{color:#249da5}.copy-button{visibility:hidden}.field-container:hover .copy-button{visibility:visible}.is-group-error-danger{background-color:#ffcdd0}.is-group-error-warning{background-color:#ffdec0}.property-error.is-error-danger{background-color:#ffeced}.field-container+.property-error.is-error-danger{background-color:transparent;color:#e13939!important}.property-error.is-error-warning{background-color:#fff5ec}.field-container+.property-error.is-error-warning{background-color:transparent;color:#ff881b!important}.property-error.is-error-danger svg-icon,.property-error.is-error-danger fa-icon,.property-error.is-error-warning svg-icon,.property-error.is-error-warning fa-icon{width:24px}::ng-deep .property-tooltip{background-color:#fff;color:#0a0a0a;border:1px solid #dbdbdb;z-index:11;max-width:30vw}\n"] }]
|
|
125
|
+
], template: "<div class=\"is-flex is-flex-direction-column is-align-items-flex-start is-align-self-stretch | files-form-container\">\n <div\n class=\"is-flex is-flex-direction-row is-justify-content-space-between is-align-items-center is-align-self-stretch is-gap-16 px-4 py-2 has-text-secondary | files-form-header\"\n (click)=\"isOpen.set(!isOpen())\"\n pointer>\n <div class=\"is-flex is-gap-8 is-align-items-center\">\n @if (nodeProperty()) {\n <he-node-icon [type]=\"schemaType()\" />\n <span class=\"has-text-weight-bold has-text-secondary is-size-5\">{{ schemaType() }}</span>\n @if (nodeProperty().hasError || hasError()) {\n <svg-icon name=\"far-times-circle\" class=\"has-text-danger is-flex is-align-items-center\" />\n <span class=\"has-text-danger has-text-weight-bold is-italic\">Error</span>\n } @else if (nodeProperty().hasWarning || hasWarning()) {\n <svg-icon name=\"exclamation-triangle\" class=\"has-text-warning is-flex is-align-items-center\" />\n <span class=\"has-text-warning has-text-weight-bold is-italic\">Warning</span>\n }\n }\n </div>\n <div class=\"is-flex is-flex-direction-row is-align-items-center is-gap-16\">\n @if (nodeUrl()) {\n <a class=\"external-link\" [href]=\"nodeUrl()\" target=\"_blank\">\n <span>View</span>\n <span class=\"is-pl-1\">{{ schemaType() }}</span>\n <fa-icon class=\"is-pl-2\" [icon]=\"faExternalLinkAlt\" />\n </a>\n }\n <fa-icon [icon]=\"isOpen() ? faMinus : faPlus\" />\n </div>\n </div>\n\n <div class=\"is-flex is-flex-direction-column is-align-items-flex-start is-align-self-stretch\">\n @if (isOpen()) {\n @if (nodeProperty()?.error) {\n <ng-container\n *ngTemplateOutlet=\"propertyError; context: { property: nodeProperty(), classes: 'is-my-1 is-px-3' }\" />\n }\n }\n\n <div class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-self-stretch\">\n @for (property of properties(); track trackByProperty($index, property)) {\n <ng-container *ngTemplateOutlet=\"showProperty; context: { $implicit: property }\" />\n }\n </div>\n\n @if (isOpen()) {\n <ng-container *ngTemplateOutlet=\"propertyMap; context: { $implicit: nodeProperty() }\" />\n }\n </div>\n\n <ng-content />\n</div>\n\n<ng-template #propertyKeyTooltip let-property=\"property\">\n <markdown [data]=\"property.schema.description\" />\n</ng-template>\n\n<ng-template #propertyKey let-property=\"property\">\n @if (property.schema?.description) {\n <span\n class=\"property-key trigger-popover\"\n [ngbTooltip]=\"propertyKeyTooltip\"\n [tooltipContext]=\"{ property }\"\n tooltipClass=\"property-tooltip\"\n placement=\"bottom-left\"\n container=\"body\">\n <span>{{ property.key }}</span>\n </span>\n } @else {\n <span>{{ property.key }}</span>\n }\n</ng-template>\n\n<ng-template #showProperty let-property>\n @if ((isOpen() || property.closedVisible) && !property.isHidden) {\n @if (property.properties.length) {\n <div class=\"is-flex is-flex-direction-column is-flex-wrap-wrap w-100 | properties-container\">\n <div\n class=\"is-flex is-flex-direction-row is-justify-content-space-between is-gap-4 has-text-secondary w-100 | properties-container--title\"\n (click)=\"property.isOpen = !property.isOpen\"\n pointer\n [class.is-open]=\"property.isOpen\"\n [class.is-group-error-danger]=\"property.hasError\"\n [class.is-group-error-warning]=\"property.hasWarning\">\n <div class=\"is-flex is-gap-8 has-text-weight-bold\">\n <!-- @if (property.schemaType === SchemaType.Term) {\n <he-node-icon [type]=\"property.schemaType\" />\n } -->\n <ng-container *ngTemplateOutlet=\"propertyKey; context: { property }\" />\n </div>\n <fa-icon [icon]=\"property.isOpen ? faMinus : faPlus\" />\n </div>\n @if (property.isOpen) {\n @if (property.error) {\n <ng-container *ngTemplateOutlet=\"propertyError; context: { property, classes: 'is-my-1 is-px-3' }\" />\n }\n @if (property.isArray) {\n <div class=\"py-2 px-3 w-100\">\n @for (prop2 of property.properties; track trackByProperty($index, prop2)) {\n @if (prop2.key) {\n <div class=\"pt-2 my-4 | property-array-container\" [id]=\"prop2.fullKey + '_' + prop2.id\">\n <div class=\"property-array-number ml-2\">\n <div class=\"is-flex is-justify-content-center is-align-items-center has-addons number-tags\">\n <span class=\"number-tag\">{{ prop2.key }}</span>\n </div>\n </div>\n @if (prop2.error) {\n <ng-container\n *ngTemplateOutlet=\"propertyError; context: { property: prop2, classes: 'is-my-1 is-px-3' }\" />\n }\n <div class=\"p-3\">\n <div class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-self-stretch\">\n @for (prop3 of prop2.properties; track trackByProperty($index, prop3)) {\n <ng-container *ngTemplateOutlet=\"showProperty; context: { $implicit: prop3 }\" />\n }\n </div>\n </div>\n </div>\n }\n }\n </div>\n } @else {\n <div class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-self-stretch\">\n @for (prop2 of property.properties; track trackByProperty($index, prop2)) {\n <ng-container *ngTemplateOutlet=\"showProperty; context: { $implicit: prop2 }\" />\n }\n </div>\n }\n }\n </div>\n } @else {\n <div\n class=\"is-flex is-flex-direction-column is-gap-4 | property-container\"\n [class.is-error-danger]=\"property.hasError\"\n [class.is-error-warning]=\"property.hasWarning\">\n <div\n class=\"is-flex is-flex-direction-row is-align-items-center is-justify-content-space-between is-gap-4 | field-container\">\n @if (property.key) {\n <div class=\"is-flex is-align-items-center is-flex-wrap-wrap is-gap-4\">\n <div class=\"is-flex has-text-secondary\">\n <ng-container *ngTemplateOutlet=\"propertyKey; context: { property }\" />\n <span>:</span>\n </div>\n\n <span class=\"is-flex is-size-7 | property-value\">\n {{ property.value }}\n </span>\n\n <div class=\"copy-button\">\n <he-clipboard [value]=\"property.value\" [hideText]=\"true\" [icon]=\"farClone\"></he-clipboard>\n </div>\n </div>\n\n <div class=\"is-flex is-gap-8\">\n @if (property.externalUrl?.url) {\n <a\n [href]=\"property.externalUrl.url + (property.externalUrl.urlParamValue ? property.value : '')\"\n target=\"_blank\"\n [title]=\"property.externalUrl.title\"\n [ngClass]=\"{ 'is-info': property.key === 'type' }\"\n [attr.disabled]=\"property.externalUrl.urlParamValue && !property.value ? true : null\">\n <fa-icon [icon]=\"property.externalUrl.icon || faExternalLinkAlt\" />\n </a>\n }\n </div>\n }\n </div>\n @if (property.hasError || property.hasWarning) {\n <ng-container *ngTemplateOutlet=\"propertyError; context: { property }\" />\n }\n </div>\n }\n }\n</ng-template>\n\n<ng-template #propertyError let-property=\"property\" let-classes=\"classes\">\n @if (property.error) {\n <div\n class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-items-center is-gap-8 is-size-6 is-m-0 w-100 has-text-grey {{\n classes\n }} | property-error\"\n [class.is-error-danger]=\"property.hasError\"\n [class.is-error-warning]=\"property.hasWarning\">\n @if (property.hasError) {\n <fa-icon class=\"has-text-danger is-py-1\" [icon]=\"farCircleXmark\" size=\"xl\" />\n } @else {\n <svg-icon class=\"has-text-warning is-pt-1\" name=\"exclamation-triangle\" />\n }\n <div [innerHTML]=\"property.error.message\"></div>\n @if (property.error.index >= 0) {\n @if (property.hasWarning) {\n <a (click)=\"resolveError(property)\">\n <fa-icon class=\"pr-2\" [icon]=\"faCheck\" />\n <span>Resolved</span>\n </a>\n }\n }\n </div>\n }\n</ng-template>\n\n<ng-template #propertyMap let-property>\n @if (showMap()) {\n <div class=\"is-flex is-flex-direction-column is-flex-wrap-wrap w-100 | properties-container\">\n <div\n class=\"is-flex is-flex-direction-row is-justify-content-space-between is-gap-4 w-100 | properties-container--title\"\n (click)=\"mapVisible.set(!mapVisible())\"\n pointer\n [class.is-open]=\"mapVisible()\">\n <span class=\"is-size-6\">View on Map</span>\n <fa-icon [icon]=\"mapVisible() ? faMinus : faPlus\" />\n </div>\n @if (mapVisible()) {\n <he-sites-maps [sites]=\"[node()]\" [showNotice]=\"false\" />\n }\n </div>\n }\n</ng-template>\n", styles: [".notification{color:#0a0a0a!important}.notification.is-success{background-color:#d5f3d8}.notification.is-success *:not(.button):not(.icon):not(.dropdown-item)>fa-icon,.notification.is-success>fa-icon{color:#48c78e}.notification.is-info{background-color:#d3ebed}.notification.is-info *:not(.button):not(.icon):not(.dropdown-item)>fa-icon,.notification.is-info>fa-icon{color:#249da5}.notification.is-warning{background-color:#ffdec0}.notification.is-warning *:not(.button):not(.icon):not(.dropdown-item)>fa-icon,.notification.is-warning>fa-icon{color:#ff881b}.notification.is-danger{background-color:#ffcdd0}.notification.is-danger *:not(.button):not(.icon):not(.dropdown-item)>fa-icon,.notification.is-danger>fa-icon{color:#f14668}he-sites-maps{height:200px}.external-link{color:#4c7194}.files-form-container{box-shadow:2px 2px 4px #00000029;background-color:#fff}.files-form-header{padding:10px 12px;background-color:#dbe3ea;border-top-left-radius:3px;border-top-right-radius:3px}.properties-container--title,.property-container{padding:6px 12px}.property-container{flex:none;flex-grow:1;min-width:50%;max-width:100%;border:1px solid #f5f5f5}.property-container.is-error-danger{border-color:#f14668}.property-container.is-error-warning{border-color:#ff881b}.properties-container--title{border-top:1px solid #dbe3ea;border-bottom:1px solid #dbe3ea;background:#f5f7f9}.properties-container .properties-container{border-radius:6px;border:1px solid #dbe3ea;margin:4px 0}.properties-container .properties-container--title{border-top:none}.property-key,.property-value{word-break:break-word}.property-array-container{border-radius:6px;border:1px solid #dbdbdb}.property-array-number .number-tags{border:1px solid #249da5;border-radius:50%;width:20px;height:20px}.property-array-number .number-tags .number-tag{color:#249da5}.copy-button{visibility:hidden}.field-container:hover .copy-button{visibility:visible}.is-group-error-danger{background-color:#ffcdd0}.is-group-error-warning{background-color:#ffdec0}.property-error.is-error-danger{background-color:#ffeced}.field-container+.property-error.is-error-danger{background-color:transparent;color:#e13939!important}.property-error.is-error-warning{background-color:#fff5ec}.field-container+.property-error.is-error-warning{background-color:transparent;color:#ff881b!important}.property-error.is-error-danger svg-icon,.property-error.is-error-danger fa-icon,.property-error.is-error-warning svg-icon,.property-error.is-error-warning fa-icon{width:24px}::ng-deep .property-tooltip{background-color:#fff;color:#0a0a0a;border:1px solid #dbdbdb;z-index:11;max-width:30vw}\n"] }]
|
|
153
126
|
}] });
|
|
154
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"files-form.component.js","sourceRoot":"","sources":["../../../../src/files/files-form/files-form.component.ts","../../../../src/files/files-form/files-form.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ElementRef,\n  output,\n  inject,\n  input,\n  computed,\n  signal,\n  ChangeDetectionStrategy,\n  model\n} from '@angular/core';\nimport { NgTemplateOutlet, NgClass } from '@angular/common';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { FormsModule } from '@angular/forms';\nimport { JSON as HestiaJson, SchemaType, NodeType, Site } from '@hestia-earth/schema';\nimport { definition } from '@hestia-earth/json-schema';\nimport { FaIconComponent } from '@fortawesome/angular-fontawesome';\nimport {\n  faCheck,\n  faComments,\n  faEdit,\n  faExternalLinkAlt,\n  faPlusCircle,\n  faMinus,\n  faPlus,\n  faClone as farClone\n} from '@fortawesome/free-solid-svg-icons';\nimport { faCircleXmark } from '@fortawesome/free-regular-svg-icons';\nimport { NgbTooltip } from '@ng-bootstrap/ng-bootstrap';\nimport { isUndefined } from '@hestia-earth/utils';\nimport { SvgIconComponent } from 'angular-svg-icon';\nimport { MarkdownComponent } from 'ngx-markdown';\n\nimport {\n  IErrorProperty,\n  INodeProperty,\n  formatPropertyError,\n  propertyError,\n  hasError,\n  hasWarning,\n  recursiveProperties,\n  propertyId,\n  groupChanged,\n  keyToDataPath,\n  formatter\n} from '../files-form.model';\nimport { errorHasError, errorHasWarning, filterError, formatError } from '../files-error.model';\nimport { IValidationError } from '../../schema/schema-validation.model';\nimport { PopoverConfirmComponent } from '../../common/popover-confirm/popover-confirm.component';\nimport { SchemaVersionLinkComponent } from '../../common/schema-version-link/schema-version-link.component';\nimport { SchemaInfoComponent } from '../../schema/schema-info/schema-info.component';\nimport { HeSchemaService } from '../../schema/schema.service';\nimport { NodeIconComponent } from '../../node/node-icon/node-icon.component';\nimport { SitesMapsComponent } from '../../sites/sites-maps/sites-maps.component';\nimport { ClipboardComponent } from '../../common';\n\nconst stringify = (value: any) => JSON.stringify(value);\n\n@Component({\n  selector: 'he-files-form',\n  templateUrl: './files-form.component.html',\n  styleUrls: ['./files-form.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [\n    FormsModule,\n    NgTemplateOutlet,\n    NgClass,\n    FaIconComponent,\n    MarkdownComponent,\n    NgbTooltip,\n    NodeIconComponent,\n    SchemaInfoComponent,\n    SchemaVersionLinkComponent,\n    PopoverConfirmComponent,\n    SitesMapsComponent,\n    SvgIconComponent,\n    ClipboardComponent\n  ]\n})\nexport class FilesFormComponent {\n  /**\n   * Query the content.\n   */\n  public readonly ref = inject(ElementRef);\n  private readonly schemaService = inject(HeSchemaService);\n\n  protected readonly faMinus = faMinus;\n  protected readonly faPlus = faPlus;\n  protected readonly faPlusCircle = faPlusCircle;\n  protected readonly farCircleXmark = faCircleXmark;\n  protected readonly faComments = faComments;\n  protected readonly faExternalLinkAlt = faExternalLinkAlt;\n  protected readonly faEdit = faEdit;\n  protected readonly faCheck = faCheck;\n  protected readonly farClone = farClone;\n\n  protected readonly isOpen = model(true);\n  protected readonly node = input.required<HestiaJson<SchemaType>>();\n  protected readonly errors = input<IValidationError[]>([]);\n  protected readonly errorsEditable = input(false);\n\n  private readonly schemas = toSignal(this.schemaService.schemas$);\n  protected readonly schemaType = computed(() => this.node()?.type || this.node()?.['@type']);\n  private readonly schema = computed(() => this.schemas()?.[this.schemaType()] ?? ({} as definition));\n\n  protected readonly nodeErorrResolved = output<number>();\n  protected readonly nodeErrorAdded = output<IValidationError>();\n\n  protected readonly NodeType = NodeType;\n  protected readonly SchemaType = SchemaType;\n  protected readonly formatPropertyError = formatPropertyError;\n  protected readonly stringify = stringify;\n\n  protected readonly mapVisible = signal(false);\n\n  protected readonly showMap = computed(\n    () =>\n      [SchemaType.Site, SchemaType.Organisation].includes(this.schemaType()) &&\n      [\n        (this.node() as Site).latitude && (this.node() as Site).longitude,\n        (this.node() as Site).region?.name,\n        (this.node() as Site).country?.name\n      ].some(Boolean)\n  );\n\n  private readonly filteredErrors = computed(() => this.errors().filter(filterError));\n  private readonly nodeError = computed(() => propertyError(this.filteredErrors())(''));\n\n  protected readonly nodeProperty = computed(() =>\n    this.schema()\n      ? {\n          id: propertyId(),\n          schema: this.schema(),\n          schemaType: this.schemaType(),\n          editable: false,\n          fullKey: '',\n          key: '',\n          value: {},\n          error: formatError(this.nodeError()),\n          hasError: errorHasError(this.nodeError()),\n          hasWarning: errorHasWarning(this.nodeError()),\n          suggestions: {},\n          newProperty: {},\n          newError: { level: 'warning' },\n          properties: [],\n          addPropertyEnabled: false,\n          formatter: formatter('')\n        }\n      : undefined\n  );\n\n  protected readonly properties = computed(() =>\n    [!isUndefined(this.schemas()), !isUndefined(this.node())].every(Boolean)\n      ? recursiveProperties(this.schemas(), this.filteredErrors(), this.schema(), false)(this.node())\n      : []\n  );\n\n  protected readonly hasError = computed(() => hasError(this.properties()));\n  protected readonly hasWarning = computed(() => hasWarning(this.properties()));\n\n  protected trackByProperty(_index: number, { id, fullKey }: INodeProperty) {\n    return [fullKey, id].join('_');\n  }\n\n  // --- Errors\n\n  protected addErrorEnabled(property: INodeProperty) {\n    return this.errorsEditable() && !property.hasError && !property.hasWarning;\n  }\n\n  protected addError(property: INodeProperty, { message, level }: IErrorProperty) {\n    if (!message) {\n      return;\n    }\n    const keys = property.fullKey.split('.');\n    keys.pop();\n    const error: IValidationError = property.isAdded\n      ? {\n          message,\n          level: level || 'error',\n          dataPath: keyToDataPath(keys.join('.')),\n          params: {\n            missingProperty: property.key\n          }\n        }\n      : {\n          message,\n          level: level || 'error',\n          dataPath: keyToDataPath(property.fullKey)\n        };\n    property.newError = { level: 'warning' };\n    property.error = error;\n    property.hasError = errorHasError(error);\n    property.hasWarning = errorHasWarning(error);\n    this.nodeErrorAdded.emit(error);\n  }\n\n  protected editError(property: INodeProperty) {\n    property.newError = property.error;\n  }\n\n  protected resolveError(property: INodeProperty) {\n    const index = property.error?.index;\n    property.error = undefined;\n    property.hasError = hasError(property.properties);\n    property.hasWarning = hasWarning(property.properties);\n    this.nodeErorrResolved.emit(index);\n    return groupChanged(this.properties(), property.fullKey);\n  }\n}\n","<div class=\"is-flex is-flex-direction-column is-align-items-flex-start is-align-self-stretch | files-form-container\">\n  <div\n    class=\"is-flex is-flex-direction-row is-justify-content-space-between is-align-items-center is-align-self-stretch is-gap-16 px-4 py-2 has-text-secondary | files-form-header\"\n    (click)=\"isOpen.set(!isOpen())\"\n    pointer>\n    <div class=\"is-flex is-gap-8 is-align-items-center\">\n      @if (nodeProperty()) {\n        <he-node-icon [type]=\"schemaType()\" />\n        <span class=\"has-text-weight-bold has-text-secondary is-size-5\">{{ schemaType() }}</span>\n        @if (nodeProperty().hasError || hasError()) {\n          <svg-icon name=\"far-times-circle\" class=\"has-text-danger is-flex is-align-items-center\" />\n          <span class=\"has-text-danger has-text-weight-bold is-italic\">Error</span>\n        } @else if (nodeProperty().hasWarning || hasWarning()) {\n          <svg-icon name=\"exclamation-triangle\" class=\"has-text-warning is-flex is-align-items-center\" />\n          <span class=\"has-text-warning has-text-weight-bold is-italic\">Warning</span>\n        }\n      }\n    </div>\n    <fa-icon [icon]=\"isOpen() ? faMinus : faPlus\" />\n  </div>\n\n  <div class=\"is-flex is-flex-direction-column is-align-items-flex-start is-align-self-stretch\">\n    @if (isOpen()) {\n      @if (errorsEditable()) {\n        <ng-container *ngTemplateOutlet=\"nodeErrorForm; context: { $implicit: nodeProperty() }\" />\n      }\n      @if (nodeProperty()?.error) {\n        <ng-container\n          *ngTemplateOutlet=\"\n            propertyError;\n            context: { property: nodeProperty(), edit: true, classes: 'is-my-1 is-px-3' }\n          \" />\n      }\n    }\n\n    <div class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-self-stretch\">\n      @for (property of properties(); track trackByProperty($index, property)) {\n        <ng-container *ngTemplateOutlet=\"showProperty; context: { $implicit: property }\" />\n      }\n    </div>\n\n    @if (isOpen()) {\n      <ng-container *ngTemplateOutlet=\"propertyMap; context: { $implicit: nodeProperty() }\" />\n    }\n  </div>\n\n  <ng-content></ng-content>\n</div>\n\n<ng-template #propertyKeyTooltip let-property=\"property\">\n  <markdown [data]=\"property.schema.description\" />\n</ng-template>\n\n<ng-template #propertyKey let-property=\"property\">\n  @if (property.schema?.description) {\n    <span\n      class=\"property-key trigger-popover\"\n      [ngbTooltip]=\"propertyKeyTooltip\"\n      [tooltipContext]=\"{ property }\"\n      tooltipClass=\"property-tooltip\"\n      placement=\"bottom-left\"\n      container=\"body\">\n      <span>{{ property.key }}</span>\n    </span>\n  } @else {\n    <span>{{ property.key }}</span>\n  }\n</ng-template>\n\n<ng-template #showProperty let-property>\n  @if ((isOpen() || property.closedVisible) && !property.isHidden) {\n    @if (property.properties.length) {\n      <div class=\"is-flex is-flex-direction-column is-flex-wrap-wrap w-100 | properties-container\">\n        <div\n          class=\"is-flex is-flex-direction-row is-justify-content-space-between is-gap-4 has-text-secondary w-100 | properties-container--title\"\n          (click)=\"property.isOpen = !property.isOpen\"\n          pointer\n          [class.is-open]=\"property.isOpen\"\n          [class.is-group-error-danger]=\"property.hasError\"\n          [class.is-group-error-warning]=\"property.hasWarning\">\n          <div class=\"is-flex is-gap-8 has-text-weight-bold\">\n            <!-- @if (property.schemaType === SchemaType.Term) {\n              <he-node-icon [type]=\"property.schemaType\" />\n            } -->\n            <ng-container *ngTemplateOutlet=\"propertyKey; context: { property }\" />\n          </div>\n          <fa-icon [icon]=\"property.isOpen ? faMinus : faPlus\" />\n        </div>\n        @if (property.isOpen) {\n          @if (property.error) {\n            <ng-container\n              *ngTemplateOutlet=\"\n                propertyError;\n                context: { property: property, edit: true, classes: 'is-my-1 is-px-3' }\n              \" />\n          }\n          @if (property.isArray) {\n            <div class=\"py-2 px-3 w-100\">\n              @if (errorsEditable()) {\n                <ng-container *ngTemplateOutlet=\"nodeErrorForm; context: { $implicit: property }\" />\n              }\n              @for (prop2 of property.properties; track trackByProperty($index, prop2)) {\n                @if (prop2.key) {\n                  <div class=\"pt-2 my-4 | property-array-container\" [id]=\"prop2.fullKey + '_' + prop2.id\">\n                    <div class=\"property-array-number ml-2\">\n                      <div class=\"is-flex is-justify-content-center is-align-items-center has-addons number-tags\">\n                        <span class=\"number-tag\">{{ prop2.key }}</span>\n                      </div>\n                    </div>\n                    @if (errorsEditable()) {\n                      <ng-container *ngTemplateOutlet=\"nodeErrorForm; context: { $implicit: prop2 }\" />\n                    }\n                    @if (prop2.error) {\n                      <ng-container\n                        *ngTemplateOutlet=\"\n                          propertyError;\n                          context: { property: prop2, edit: true, classes: 'is-my-1 is-px-3' }\n                        \" />\n                    }\n                    <div class=\"p-3\">\n                      <div class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-self-stretch\">\n                        @for (prop3 of prop2.properties; track trackByProperty($index, prop3)) {\n                          <ng-container *ngTemplateOutlet=\"showProperty; context: { $implicit: prop3 }\" />\n                        }\n                      </div>\n                    </div>\n                  </div>\n                }\n              }\n            </div>\n          } @else {\n            @if (errorsEditable()) {\n              <ng-container *ngTemplateOutlet=\"nodeErrorForm; context: { $implicit: property }\" />\n            }\n            <div class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-self-stretch\">\n              @for (prop2 of property.properties; track trackByProperty($index, prop2)) {\n                <ng-container *ngTemplateOutlet=\"showProperty; context: { $implicit: prop2 }\" />\n              }\n            </div>\n          }\n        }\n      </div>\n    } @else {\n      <div\n        class=\"is-flex is-flex-direction-column is-gap-4 | property-container\"\n        [class.is-error-danger]=\"property.hasError\"\n        [class.is-error-warning]=\"property.hasWarning\">\n        <div\n          class=\"is-flex is-flex-direction-row is-align-items-center is-justify-content-space-between is-gap-4 | field-container\">\n          @if (property.key) {\n            <div class=\"is-flex is-align-items-center is-flex-wrap-wrap is-gap-4\">\n              <div class=\"is-flex has-text-secondary\">\n                <ng-container *ngTemplateOutlet=\"propertyKey; context: { property }\" />\n                <span>:</span>\n              </div>\n\n              <span class=\"is-flex is-size-7 | property-value\">\n                {{ property.value }}\n              </span>\n\n              <div class=\"copy-button\">\n                <he-clipboard [value]=\"property.value\" [hideText]=\"true\" [icon]=\"farClone\"></he-clipboard>\n              </div>\n            </div>\n\n            <div class=\"is-flex is-gap-8\">\n              @if (addErrorEnabled(property)) {\n                <he-popover-confirm\n                  class=\"is-flex\"\n                  position=\"left\"\n                  [content]=\"popupErrorForm\"\n                  (confirmed)=\"addError(property, $event)\">\n                  <fa-icon [icon]=\"faComments\" />\n                </he-popover-confirm>\n              }\n              @if (property.externalUrl?.url) {\n                <a\n                  [href]=\"property.externalUrl.url + (property.externalUrl.urlParamValue ? property.value : '')\"\n                  target=\"_blank\"\n                  [title]=\"property.externalUrl.title\"\n                  [ngClass]=\"{ 'is-info': property.key === 'type' }\"\n                  [attr.disabled]=\"property.externalUrl.urlParamValue && !property.value ? true : null\">\n                  <fa-icon [icon]=\"property.externalUrl.icon || faExternalLinkAlt\" />\n                </a>\n              }\n            </div>\n          }\n        </div>\n        @if (property.hasError || property.hasWarning) {\n          <ng-container *ngTemplateOutlet=\"propertyError; context: { property: property, edit: false }\" />\n        }\n      </div>\n    }\n  }\n</ng-template>\n\n<ng-template #propertyError let-property=\"property\" let-edit=\"edit\" let-classes=\"classes\">\n  @if (property.error) {\n    <div\n      class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-items-center is-gap-8 is-size-6 is-m-0 w-100 has-text-grey {{\n        classes\n      }} | property-error\"\n      [class.is-error-danger]=\"property.hasError\"\n      [class.is-error-warning]=\"property.hasWarning\">\n      @if (property.hasError) {\n        <fa-icon class=\"has-text-danger is-py-1\" [icon]=\"farCircleXmark\" size=\"xl\" />\n      } @else {\n        <svg-icon class=\"has-text-warning is-pt-1\" name=\"exclamation-triangle\" />\n      }\n      <div [innerHTML]=\"property.error.message\"></div>\n      @if (property.error.index >= 0) {\n        @if (edit && errorsEditable()) {\n          <a (click)=\"editError(property)\">\n            <fa-icon class=\"pr-2\" [icon]=\"faEdit\" />\n            <span>Edit</span>\n          </a>\n        }\n        @if (property.hasWarning || errorsEditable()) {\n          <a (click)=\"resolveError(property)\">\n            <fa-icon class=\"pr-2\" [icon]=\"faCheck\" />\n            <span>Resolved</span>\n          </a>\n        }\n      }\n    </div>\n  }\n</ng-template>\n\n<ng-template #propertyMap let-property>\n  @if (showMap()) {\n    <div class=\"is-flex is-flex-direction-column is-flex-wrap-wrap w-100 | properties-container\">\n      <div\n        class=\"is-flex is-flex-direction-row is-justify-content-space-between is-gap-4 w-100 | properties-container--title\"\n        (click)=\"mapVisible.set(!mapVisible())\"\n        pointer\n        [class.is-open]=\"mapVisible()\">\n        <span class=\"is-size-6\">View on Map</span>\n        <fa-icon [icon]=\"mapVisible() ? faMinus : faPlus\" />\n      </div>\n      @if (mapVisible()) {\n        <he-sites-maps [sites]=\"[node()]\" [showNotice]=\"false\" />\n      }\n    </div>\n  }\n</ng-template>\n\n<ng-template #nodeErrorForm let-property>\n  @if (property.newError) {\n    <div class=\"field has-addons w-100 is-p-2 is-m-0\">\n      <div class=\"control\">\n        <div class=\"select is-small\">\n          <select [(ngModel)]=\"property.newError.level\">\n            <option [value]=\"undefined\">Select Level</option>\n            <option value=\"error\">Error</option>\n            <option value=\"warning\">Warning</option>\n          </select>\n        </div>\n      </div>\n      <div class=\"control is-expanded\">\n        <textarea\n          class=\"textarea is-small\"\n          [(ngModel)]=\"property.newError.message\"\n          placeholder=\"Enter your message here\"\n          rows=\"1\"></textarea>\n      </div>\n      <div class=\"control\">\n        <button\n          class=\"button is-small\"\n          [disabled]=\"!property.newError.level || !property.newError.message\"\n          (click)=\"addError(property, property.newError)\">\n          <fa-icon [icon]=\"faPlusCircle\" />\n          <span class=\"pl-2\">Add</span>\n        </button>\n      </div>\n    </div>\n  }\n</ng-template>\n\n<ng-template #popupErrorForm let-data=\"data\">\n  <div class=\"field has-addons\">\n    <div class=\"control\">\n      <div class=\"select\">\n        <select [(ngModel)]=\"data.level\">\n          <option [value]=\"undefined\">Select Level</option>\n          <option value=\"error\">Error</option>\n          <option value=\"warning\">Warning</option>\n        </select>\n      </div>\n    </div>\n    <div class=\"control\">\n      <input class=\"input\" [(ngModel)]=\"data.message\" placeholder=\"Enter your message here\" />\n    </div>\n  </div>\n</ng-template>\n"]}
|
|
127
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"files-form.component.js","sourceRoot":"","sources":["../../../../src/files/files-form/files-form.component.ts","../../../../src/files/files-form/files-form.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ElementRef,\n  output,\n  inject,\n  input,\n  computed,\n  signal,\n  ChangeDetectionStrategy,\n  model\n} from '@angular/core';\nimport { NgTemplateOutlet, NgClass } from '@angular/common';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { FormsModule } from '@angular/forms';\nimport { JSON as HestiaJson, SchemaType, NodeType, Site } from '@hestia-earth/schema';\nimport { definition } from '@hestia-earth/json-schema';\nimport { FaIconComponent } from '@fortawesome/angular-fontawesome';\nimport {\n  faCheck,\n  faComments,\n  faEdit,\n  faExternalLinkAlt,\n  faPlusCircle,\n  faMinus,\n  faPlus,\n  faClone as farClone\n} from '@fortawesome/free-solid-svg-icons';\nimport { faCircleXmark } from '@fortawesome/free-regular-svg-icons';\nimport { NgbTooltip } from '@ng-bootstrap/ng-bootstrap';\nimport { isUndefined } from '@hestia-earth/utils';\nimport { SvgIconComponent } from 'angular-svg-icon';\nimport { MarkdownComponent } from 'ngx-markdown';\n\nimport {\n  INodeProperty,\n  formatPropertyError,\n  propertyError,\n  hasError,\n  hasWarning,\n  recursiveProperties,\n  propertyId,\n  groupChanged,\n  formatter\n} from '../files-form.model';\nimport { errorHasError, errorHasWarning, filterError, formatError } from '../files-error.model';\nimport { IValidationError } from '../../schema/schema-validation.model';\nimport { PopoverConfirmComponent } from '../../common/popover-confirm/popover-confirm.component';\nimport { SchemaVersionLinkComponent } from '../../common/schema-version-link/schema-version-link.component';\nimport { SchemaInfoComponent } from '../../schema/schema-info/schema-info.component';\nimport { HeSchemaService, linkTypeEnabled } from '../../schema/schema.service';\nimport { NodeIconComponent } from '../../node/node-icon/node-icon.component';\nimport { SitesMapsComponent } from '../../sites/sites-maps/sites-maps.component';\nimport { ClipboardComponent, isExternal } from '../../common';\n\nconst stringify = (value: any) => JSON.stringify(value);\n\n@Component({\n  selector: 'he-files-form',\n  templateUrl: './files-form.component.html',\n  styleUrls: ['./files-form.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [\n    FormsModule,\n    NgTemplateOutlet,\n    NgClass,\n    FaIconComponent,\n    MarkdownComponent,\n    NgbTooltip,\n    NodeIconComponent,\n    SchemaInfoComponent,\n    SchemaVersionLinkComponent,\n    PopoverConfirmComponent,\n    SitesMapsComponent,\n    SvgIconComponent,\n    ClipboardComponent\n  ]\n})\nexport class FilesFormComponent {\n  /**\n   * Query the content.\n   */\n  public readonly ref = inject(ElementRef);\n  private readonly schemaService = inject(HeSchemaService);\n\n  protected readonly faMinus = faMinus;\n  protected readonly faPlus = faPlus;\n  protected readonly faPlusCircle = faPlusCircle;\n  protected readonly farCircleXmark = faCircleXmark;\n  protected readonly faComments = faComments;\n  protected readonly faExternalLinkAlt = faExternalLinkAlt;\n  protected readonly faEdit = faEdit;\n  protected readonly faCheck = faCheck;\n  protected readonly farClone = farClone;\n\n  protected readonly isOpen = model(true);\n  protected readonly node = input.required<HestiaJson<SchemaType>>();\n  protected readonly errors = input<IValidationError[]>([]);\n\n  private readonly schemas = toSignal(this.schemaService.schemas$);\n  protected readonly schemaType = computed(() => this.node()?.type || this.node()?.['@type']);\n  private readonly schema = computed(() => this.schemas()?.[this.schemaType()] ?? ({} as definition));\n\n  protected readonly nodeErorrResolved = output<number>();\n\n  protected readonly NodeType = NodeType;\n  protected readonly SchemaType = SchemaType;\n  protected readonly formatPropertyError = formatPropertyError;\n  protected readonly stringify = stringify;\n\n  protected readonly mapVisible = signal(false);\n\n  protected readonly showMap = computed(\n    () =>\n      [SchemaType.Site, SchemaType.Organisation].includes(this.schemaType()) &&\n      [\n        (this.node() as Site).latitude && (this.node() as Site).longitude,\n        (this.node() as Site).region?.name,\n        (this.node() as Site).country?.name\n      ].some(Boolean)\n  );\n  private readonly nodeId = computed(() => this.node()?.id || this.node()?.['@id']);\n  protected readonly nodeUrl = computed(() =>\n    [\n      linkTypeEnabled(this.schemaType()),\n      // handle links on Community Edition\n      this.node()?.['@id'] || (isExternal() && this.nodeId())\n    ].every(Boolean)\n      ? `/${this.schemaType().toLowerCase()}/${this.nodeId()}`\n      : ''\n  );\n\n  private readonly filteredErrors = computed(() => this.errors().filter(filterError));\n  private readonly nodeError = computed(() => propertyError(this.filteredErrors())(''));\n\n  protected readonly nodeProperty = computed(() =>\n    this.schema()\n      ? {\n          id: propertyId(),\n          schema: this.schema(),\n          schemaType: this.schemaType(),\n          editable: false,\n          fullKey: '',\n          key: '',\n          value: {},\n          error: formatError(this.nodeError()),\n          hasError: errorHasError(this.nodeError()),\n          hasWarning: errorHasWarning(this.nodeError()),\n          suggestions: {},\n          newProperty: {},\n          newError: { level: 'warning' },\n          properties: [],\n          addPropertyEnabled: false,\n          formatter: formatter('')\n        }\n      : undefined\n  );\n\n  protected readonly properties = computed(() =>\n    [!isUndefined(this.schemas()), !isUndefined(this.node())].every(Boolean)\n      ? recursiveProperties(this.schemas(), this.filteredErrors(), this.schema(), false)(this.node())\n      : []\n  );\n\n  protected readonly hasError = computed(() => hasError(this.properties()));\n  protected readonly hasWarning = computed(() => hasWarning(this.properties()));\n\n  protected trackByProperty(_index: number, { id, fullKey }: INodeProperty) {\n    return [fullKey, id].join('_');\n  }\n\n  // --- Errors\n\n  protected resolveError(property: INodeProperty) {\n    const index = property.error?.index;\n    property.error = undefined;\n    property.hasError = hasError(property.properties);\n    property.hasWarning = hasWarning(property.properties);\n    this.nodeErorrResolved.emit(index);\n    return groupChanged(this.properties(), property.fullKey);\n  }\n}\n","<div class=\"is-flex is-flex-direction-column is-align-items-flex-start is-align-self-stretch | files-form-container\">\n  <div\n    class=\"is-flex is-flex-direction-row is-justify-content-space-between is-align-items-center is-align-self-stretch is-gap-16 px-4 py-2 has-text-secondary | files-form-header\"\n    (click)=\"isOpen.set(!isOpen())\"\n    pointer>\n    <div class=\"is-flex is-gap-8 is-align-items-center\">\n      @if (nodeProperty()) {\n        <he-node-icon [type]=\"schemaType()\" />\n        <span class=\"has-text-weight-bold has-text-secondary is-size-5\">{{ schemaType() }}</span>\n        @if (nodeProperty().hasError || hasError()) {\n          <svg-icon name=\"far-times-circle\" class=\"has-text-danger is-flex is-align-items-center\" />\n          <span class=\"has-text-danger has-text-weight-bold is-italic\">Error</span>\n        } @else if (nodeProperty().hasWarning || hasWarning()) {\n          <svg-icon name=\"exclamation-triangle\" class=\"has-text-warning is-flex is-align-items-center\" />\n          <span class=\"has-text-warning has-text-weight-bold is-italic\">Warning</span>\n        }\n      }\n    </div>\n    <div class=\"is-flex is-flex-direction-row is-align-items-center is-gap-16\">\n      @if (nodeUrl()) {\n        <a class=\"external-link\" [href]=\"nodeUrl()\" target=\"_blank\">\n          <span>View</span>\n          <span class=\"is-pl-1\">{{ schemaType() }}</span>\n          <fa-icon class=\"is-pl-2\" [icon]=\"faExternalLinkAlt\" />\n        </a>\n      }\n      <fa-icon [icon]=\"isOpen() ? faMinus : faPlus\" />\n    </div>\n  </div>\n\n  <div class=\"is-flex is-flex-direction-column is-align-items-flex-start is-align-self-stretch\">\n    @if (isOpen()) {\n      @if (nodeProperty()?.error) {\n        <ng-container\n          *ngTemplateOutlet=\"propertyError; context: { property: nodeProperty(), classes: 'is-my-1 is-px-3' }\" />\n      }\n    }\n\n    <div class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-self-stretch\">\n      @for (property of properties(); track trackByProperty($index, property)) {\n        <ng-container *ngTemplateOutlet=\"showProperty; context: { $implicit: property }\" />\n      }\n    </div>\n\n    @if (isOpen()) {\n      <ng-container *ngTemplateOutlet=\"propertyMap; context: { $implicit: nodeProperty() }\" />\n    }\n  </div>\n\n  <ng-content />\n</div>\n\n<ng-template #propertyKeyTooltip let-property=\"property\">\n  <markdown [data]=\"property.schema.description\" />\n</ng-template>\n\n<ng-template #propertyKey let-property=\"property\">\n  @if (property.schema?.description) {\n    <span\n      class=\"property-key trigger-popover\"\n      [ngbTooltip]=\"propertyKeyTooltip\"\n      [tooltipContext]=\"{ property }\"\n      tooltipClass=\"property-tooltip\"\n      placement=\"bottom-left\"\n      container=\"body\">\n      <span>{{ property.key }}</span>\n    </span>\n  } @else {\n    <span>{{ property.key }}</span>\n  }\n</ng-template>\n\n<ng-template #showProperty let-property>\n  @if ((isOpen() || property.closedVisible) && !property.isHidden) {\n    @if (property.properties.length) {\n      <div class=\"is-flex is-flex-direction-column is-flex-wrap-wrap w-100 | properties-container\">\n        <div\n          class=\"is-flex is-flex-direction-row is-justify-content-space-between is-gap-4 has-text-secondary w-100 | properties-container--title\"\n          (click)=\"property.isOpen = !property.isOpen\"\n          pointer\n          [class.is-open]=\"property.isOpen\"\n          [class.is-group-error-danger]=\"property.hasError\"\n          [class.is-group-error-warning]=\"property.hasWarning\">\n          <div class=\"is-flex is-gap-8 has-text-weight-bold\">\n            <!-- @if (property.schemaType === SchemaType.Term) {\n              <he-node-icon [type]=\"property.schemaType\" />\n            } -->\n            <ng-container *ngTemplateOutlet=\"propertyKey; context: { property }\" />\n          </div>\n          <fa-icon [icon]=\"property.isOpen ? faMinus : faPlus\" />\n        </div>\n        @if (property.isOpen) {\n          @if (property.error) {\n            <ng-container *ngTemplateOutlet=\"propertyError; context: { property, classes: 'is-my-1 is-px-3' }\" />\n          }\n          @if (property.isArray) {\n            <div class=\"py-2 px-3 w-100\">\n              @for (prop2 of property.properties; track trackByProperty($index, prop2)) {\n                @if (prop2.key) {\n                  <div class=\"pt-2 my-4 | property-array-container\" [id]=\"prop2.fullKey + '_' + prop2.id\">\n                    <div class=\"property-array-number ml-2\">\n                      <div class=\"is-flex is-justify-content-center is-align-items-center has-addons number-tags\">\n                        <span class=\"number-tag\">{{ prop2.key }}</span>\n                      </div>\n                    </div>\n                    @if (prop2.error) {\n                      <ng-container\n                        *ngTemplateOutlet=\"propertyError; context: { property: prop2, classes: 'is-my-1 is-px-3' }\" />\n                    }\n                    <div class=\"p-3\">\n                      <div class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-self-stretch\">\n                        @for (prop3 of prop2.properties; track trackByProperty($index, prop3)) {\n                          <ng-container *ngTemplateOutlet=\"showProperty; context: { $implicit: prop3 }\" />\n                        }\n                      </div>\n                    </div>\n                  </div>\n                }\n              }\n            </div>\n          } @else {\n            <div class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-self-stretch\">\n              @for (prop2 of property.properties; track trackByProperty($index, prop2)) {\n                <ng-container *ngTemplateOutlet=\"showProperty; context: { $implicit: prop2 }\" />\n              }\n            </div>\n          }\n        }\n      </div>\n    } @else {\n      <div\n        class=\"is-flex is-flex-direction-column is-gap-4 | property-container\"\n        [class.is-error-danger]=\"property.hasError\"\n        [class.is-error-warning]=\"property.hasWarning\">\n        <div\n          class=\"is-flex is-flex-direction-row is-align-items-center is-justify-content-space-between is-gap-4 | field-container\">\n          @if (property.key) {\n            <div class=\"is-flex is-align-items-center is-flex-wrap-wrap is-gap-4\">\n              <div class=\"is-flex has-text-secondary\">\n                <ng-container *ngTemplateOutlet=\"propertyKey; context: { property }\" />\n                <span>:</span>\n              </div>\n\n              <span class=\"is-flex is-size-7 | property-value\">\n                {{ property.value }}\n              </span>\n\n              <div class=\"copy-button\">\n                <he-clipboard [value]=\"property.value\" [hideText]=\"true\" [icon]=\"farClone\"></he-clipboard>\n              </div>\n            </div>\n\n            <div class=\"is-flex is-gap-8\">\n              @if (property.externalUrl?.url) {\n                <a\n                  [href]=\"property.externalUrl.url + (property.externalUrl.urlParamValue ? property.value : '')\"\n                  target=\"_blank\"\n                  [title]=\"property.externalUrl.title\"\n                  [ngClass]=\"{ 'is-info': property.key === 'type' }\"\n                  [attr.disabled]=\"property.externalUrl.urlParamValue && !property.value ? true : null\">\n                  <fa-icon [icon]=\"property.externalUrl.icon || faExternalLinkAlt\" />\n                </a>\n              }\n            </div>\n          }\n        </div>\n        @if (property.hasError || property.hasWarning) {\n          <ng-container *ngTemplateOutlet=\"propertyError; context: { property }\" />\n        }\n      </div>\n    }\n  }\n</ng-template>\n\n<ng-template #propertyError let-property=\"property\" let-classes=\"classes\">\n  @if (property.error) {\n    <div\n      class=\"is-flex is-flex-direction-row is-flex-wrap-wrap is-align-items-center is-gap-8 is-size-6 is-m-0 w-100 has-text-grey {{\n        classes\n      }} | property-error\"\n      [class.is-error-danger]=\"property.hasError\"\n      [class.is-error-warning]=\"property.hasWarning\">\n      @if (property.hasError) {\n        <fa-icon class=\"has-text-danger is-py-1\" [icon]=\"farCircleXmark\" size=\"xl\" />\n      } @else {\n        <svg-icon class=\"has-text-warning is-pt-1\" name=\"exclamation-triangle\" />\n      }\n      <div [innerHTML]=\"property.error.message\"></div>\n      @if (property.error.index >= 0) {\n        @if (property.hasWarning) {\n          <a (click)=\"resolveError(property)\">\n            <fa-icon class=\"pr-2\" [icon]=\"faCheck\" />\n            <span>Resolved</span>\n          </a>\n        }\n      }\n    </div>\n  }\n</ng-template>\n\n<ng-template #propertyMap let-property>\n  @if (showMap()) {\n    <div class=\"is-flex is-flex-direction-column is-flex-wrap-wrap w-100 | properties-container\">\n      <div\n        class=\"is-flex is-flex-direction-row is-justify-content-space-between is-gap-4 w-100 | properties-container--title\"\n        (click)=\"mapVisible.set(!mapVisible())\"\n        pointer\n        [class.is-open]=\"mapVisible()\">\n        <span class=\"is-size-6\">View on Map</span>\n        <fa-icon [icon]=\"mapVisible() ? faMinus : faPlus\" />\n      </div>\n      @if (mapVisible()) {\n        <he-sites-maps [sites]=\"[node()]\" [showNotice]=\"false\" />\n      }\n    </div>\n  }\n</ng-template>\n"]}
|