@hestia-earth/ui-components 0.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +24 -0
- package/auth/auth.service.d.ts +10 -0
- package/auth/public-api.d.ts +1 -0
- package/bibliographies/bibliographies-search-confirm/bibliographies-search-confirm.component.d.ts +40 -0
- package/bibliographies/bibliographies.module.d.ts +10 -0
- package/bibliographies/public-api.d.ts +2 -0
- package/bundles/hestia-earth-ui-components.umd.js +7570 -0
- package/bundles/hestia-earth-ui-components.umd.js.map +1 -0
- package/common/bind-once.directive.d.ts +10 -0
- package/common/blank-node-diffs/blank-node-diffs.component.d.ts +18 -0
- package/common/blank-node-state/blank-node-state.component.d.ts +36 -0
- package/common/blank-node-state-notice/blank-node-state-notice.component.d.ts +9 -0
- package/common/blank-node-value-delta/blank-node-value-delta.component.d.ts +13 -0
- package/common/click-outside.directive.d.ts +10 -0
- package/common/clipboard/clipboard.component.d.ts +16 -0
- package/common/common.module.d.ts +37 -0
- package/common/common.service.d.ts +11 -0
- package/common/default.pipe.d.ts +7 -0
- package/common/ellipsis.pipe.d.ts +7 -0
- package/common/get.pipe.d.ts +7 -0
- package/common/key-to-label.pipe.d.ts +7 -0
- package/common/keys.pipe.d.ts +7 -0
- package/common/link-key-value/link-key-value.component.d.ts +16 -0
- package/common/logs-utils.d.ts +17 -0
- package/common/maps-drawing-confirm/maps-drawing-confirm.component.d.ts +28 -0
- package/common/maps-utils.d.ts +70 -0
- package/common/pluralize.pipe.d.ts +7 -0
- package/common/popover/popover.component.d.ts +16 -0
- package/common/popover-confirm/popover-confirm.component.d.ts +15 -0
- package/common/precision.pipe.d.ts +7 -0
- package/common/public-api.d.ts +30 -0
- package/common/schema-version-link/schema-version-link.component.d.ts +10 -0
- package/common/skeleton-text/skeleton-text.component.d.ts +11 -0
- package/common/social-tags/social-tags.component.d.ts +12 -0
- package/common/tags-input.directive.d.ts +20 -0
- package/common/times.pipe.d.ts +7 -0
- package/common/toast/toast.component.d.ts +19 -0
- package/common/toast.service.d.ts +16 -0
- package/common/unit-converter/unit-converter.component.d.ts +25 -0
- package/common/utils.d.ts +88 -0
- package/cycles/cycles-activity/cycles-activity.component.d.ts +31 -0
- package/cycles/cycles-activity-logs/cycles-activity-logs.component.d.ts +25 -0
- package/cycles/cycles-completeness/cycles-completeness.component.d.ts +14 -0
- package/cycles/cycles-emissions/cycles-emissions.component.d.ts +40 -0
- package/cycles/cycles-emissions-chart/cycles-emissions-chart.component.d.ts +18 -0
- package/cycles/cycles-emissions-logs/cycles-emissions-logs.component.d.ts +23 -0
- package/cycles/cycles-functional-unit-measure/cycles-functional-unit-measure.component.d.ts +8 -0
- package/cycles/cycles-result/cycles-result.component.d.ts +13 -0
- package/cycles/cycles-suggest-form/cycles-suggest-form.component.d.ts +24 -0
- package/cycles/cycles.model.d.ts +7 -0
- package/cycles/cycles.module.d.ts +19 -0
- package/cycles/public-api.d.ts +10 -0
- package/engine/aggregation-engine.service.d.ts +71 -0
- package/engine/engine.service.d.ts +61 -0
- package/engine/public-api.d.ts +2 -0
- package/esm2015/auth/auth.service.js +19 -0
- package/esm2015/auth/public-api.js +2 -0
- package/esm2015/bibliographies/bibliographies-search-confirm/bibliographies-search-confirm.component.js +100 -0
- package/esm2015/bibliographies/bibliographies.module.js +30 -0
- package/esm2015/bibliographies/public-api.js +3 -0
- package/esm2015/common/bind-once.directive.js +23 -0
- package/esm2015/common/blank-node-diffs/blank-node-diffs.component.js +47 -0
- package/esm2015/common/blank-node-state/blank-node-state.component.js +90 -0
- package/esm2015/common/blank-node-state-notice/blank-node-state-notice.component.js +27 -0
- package/esm2015/common/blank-node-value-delta/blank-node-value-delta.component.js +41 -0
- package/esm2015/common/click-outside.directive.js +28 -0
- package/esm2015/common/clipboard/clipboard.component.js +69 -0
- package/esm2015/common/common.module.js +144 -0
- package/esm2015/common/common.service.js +29 -0
- package/esm2015/common/default.pipe.js +23 -0
- package/esm2015/common/ellipsis.pipe.js +17 -0
- package/esm2015/common/get.pipe.js +17 -0
- package/esm2015/common/key-to-label.pipe.js +18 -0
- package/esm2015/common/keys.pipe.js +20 -0
- package/esm2015/common/link-key-value/link-key-value.component.js +55 -0
- package/esm2015/common/logs-utils.js +49 -0
- package/esm2015/common/maps-drawing-confirm/maps-drawing-confirm.component.js +89 -0
- package/esm2015/common/maps-utils.js +101 -0
- package/esm2015/common/pluralize.pipe.js +17 -0
- package/esm2015/common/popover/popover.component.js +37 -0
- package/esm2015/common/popover-confirm/popover-confirm.component.js +38 -0
- package/esm2015/common/precision.pipe.js +20 -0
- package/esm2015/common/public-api.js +31 -0
- package/esm2015/common/schema-version-link/schema-version-link.component.js +39 -0
- package/esm2015/common/skeleton-text/skeleton-text.component.js +44 -0
- package/esm2015/common/social-tags/social-tags.component.js +37 -0
- package/esm2015/common/tags-input.directive.js +74 -0
- package/esm2015/common/times.pipe.js +16 -0
- package/esm2015/common/toast/toast.component.js +37 -0
- package/esm2015/common/toast.service.js +36 -0
- package/esm2015/common/unit-converter/unit-converter.component.js +73 -0
- package/esm2015/common/utils.js +245 -0
- package/esm2015/cycles/cycles-activity/cycles-activity.component.js +92 -0
- package/esm2015/cycles/cycles-activity-logs/cycles-activity-logs.component.js +48 -0
- package/esm2015/cycles/cycles-completeness/cycles-completeness.component.js +41 -0
- package/esm2015/cycles/cycles-emissions/cycles-emissions.component.js +121 -0
- package/esm2015/cycles/cycles-emissions-chart/cycles-emissions-chart.component.js +93 -0
- package/esm2015/cycles/cycles-emissions-logs/cycles-emissions-logs.component.js +63 -0
- package/esm2015/cycles/cycles-functional-unit-measure/cycles-functional-unit-measure.component.js +22 -0
- package/esm2015/cycles/cycles-result/cycles-result.component.js +92 -0
- package/esm2015/cycles/cycles-suggest-form/cycles-suggest-form.component.js +83 -0
- package/esm2015/cycles/cycles.model.js +2 -0
- package/esm2015/cycles/cycles.module.js +66 -0
- package/esm2015/cycles/public-api.js +11 -0
- package/esm2015/engine/aggregation-engine.service.js +126 -0
- package/esm2015/engine/engine.service.js +78 -0
- package/esm2015/engine/public-api.js +3 -0
- package/esm2015/files/files-error.model.js +180 -0
- package/esm2015/files/files-form/files-form.component.js +439 -0
- package/esm2015/files/files-form.model.js +548 -0
- package/esm2015/files/files.module.js +42 -0
- package/esm2015/files/public-api.js +3 -0
- package/esm2015/fontawesome/fontawesome.module.js +36 -0
- package/esm2015/fontawesome/public-api.js +2 -0
- package/esm2015/hestia-earth-ui-components.js +5 -0
- package/esm2015/impact-assessments/impact-assessments-indicator-breakdown-chart/impact-assessments-indicator-breakdown-chart.component.js +161 -0
- package/esm2015/impact-assessments/impact-assessments-indicators-chart/impact-assessments-indicators-chart.component.js +142 -0
- package/esm2015/impact-assessments/impact-assessments-products/impact-assessments-products.component.js +252 -0
- package/esm2015/impact-assessments/impact-assessments-products-logs/impact-assessments-products-logs.component.js +68 -0
- package/esm2015/impact-assessments/impact-assessments.model.js +2 -0
- package/esm2015/impact-assessments/impact-assessments.module.js +47 -0
- package/esm2015/impact-assessments/public-api.js +7 -0
- package/esm2015/mendeley/mendeley.service.js +35 -0
- package/esm2015/mendeley/public-api.js +2 -0
- package/esm2015/node/node-csv-export-confirm/node-csv-export-confirm.component.js +65 -0
- package/esm2015/node/node-csv-select-headers/node-csv-select-headers.component.js +119 -0
- package/esm2015/node/node-csv.service.js +24 -0
- package/esm2015/node/node-diffs/node-diffs.component.js +67 -0
- package/esm2015/node/node-diffs/node-diffs.model.js +6 -0
- package/esm2015/node/node-icon/node-icon.component.js +37 -0
- package/esm2015/node/node-link/node-link.component.js +33 -0
- package/esm2015/node/node-logs-file/node-logs-file.component.js +63 -0
- package/esm2015/node/node-logs-models/node-logs-models.component.js +188 -0
- package/esm2015/node/node-missing-lookup-factors/node-missing-lookup-factors.component.js +34 -0
- package/esm2015/node/node-value-details/node-value-details.component.js +46 -0
- package/esm2015/node/node.module.js +67 -0
- package/esm2015/node/node.service.js +113 -0
- package/esm2015/node/public-api.js +13 -0
- package/esm2015/public-api.js +16 -0
- package/esm2015/schema/public-api.js +2 -0
- package/esm2015/schema/schema.service.js +167 -0
- package/esm2015/search/public-api.js +4 -0
- package/esm2015/search/search.model.js +204 -0
- package/esm2015/search/search.module.js +13 -0
- package/esm2015/search/search.service.js +99 -0
- package/esm2015/sites/public-api.js +6 -0
- package/esm2015/sites/sites-maps/sites-maps.component.js +162 -0
- package/esm2015/sites/sites-measurements/sites-measurements.component.js +78 -0
- package/esm2015/sites/sites-measurements-logs/sites-measurements-logs.component.js +63 -0
- package/esm2015/sites/sites.model.js +10 -0
- package/esm2015/sites/sites.module.js +42 -0
- package/esm2015/terms/public-api.js +2 -0
- package/esm2015/terms/terms.model.js +184 -0
- package/esm2015/users/public-api.js +2 -0
- package/esm2015/users/users.service.js +17 -0
- package/fesm2015/hestia-earth-ui-components.js +5920 -0
- package/fesm2015/hestia-earth-ui-components.js.map +1 -0
- package/files/files-error.model.d.ts +40 -0
- package/files/files-form/files-form.component.d.ts +94 -0
- package/files/files-form.model.d.ts +113 -0
- package/files/files.module.d.ts +13 -0
- package/files/public-api.d.ts +2 -0
- package/fontawesome/fontawesome.module.d.ts +10 -0
- package/fontawesome/public-api.d.ts +1 -0
- package/hestia-earth-ui-components.d.ts +5 -0
- package/impact-assessments/impact-assessments-indicator-breakdown-chart/impact-assessments-indicator-breakdown-chart.component.d.ts +28 -0
- package/impact-assessments/impact-assessments-indicators-chart/impact-assessments-indicators-chart.component.d.ts +25 -0
- package/impact-assessments/impact-assessments-products/impact-assessments-products.component.d.ts +77 -0
- package/impact-assessments/impact-assessments-products-logs/impact-assessments-products-logs.component.d.ts +25 -0
- package/impact-assessments/impact-assessments.model.d.ts +6 -0
- package/impact-assessments/impact-assessments.module.d.ts +15 -0
- package/impact-assessments/public-api.d.ts +6 -0
- package/mendeley/mendeley.service.d.ts +25 -0
- package/mendeley/public-api.d.ts +1 -0
- package/node/node-csv-export-confirm/node-csv-export-confirm.component.d.ts +32 -0
- package/node/node-csv-select-headers/node-csv-select-headers.component.d.ts +42 -0
- package/node/node-csv.service.d.ts +9 -0
- package/node/node-diffs/node-diffs.component.d.ts +23 -0
- package/node/node-diffs/node-diffs.model.d.ts +4 -0
- package/node/node-icon/node-icon.component.d.ts +8 -0
- package/node/node-link/node-link.component.d.ts +11 -0
- package/node/node-logs-file/node-logs-file.component.d.ts +31 -0
- package/node/node-logs-models/node-logs-models.component.d.ts +72 -0
- package/node/node-missing-lookup-factors/node-missing-lookup-factors.component.d.ts +14 -0
- package/node/node-value-details/node-value-details.component.d.ts +17 -0
- package/node/node.module.d.ts +20 -0
- package/node/node.service.d.ts +70 -0
- package/node/public-api.d.ts +12 -0
- package/package.json +53 -0
- package/public-api.d.ts +15 -0
- package/schema/public-api.d.ts +1 -0
- package/schema/schema.service.d.ts +51 -0
- package/search/public-api.d.ts +3 -0
- package/search/search.model.d.ts +243 -0
- package/search/search.module.d.ts +7 -0
- package/search/search.service.d.ts +129 -0
- package/sites/public-api.d.ts +5 -0
- package/sites/sites-maps/sites-maps.component.d.ts +35 -0
- package/sites/sites-measurements/sites-measurements.component.d.ts +31 -0
- package/sites/sites-measurements-logs/sites-measurements-logs.component.d.ts +23 -0
- package/sites/sites.model.d.ts +10 -0
- package/sites/sites.module.d.ts +13 -0
- package/styles.scss +197 -0
- package/terms/public-api.d.ts +1 -0
- package/terms/terms.model.d.ts +21 -0
- package/users/public-api.d.ts +1 -0
- package/users/users.service.d.ts +7 -0
|
@@ -0,0 +1,439 @@
|
|
|
1
|
+
import { __awaiter, __rest } from "tslib";
|
|
2
|
+
/* eslint-disable complexity */
|
|
3
|
+
import { Component, Input, Output, EventEmitter } from '@angular/core';
|
|
4
|
+
import { of } from 'rxjs';
|
|
5
|
+
import { debounceTime, distinctUntilChanged, map, mergeMap, switchMap, tap } from 'rxjs/operators';
|
|
6
|
+
import * as pluralize from 'pluralize';
|
|
7
|
+
import { moveItemInArray } from '@angular/cdk/drag-drop';
|
|
8
|
+
import { SchemaType, NodeType } from '@hestia-earth/schema';
|
|
9
|
+
import { isEmpty } from '@hestia-earth/utils';
|
|
10
|
+
import { formatPropertyError, propertyError, hasError, hasWarning, formatSuggestion, formatLinkNodesSuggestions, defaultSuggestionType, recursiveProperties, typeToNewProperty, propertyId, isAddPropertyEnabled, nodeAvailableProperties, sortProperties, groupChanged, parentProperty, siblingProperty, updateProperties, refreshPropertyKeys, keyToDataPath, findProperty, calculateCycleDurationEnabled, calculateCycleDuration, calculateCycleStartDateEnabled, calculateCycleStartDate, parseNewValue } from '../files-form.model';
|
|
11
|
+
import { errorHasError, errorHasWarning, filterError, formatError } from '../files-error.model';
|
|
12
|
+
import { matchType, matchPhraseQuery, matchPhrasePrefixQuery, matchBoolPrefixQuery } from '../../search/search.model';
|
|
13
|
+
import { safeJSONParse } from '../../common/utils';
|
|
14
|
+
import * as i0 from "@angular/core";
|
|
15
|
+
import * as i1 from "../../search/search.service";
|
|
16
|
+
import * as i2 from "../../users/users.service";
|
|
17
|
+
import * as i3 from "@fortawesome/angular-fontawesome";
|
|
18
|
+
import * as i4 from "../../node/node-icon/node-icon.component";
|
|
19
|
+
import * as i5 from "../../common/maps-drawing-confirm/maps-drawing-confirm.component";
|
|
20
|
+
import * as i6 from "../../bibliographies/bibliographies-search-confirm/bibliographies-search-confirm.component";
|
|
21
|
+
import * as i7 from "../../common/schema-version-link/schema-version-link.component";
|
|
22
|
+
import * as i8 from "../../common/popover-confirm/popover-confirm.component";
|
|
23
|
+
import * as i9 from "../../sites/sites-maps/sites-maps.component";
|
|
24
|
+
import * as i10 from "@ng-bootstrap/ng-bootstrap";
|
|
25
|
+
import * as i11 from "../../common/unit-converter/unit-converter.component";
|
|
26
|
+
import * as i12 from "@angular/common";
|
|
27
|
+
import * as i13 from "../../common/bind-once.directive";
|
|
28
|
+
import * as i14 from "@angular/forms";
|
|
29
|
+
import * as i15 from "../../common/tags-input.directive";
|
|
30
|
+
import * as i16 from "../../common/pluralize.pipe";
|
|
31
|
+
const get = require('lodash.get');
|
|
32
|
+
const MIN_TYPEAHEAD_LENGTH = 2;
|
|
33
|
+
const populateTermFields = [
|
|
34
|
+
'termType', 'units',
|
|
35
|
+
'defaultProperties.term.name', 'defaultProperties.term.units', 'defaultProperties.value'
|
|
36
|
+
];
|
|
37
|
+
export class FilesFormComponent {
|
|
38
|
+
constructor(ref, searchService, usersService) {
|
|
39
|
+
this.ref = ref;
|
|
40
|
+
this.searchService = searchService;
|
|
41
|
+
this.usersService = usersService;
|
|
42
|
+
this.errors = [];
|
|
43
|
+
this.nodeMap = {};
|
|
44
|
+
this.editable = false;
|
|
45
|
+
this.errorMode = false;
|
|
46
|
+
this.deepEditable = true;
|
|
47
|
+
this.errorsEditable = false;
|
|
48
|
+
this.nodeChange = new EventEmitter();
|
|
49
|
+
this.nodeErorrResolved = new EventEmitter();
|
|
50
|
+
this.nodeErrorAdded = new EventEmitter();
|
|
51
|
+
this.NodeType = NodeType;
|
|
52
|
+
this.SchemaType = SchemaType;
|
|
53
|
+
this.isOpen = true;
|
|
54
|
+
this.mapVisible = false;
|
|
55
|
+
this.pluralize = pluralize;
|
|
56
|
+
this.formatPropertyError = formatPropertyError;
|
|
57
|
+
this.properties = [];
|
|
58
|
+
this.formatter = (value) => { var _a; return typeof value === 'object' ? ('bibliography' in value ? (_a = value.bibliography) === null || _a === void 0 ? void 0 : _a.title : value.name) : value; };
|
|
59
|
+
this.suggestNewProperty = (fullKey) => (text$) => text$.pipe(distinctUntilChanged(), switchMap(term => of(this.newProperties(findProperty(this.properties, fullKey) || this.nodeProperty, term))));
|
|
60
|
+
this.suggestExistingNode = (type, term, uniqueKey = '@id') => term.length < MIN_TYPEAHEAD_LENGTH ?
|
|
61
|
+
of([]) :
|
|
62
|
+
this.searchService.suggest(term, type, [], {
|
|
63
|
+
bool: {
|
|
64
|
+
must: [matchType(type)],
|
|
65
|
+
should: [
|
|
66
|
+
matchPhraseQuery(term, 100, uniqueKey),
|
|
67
|
+
matchPhrasePrefixQuery(term, 20, uniqueKey),
|
|
68
|
+
matchBoolPrefixQuery(term, 10, uniqueKey)
|
|
69
|
+
],
|
|
70
|
+
minimum_should_match: 1
|
|
71
|
+
}
|
|
72
|
+
}, 5, populateTermFields);
|
|
73
|
+
this.suggestNode = (type, suggestDefault) => (term, property) => {
|
|
74
|
+
var _a, _b;
|
|
75
|
+
return (((_a = property.suggestions) === null || _a === void 0 ? void 0 : _a.isUniqueKey) ? (this.suggestExistingNode(type, term, property.suggestions.isUniqueKey)) : (((_b = property.suggestions) === null || _b === void 0 ? void 0 : _b.isLinkNode) ?
|
|
76
|
+
of(formatLinkNodesSuggestions(this.nodeMap, type)) : (suggestDefault ? suggestDefault(term, property) : of([])))).pipe(map((res) => res.map(formatSuggestion)));
|
|
77
|
+
};
|
|
78
|
+
this.suggester = {
|
|
79
|
+
default: (term, property) => {
|
|
80
|
+
var _a;
|
|
81
|
+
return of((((_a = property === null || property === void 0 ? void 0 : property.suggestions) === null || _a === void 0 ? void 0 : _a.values) || [])
|
|
82
|
+
.map(v => `${v}`)
|
|
83
|
+
.filter(v => v.toLowerCase().includes(term.toLowerCase()))
|
|
84
|
+
.map(name => ({ name })));
|
|
85
|
+
},
|
|
86
|
+
[NodeType.Actor]: this.suggestNode(NodeType.Actor),
|
|
87
|
+
[NodeType.Cycle]: this.suggestNode(NodeType.Cycle),
|
|
88
|
+
[NodeType.ImpactAssessment]: this.suggestNode(NodeType.ImpactAssessment),
|
|
89
|
+
[NodeType.Organisation]: this.suggestNode(NodeType.Organisation),
|
|
90
|
+
[NodeType.Site]: this.suggestNode(NodeType.Site),
|
|
91
|
+
[NodeType.Source]: this.suggestNode(NodeType.Source, (term, { key }) => term.length < MIN_TYPEAHEAD_LENGTH ?
|
|
92
|
+
of([]) :
|
|
93
|
+
this.searchService.suggestSource(term, 5, [`bibliography.${key}`], [`bibliography.${key}`])),
|
|
94
|
+
[NodeType.Term]: this.suggestNode(NodeType.Term, (term, { suggestions }) => term.length < MIN_TYPEAHEAD_LENGTH ?
|
|
95
|
+
of([]) :
|
|
96
|
+
this.searchService.suggest(term, NodeType.Term, (suggestions === null || suggestions === void 0 ? void 0 : suggestions.queries) || [], null, 5, populateTermFields))
|
|
97
|
+
};
|
|
98
|
+
this.propertySuggest = (fullKey, suggestType) => (text$) => text$.pipe(debounceTime(suggestType === defaultSuggestionType ? 0 : 300), distinctUntilChanged(), mergeMap(term => {
|
|
99
|
+
const property = findProperty(this.properties, fullKey);
|
|
100
|
+
return (property === null || property === void 0 ? void 0 : property.editable) && suggestType in this.suggester ?
|
|
101
|
+
of(term).pipe(tap(() => property.loading = true), switchMap(() => this.suggester[suggestType](term, property)), tap(() => property.loading = false)) :
|
|
102
|
+
of([]);
|
|
103
|
+
}));
|
|
104
|
+
}
|
|
105
|
+
ngOnInit() {
|
|
106
|
+
const schemaType = this.node.type || this.node['@type'];
|
|
107
|
+
const schema = this.schemas ? this.schemas[schemaType] : {};
|
|
108
|
+
const errors = this.errors.filter(filterError);
|
|
109
|
+
const error = propertyError(errors)('');
|
|
110
|
+
this.nodeProperty = this.schemas ? {
|
|
111
|
+
id: propertyId(),
|
|
112
|
+
schema,
|
|
113
|
+
schemaType,
|
|
114
|
+
editable: !!this.node.type,
|
|
115
|
+
fullKey: '',
|
|
116
|
+
key: '',
|
|
117
|
+
value: {},
|
|
118
|
+
error: formatError(error),
|
|
119
|
+
hasError: errorHasError(error),
|
|
120
|
+
hasWarning: errorHasWarning(error),
|
|
121
|
+
suggestions: {},
|
|
122
|
+
newProperty: {},
|
|
123
|
+
newError: { level: 'warning' },
|
|
124
|
+
properties: [],
|
|
125
|
+
addPropertyEnabled: isAddPropertyEnabled(this.node, schemaType, schema)
|
|
126
|
+
} : undefined;
|
|
127
|
+
const recurser = recursiveProperties(this.schemas, errors, schema, this.deepEditable);
|
|
128
|
+
this.properties = recurser(this.node);
|
|
129
|
+
}
|
|
130
|
+
trackByIndex(index, _value) {
|
|
131
|
+
return index;
|
|
132
|
+
}
|
|
133
|
+
trackByProperty(_index, { id, fullKey }) {
|
|
134
|
+
return [fullKey, id].join('_');
|
|
135
|
+
}
|
|
136
|
+
hasAddons(property) {
|
|
137
|
+
var _a;
|
|
138
|
+
return property && (this.enableAddError(property) ||
|
|
139
|
+
(this.editable && property.editable) ||
|
|
140
|
+
this.addPropertyEnabled(property) ||
|
|
141
|
+
this.isRequired(property) ||
|
|
142
|
+
((_a = property.externalUrl) === null || _a === void 0 ? void 0 : _a.url) ||
|
|
143
|
+
this.mapDrawingModes(property).length ||
|
|
144
|
+
this.unitConverterEnabled(property));
|
|
145
|
+
}
|
|
146
|
+
isRequired({ value, isRequired }) {
|
|
147
|
+
return isRequired && isEmpty(value);
|
|
148
|
+
}
|
|
149
|
+
// --- Errors
|
|
150
|
+
enableAddError(property) {
|
|
151
|
+
return this.errorsEditable && !property.hasError && !property.hasWarning;
|
|
152
|
+
}
|
|
153
|
+
addError(property, { message, level }) {
|
|
154
|
+
if (!message) {
|
|
155
|
+
return;
|
|
156
|
+
}
|
|
157
|
+
const keys = property.fullKey.split('.');
|
|
158
|
+
keys.pop();
|
|
159
|
+
const error = property.isAdded ? {
|
|
160
|
+
message,
|
|
161
|
+
level: level || 'error',
|
|
162
|
+
dataPath: keyToDataPath(keys.join('.')),
|
|
163
|
+
params: {
|
|
164
|
+
missingProperty: property.key
|
|
165
|
+
}
|
|
166
|
+
} : {
|
|
167
|
+
message,
|
|
168
|
+
level: level || 'error',
|
|
169
|
+
dataPath: keyToDataPath(property.fullKey)
|
|
170
|
+
};
|
|
171
|
+
property.newError = { level: 'warning' };
|
|
172
|
+
property.error = error;
|
|
173
|
+
property.hasError = errorHasError(error);
|
|
174
|
+
property.hasWarning = errorHasWarning(error);
|
|
175
|
+
this.nodeErrorAdded.emit(error);
|
|
176
|
+
}
|
|
177
|
+
editError(property) {
|
|
178
|
+
property.newError = property.error;
|
|
179
|
+
}
|
|
180
|
+
resolveError(property) {
|
|
181
|
+
var _a;
|
|
182
|
+
const index = (_a = property.error) === null || _a === void 0 ? void 0 : _a.index;
|
|
183
|
+
property.error = undefined;
|
|
184
|
+
property.hasError = hasError(property.properties);
|
|
185
|
+
property.hasWarning = hasWarning(property.properties);
|
|
186
|
+
this.nodeErorrResolved.emit(index);
|
|
187
|
+
return groupChanged(this.properties, property.fullKey);
|
|
188
|
+
}
|
|
189
|
+
// ---
|
|
190
|
+
typeaheadFocus(e) {
|
|
191
|
+
if (e.bubbles) {
|
|
192
|
+
return;
|
|
193
|
+
}
|
|
194
|
+
e.stopPropagation();
|
|
195
|
+
setTimeout(() => {
|
|
196
|
+
var _a;
|
|
197
|
+
const inputEvent = new Event('input');
|
|
198
|
+
(_a = e.target) === null || _a === void 0 ? void 0 : _a.dispatchEvent(inputEvent);
|
|
199
|
+
}, 0);
|
|
200
|
+
}
|
|
201
|
+
addPropertyEnabled(property) {
|
|
202
|
+
return this.editable && !this.errorsEditable && property.addPropertyEnabled;
|
|
203
|
+
}
|
|
204
|
+
newProperties(property, field) {
|
|
205
|
+
const properties = nodeAvailableProperties(this.node, property);
|
|
206
|
+
return Object.keys(properties)
|
|
207
|
+
.filter(prop => prop.toLowerCase().includes(field.toLowerCase()))
|
|
208
|
+
.map(name => ({ name, prop: properties[name] }));
|
|
209
|
+
}
|
|
210
|
+
replacePropertyData(property, data, editable, replaceParent = true) {
|
|
211
|
+
const parent = replaceParent ? parentProperty(this.properties, property) : property;
|
|
212
|
+
const properties = recursiveProperties(this.schemas, [], this.schemas[this.node.type], this.deepEditable, parent.fullKey)(data);
|
|
213
|
+
updateProperties({ properties }, prop => {
|
|
214
|
+
prop.editable = typeof editable === 'undefined' ? prop.editable : editable;
|
|
215
|
+
return prop;
|
|
216
|
+
});
|
|
217
|
+
parent.editable = typeof editable === 'undefined' ? parent.editable : editable;
|
|
218
|
+
parent.properties = properties;
|
|
219
|
+
parent.value = data;
|
|
220
|
+
return this.propertyChanged(data, parent);
|
|
221
|
+
}
|
|
222
|
+
suggestionSelected(_a, property) {
|
|
223
|
+
var _b;
|
|
224
|
+
var { '@id': _id, name, type, id } = _a, data = __rest(_a, ['@id', "name", "type", "id"]);
|
|
225
|
+
return _id ?
|
|
226
|
+
// suggestions from existing nodes
|
|
227
|
+
this.replacePropertyData(property, Object.assign({ '@id': _id, '@type': (_b = property.suggestions) === null || _b === void 0 ? void 0 : _b.type, name }, data), false) :
|
|
228
|
+
(
|
|
229
|
+
// suggestion from linked nodes
|
|
230
|
+
type && id ?
|
|
231
|
+
this.replacePropertyData(property, Object.assign({ type, id }, data), true) :
|
|
232
|
+
// suggestion by name => enum values
|
|
233
|
+
this.propertyChanged(name, property));
|
|
234
|
+
}
|
|
235
|
+
addArrayGroup(array) {
|
|
236
|
+
const prop = this.schemas[array.schemaType];
|
|
237
|
+
const value = get(this.node, array.fullKey);
|
|
238
|
+
const newProperty = typeToNewProperty(this.schemas, prop, prop, value, `${array.fullKey}[${array.properties.length}]`);
|
|
239
|
+
array.properties.push(newProperty);
|
|
240
|
+
this.nodeChange.emit({ key: array.fullKey, value });
|
|
241
|
+
return groupChanged(this.properties, array.fullKey);
|
|
242
|
+
}
|
|
243
|
+
duplicateArrayGroup(array, property) {
|
|
244
|
+
const index = +property.key;
|
|
245
|
+
const newProperty = JSON.parse(JSON.stringify(property));
|
|
246
|
+
newProperty.id = propertyId();
|
|
247
|
+
const value = [...get(this.node, array.fullKey)];
|
|
248
|
+
value.splice(index, 0, safeJSONParse(newProperty.value));
|
|
249
|
+
array.properties.splice(index, 0, newProperty);
|
|
250
|
+
// need to change key on every properties recursively
|
|
251
|
+
refreshPropertyKeys(array);
|
|
252
|
+
this.nodeChange.emit({ key: array.fullKey, value });
|
|
253
|
+
return groupChanged(this.properties, array.fullKey);
|
|
254
|
+
}
|
|
255
|
+
removeArrayGroup(array, { key }) {
|
|
256
|
+
const index = +key;
|
|
257
|
+
const value = [...get(this.node, array.fullKey)];
|
|
258
|
+
value.splice(index, 1);
|
|
259
|
+
array.properties.splice(index, 1);
|
|
260
|
+
// need to change key on every properties recursively
|
|
261
|
+
refreshPropertyKeys(array);
|
|
262
|
+
this.nodeChange.emit({ key: array.fullKey, value });
|
|
263
|
+
// if no more values, remove parent group entirely
|
|
264
|
+
return array.properties.length ? null : this.propertyChanged(null, array);
|
|
265
|
+
}
|
|
266
|
+
moveArrayGroupToPosition(array, currentIndex, newIndex) {
|
|
267
|
+
const value = [...get(this.node, array.fullKey)];
|
|
268
|
+
moveItemInArray(value, currentIndex, newIndex);
|
|
269
|
+
moveItemInArray(array.properties, currentIndex, newIndex);
|
|
270
|
+
// need to change key on every properties recursively
|
|
271
|
+
refreshPropertyKeys(array);
|
|
272
|
+
this.nodeChange.emit({ key: array.fullKey, value });
|
|
273
|
+
return groupChanged(this.properties, array.fullKey);
|
|
274
|
+
}
|
|
275
|
+
moveArrayGroupUp(array, { key }) {
|
|
276
|
+
const index = +key;
|
|
277
|
+
return this.moveArrayGroupToPosition(array, index, index - 1);
|
|
278
|
+
}
|
|
279
|
+
moveArrayGroupDown(array, { key }) {
|
|
280
|
+
const index = +key;
|
|
281
|
+
return this.moveArrayGroupToPosition(array, index, index + 1);
|
|
282
|
+
}
|
|
283
|
+
propertyChanged(value, property) {
|
|
284
|
+
groupChanged(this.properties, property.fullKey, value);
|
|
285
|
+
property.addPropertyEnabled = isAddPropertyEnabled(this.node, property.schemaType, property.schema, property.fullKey);
|
|
286
|
+
if (value === null) {
|
|
287
|
+
value = undefined;
|
|
288
|
+
}
|
|
289
|
+
else {
|
|
290
|
+
property.value = value;
|
|
291
|
+
property.changed = true;
|
|
292
|
+
}
|
|
293
|
+
this.nodeChange.emit({ key: property.fullKey, value: parseNewValue(property.schema, value) });
|
|
294
|
+
}
|
|
295
|
+
addDefaultProperty(property, key, def) {
|
|
296
|
+
const node = property.key ? get(this.node, property.fullKey) : this.node;
|
|
297
|
+
const fullKey = property.key ? `${property.fullKey}.${key}` : key;
|
|
298
|
+
const properties = property.key ? property.properties : this.properties;
|
|
299
|
+
const newProperty = typeToNewProperty(this.schemas, def, this.schemas[property.schemaType], node, fullKey);
|
|
300
|
+
const existingPropertyIndex = properties.findIndex(v => v.fullKey === fullKey);
|
|
301
|
+
existingPropertyIndex >= 0 ? (properties[existingPropertyIndex] = newProperty) : properties.push(newProperty);
|
|
302
|
+
properties.sort(sortProperties);
|
|
303
|
+
return this.propertyChanged(newProperty.value, newProperty);
|
|
304
|
+
}
|
|
305
|
+
addMissingProperty(property) {
|
|
306
|
+
const parent = parentProperty(this.properties, property);
|
|
307
|
+
const newProperty = parent === property ? this.nodeProperty : parent;
|
|
308
|
+
const newProperties = this.newProperties(newProperty, property.key);
|
|
309
|
+
newProperty.newProperty = newProperties[0];
|
|
310
|
+
return newProperty.newProperty ? this.addProperty(newProperty) : null;
|
|
311
|
+
}
|
|
312
|
+
addProperty(property) {
|
|
313
|
+
const { name, prop } = property.newProperty;
|
|
314
|
+
property.newProperty = {};
|
|
315
|
+
return name === '@id' ?
|
|
316
|
+
this.replacePropertyData(property, { '@id': '', type: property.schemaType }, undefined, false) :
|
|
317
|
+
this.addDefaultProperty(property, name, prop);
|
|
318
|
+
}
|
|
319
|
+
updatePropertyCoordinates(property, value) {
|
|
320
|
+
const latitude = siblingProperty(this.properties, property, 'latitude');
|
|
321
|
+
const longitude = siblingProperty(this.properties, property, 'longitude');
|
|
322
|
+
return [
|
|
323
|
+
latitude ? this.propertyChanged(value.lat, latitude) : null,
|
|
324
|
+
longitude ? this.propertyChanged(value.lng, longitude) : null
|
|
325
|
+
];
|
|
326
|
+
}
|
|
327
|
+
// Map
|
|
328
|
+
get showMap() {
|
|
329
|
+
return (!this.editable || this.errorMode) && [SchemaType.Site, SchemaType.Organisation].includes(this.node.type);
|
|
330
|
+
}
|
|
331
|
+
mapDrawingModes({ key }) {
|
|
332
|
+
return ['boundary'].includes(key) ? [
|
|
333
|
+
google.maps.drawing.OverlayType.POLYGON
|
|
334
|
+
] : ([
|
|
335
|
+
'latitude', 'longitude'
|
|
336
|
+
].includes(key) ? [
|
|
337
|
+
google.maps.drawing.OverlayType.MARKER
|
|
338
|
+
] : []);
|
|
339
|
+
}
|
|
340
|
+
onMapDrawingClosed(newValue) {
|
|
341
|
+
const property = this.mapDrawingProperty;
|
|
342
|
+
this.mapDrawingProperty = undefined;
|
|
343
|
+
return ['latitude', 'longitude'].includes(property.key) ?
|
|
344
|
+
this.updatePropertyCoordinates(property, newValue) :
|
|
345
|
+
this.propertyChanged(newValue, property);
|
|
346
|
+
}
|
|
347
|
+
// Bibliography Search
|
|
348
|
+
bibliographiesSearchKey({ key, schemaType }) {
|
|
349
|
+
const validSchema = [SchemaType.Bibliography].includes(schemaType);
|
|
350
|
+
const valieKey = ['title', 'documentDOI', 'scopus'].includes(key);
|
|
351
|
+
return validSchema && valieKey ? key : null;
|
|
352
|
+
}
|
|
353
|
+
updateBibliography(value) {
|
|
354
|
+
const parent = parentProperty(this.properties, this.bibliographiesSearchProperty);
|
|
355
|
+
const newValue = this.bibliographiesSearchSources ? Object.assign(Object.assign({}, parent.value), { bibliography: value }) : value;
|
|
356
|
+
this.replacePropertyData(this.bibliographiesSearchProperty, newValue, true);
|
|
357
|
+
return undefined;
|
|
358
|
+
}
|
|
359
|
+
updateSource(value) {
|
|
360
|
+
// cannot replace a top-level source
|
|
361
|
+
this.bibliographiesSearchSources && this.replacePropertyData(this.bibliographiesSearchProperty, Object.assign({ '@type': NodeType.Source }, value), !('@id' in value));
|
|
362
|
+
return undefined;
|
|
363
|
+
}
|
|
364
|
+
get bibliographiesSearchSources() {
|
|
365
|
+
return this.bibliographiesSearchProperty.fullKey.split('.').length > 2;
|
|
366
|
+
}
|
|
367
|
+
onBibliographiesSearchClosed(value) {
|
|
368
|
+
this.bibliographiesSearchProperty = value ? (value.type === SchemaType.Bibliography ?
|
|
369
|
+
this.updateBibliography(value) :
|
|
370
|
+
this.updateSource(value)) : undefined;
|
|
371
|
+
}
|
|
372
|
+
// Actor.@id
|
|
373
|
+
setUserActorId(property) {
|
|
374
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
375
|
+
const { actorId, name } = yield this.usersService.loggedInUser;
|
|
376
|
+
return this.replacePropertyData(property, { '@id': actorId, '@type': NodeType.Actor, name }, false);
|
|
377
|
+
});
|
|
378
|
+
}
|
|
379
|
+
// Cycle.cycleDuration
|
|
380
|
+
calculateCycleDurationEnabled(property) {
|
|
381
|
+
return calculateCycleDurationEnabled(this.properties, property);
|
|
382
|
+
}
|
|
383
|
+
calculateCycleDuration(property) {
|
|
384
|
+
const duration = calculateCycleDuration(this.properties, property);
|
|
385
|
+
return this.propertyChanged(duration, property);
|
|
386
|
+
}
|
|
387
|
+
// Cycle.startDate
|
|
388
|
+
calculateCycleStartDateEnabled(property) {
|
|
389
|
+
return calculateCycleStartDateEnabled(this.properties, property);
|
|
390
|
+
}
|
|
391
|
+
calculateCycleStartDate(property) {
|
|
392
|
+
const startDate = calculateCycleStartDate(this.properties, property);
|
|
393
|
+
return this.propertyChanged(startDate, property);
|
|
394
|
+
}
|
|
395
|
+
// Units converter
|
|
396
|
+
unitConverterEnabled(property) {
|
|
397
|
+
return this.editable && property.editable && ['value', 'min', 'max', 'sd'].includes(property.key);
|
|
398
|
+
}
|
|
399
|
+
openUnitConverter(popover, property) {
|
|
400
|
+
const value = property.value;
|
|
401
|
+
const termProperty = siblingProperty(this.properties, property, 'term');
|
|
402
|
+
const term = safeJSONParse(termProperty === null || termProperty === void 0 ? void 0 : termProperty.value);
|
|
403
|
+
const units = term === null || term === void 0 ? void 0 : term.units;
|
|
404
|
+
return popover.isOpen() ? popover.close() : popover.open({ value, term, units });
|
|
405
|
+
}
|
|
406
|
+
}
|
|
407
|
+
FilesFormComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: FilesFormComponent, deps: [{ token: i0.ElementRef }, { token: i1.HeSearchService }, { token: i2.HeUsersService }], target: i0.ɵɵFactoryTarget.Component });
|
|
408
|
+
FilesFormComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.16", type: FilesFormComponent, selector: "he-files-form", inputs: { schemas: "schemas", errors: "errors", node: "node", nodeMap: "nodeMap", editable: "editable", errorMode: "errorMode", deepEditable: "deepEditable", errorsEditable: "errorsEditable" }, outputs: { nodeChange: "nodeChange", nodeErorrResolved: "nodeErorrResolved", nodeErrorAdded: "nodeErrorAdded" }, ngImport: i0, template: "<div class=\"card\">\n <div class=\"card-toggle p-4\" (click)=\"isOpen = !isOpen\" pointer>\n <fa-icon icon=\"angle-down\" [class.is-hidden]=\"!isOpen\"></fa-icon>\n <fa-icon icon=\"angle-right\" [class.is-hidden]=\"isOpen\"></fa-icon>\n <span *ngIf=\"nodeProperty\" class=\"is-px-2\"\n [class.has-text-danger]=\"nodeProperty.hasError\"\n [class.has-text-warning]=\"nodeProperty.hasWarning\"\n >\n <he-node-icon [type]=\"nodeProperty.schemaType\"></he-node-icon>\n </span>\n </div>\n\n <ng-container *ngIf=\"editable && isOpen && nodeProperty\">\n <ng-container *ngTemplateOutlet=\"showNewProperty; context: {$implicit: nodeProperty}\"></ng-container>\n </ng-container>\n\n <div class=\"card-content\">\n <ng-container *ngIf=\"isOpen\">\n <div class=\"pb-3 mb-2\" *ngIf=\"errorsEditable\">\n <ng-container *ngTemplateOutlet=\"nodeErrorForm; context: {$implicit: nodeProperty}\"></ng-container>\n </div>\n\n <div class=\"mb-4\" *ngIf=\"nodeProperty?.error\">\n <ng-container *ngTemplateOutlet=\"propertyError; context: {property: nodeProperty, edit: true}\"></ng-container>\n </div>\n </ng-container>\n\n <div class=\"columns is-multiline\">\n <ng-container *ngFor=\"let property of properties; trackBy: trackByProperty\">\n <ng-container *ngTemplateOutlet=\"showProperty; context: {$implicit: property}\"></ng-container>\n </ng-container>\n </div>\n\n <ng-container *ngIf=\"isOpen\">\n <ng-container *ngTemplateOutlet=\"propertyMap; context: {$implicit: nodeProperty}\"></ng-container>\n </ng-container>\n </div>\n</div>\n\n<he-maps-drawing-confirm *ngIf=\"!!mapDrawingProperty\"\n [value]=\"mapDrawingProperty.value\" [modes]=\"mapDrawingModes(mapDrawingProperty)\"\n (closed)=\"onMapDrawingClosed($event)\"\n></he-maps-drawing-confirm>\n\n<he-bibliographies-search-confirm *ngIf=\"!!bibliographiesSearchProperty\"\n [search]=\"bibliographiesSearchProperty.value\"\n [searchBy]=\"bibliographiesSearchKey(bibliographiesSearchProperty)\"\n [searchSources]=\"bibliographiesSearchSources\"\n (closed)=\"onBibliographiesSearchClosed($event)\"\n></he-bibliographies-search-confirm>\n\n<ng-template #labelDescription let-property>\n <span\n class=\"trigger-popover\"\n [ngbPopover]=\"property.schema?.description\" [autoClose]=\"'outside'\"\n triggers=\"hover\" placement=\"right\" container=\"body\"\n >\n <span>{{property.key}}</span>\n </span>\n</ng-template>\n\n<ng-template #labelDefault let-property>\n <span>{{property.key}}</span>\n</ng-template>\n\n<ng-template #showProperty let-property>\n <ng-container *ngIf=\"(isOpen || property.closedVisible) && !property.isHidden\">\n <div class=\"column is-6\"\n [id]=\"property.fullKey + '_' + property.id\"\n [class.is-12]=\"property.properties.length || !property.key\"\n [ngSwitch]=\"!!property.properties.length\"\n >\n <div class=\"columns is-multiline is-variable is-1\" *ngSwitchCase=\"false\">\n <div class=\"column is-3 py-1\" *ngIf=\"property.key\">\n <label class=\"label has-text-right-tablet has-text-ellipsis\"\n *bindOnce=\"property\"\n [for]=\"property.id\"\n >\n <ng-container\n *ngTemplateOutlet=\"property.schema?.description && editable && property.editable ? labelDescription : labelDefault; context: {$implicit: property}\">\n </ng-container>\n </label>\n </div>\n\n <div class=\"column is-9 py-1\" [class.is-12]=\"!property.key\">\n <div class=\"field\">\n <ng-container *ngTemplateOutlet=\"inputForm; context: {$implicit: property}\"></ng-container>\n\n <ng-container *ngIf=\"(editable || errorsEditable) && (property.hasError || property.hasWarning)\">\n <ng-container *ngTemplateOutlet=\"propertyError; context: {property: property, edit: false}\"></ng-container>\n </ng-container>\n </div>\n </div>\n </div>\n\n <div class=\"panel\" *ngSwitchCase=\"true\"\n [class.is-link]=\"errorMode && property.changed\"\n [class.is-danger]=\"!property.changed && property.hasError\"\n [class.is-warning]=\"!property.changed && property.hasWarning\"\n [class.is-default]=\"!property.changed && !property.hasError && !property.hasWarning\"\n >\n <div class=\"open-group panel-heading py-0\"\n (click)=\"property.isOpen = !property.isOpen\" pointer\n [class.is-open]=\"property.isOpen\"\n [class.has-text-white]=\"(errorMode && property.changed) || property.hasError || property.hasWarning\"\n >\n <div class=\"columns is-mobile is-vcentered\">\n <span class=\"column is-narrow py-1 my-0\">\n <fa-icon icon=\"angle-down\" [class.is-hidden]=\"!property.isOpen\"></fa-icon>\n <fa-icon icon=\"angle-right\" [class.is-hidden]=\"property.isOpen\"></fa-icon>\n </span>\n <span class=\"column py-1 my-0\">\n <span class=\"is-size-6\">{{property.key}}</span>\n </span>\n <span *ngIf=\"property.schemaType\" class=\"column is-narrow py-1 my-0\">\n <span class=\"tags mb-0 has-addons\">\n <span class=\"tag mb-0 is-light\">Type</span>\n <span class=\"tag mb-0 is-white\">\n <he-schema-version-link linkClass=\"is-small\" [node]=\"{'@type': property.schemaType}\">\n <span>{{property.schemaType}}</span>\n </he-schema-version-link>\n </span>\n </span>\n </span>\n <he-popover-confirm class=\"column is-narrow py-1 my-0 px-0\"\n *ngIf=\"editable && !errorsEditable\"\n ngbTooltip=\"Remove group\" placement=\"top\"\n [message]=\"'<p>This will remove the group completely.</p>' + (property.isRequired ? '<p><u>Warning: this field is required.</u></p>' : '') + '<p>Do you confirm?</p>'\"\n popoverClass=\"px-3\"\n (confirmed)=\"propertyChanged(null, property)\"\n >\n <fa-icon icon=\"times\" size=\"sm\"></fa-icon>\n </he-popover-confirm>\n </div>\n </div>\n <div class=\"panel-block is-block p-0\" *ngIf=\"property.isOpen\">\n <ng-container [ngSwitch]=\"property.isArray\">\n <div class=\"px-3 pt-4\" *ngIf=\"property.error\">\n <ng-container *ngTemplateOutlet=\"propertyError; context: {property: property, edit: true}\"></ng-container>\n </div>\n\n <div class=\"mt-3\" *ngSwitchCase=\"false\">\n <p class=\"help py-1 px-2\" *ngIf=\"editable && !property.editable\">\n To change the {{property.key}}, please delete it first, then add the field again\n </p>\n <ng-container *ngIf=\"editable && property.editable\">\n <ng-container [ngSwitch]=\"property.schema?.title\">\n <p class=\"help py-1 px-2\" *ngSwitchCase=\"'Bibliography'\">\n Search by Title or Document DOI to auto-populate the fields using the Mendeley catalogue\n </p>\n </ng-container>\n </ng-container>\n\n <ng-container *ngTemplateOutlet=\"showNewProperty; context: {$implicit: property}\"></ng-container>\n\n <div class=\"px-3 mt-1\" *ngIf=\"errorsEditable\">\n <ng-container *ngTemplateOutlet=\"nodeErrorForm; context: {$implicit: property}\"></ng-container>\n </div>\n\n <div class=\"property-group py-2 px-3 mt-2\">\n <div class=\"columns is-multiline mb-0\">\n <ng-container *ngFor=\"let prop2 of property.properties; trackBy: trackByProperty\">\n <ng-container *ngTemplateOutlet=\"showProperty; context: {$implicit: prop2}\"></ng-container>\n </ng-container>\n </div>\n </div>\n </div>\n\n <div class=\"py-2 px-3 mt-2\" *ngSwitchCase=\"true\">\n <div class=\"mt-1\" *ngIf=\"errorsEditable\">\n <ng-container *ngTemplateOutlet=\"nodeErrorForm; context: {$implicit: property}\"></ng-container>\n </div>\n\n <ng-container *ngFor=\"let prop2 of property.properties; trackBy: trackByProperty\">\n <div class=\"card p-0 my-4\" *ngIf=\"prop2.key\"\n [id]=\"prop2.fullKey + '_' + prop2.id\"\n >\n <div class=\"property-array-number\">\n <div class=\"tags has-addons\">\n <span class=\"tag is-dark\">{{prop2.key}}</span>\n <ng-container *ngIf=\"editable && !errorsEditable && property.editable\">\n <span class=\"tag is-info\" pointer\n (click)=\"duplicateArrayGroup(property, prop2)\"\n [ngbTooltip]=\"'Duplicate ' + pluralize(property.key, 1)\" placement=\"top\"\n >\n <fa-icon icon=\"clone\" size=\"sm\"></fa-icon>\n </span>\n <span class=\"tag is-light\" pointer\n *ngIf=\"prop2.key !== '0'\"\n (click)=\"moveArrayGroupUp(property, prop2)\"\n ngbTooltip=\"Move Up\" placement=\"top\"\n >\n <fa-icon icon=\"long-arrow-alt-up\" size=\"sm\"></fa-icon>\n </span>\n <span class=\"tag is-light\" pointer\n *ngIf=\"prop2.key !== property.properties.length - 1\"\n (click)=\"moveArrayGroupDown(property, prop2)\"\n ngbTooltip=\"Move Down\" placement=\"top\"\n >\n <fa-icon icon=\"long-arrow-alt-down\" size=\"sm\"></fa-icon>\n </span>\n <he-popover-confirm class=\"tag is-delete\"\n [ngbTooltip]=\"'Remove ' + pluralize(property.key, 1)\" placement=\"top\"\n message=\"This will remove the group completely. Do you confirm?\" position=\"right\"\n (confirmed)=\"removeArrayGroup(property, prop2)\"\n ></he-popover-confirm>\n </ng-container>\n </div>\n </div>\n\n <ng-container *ngTemplateOutlet=\"showNewProperty; context: {$implicit: prop2}\"></ng-container>\n\n <div class=\"px-4 mt-2\" *ngIf=\"errorsEditable\">\n <ng-container *ngTemplateOutlet=\"nodeErrorForm; context: {$implicit: prop2}\"></ng-container>\n </div>\n\n <div class=\"px-4 mt-2\" *ngIf=\"prop2.error\">\n <ng-container *ngTemplateOutlet=\"propertyError; context: {property: prop2, edit: true}\"></ng-container>\n </div>\n\n <div class=\"property-group card-content p-3\">\n <div class=\"columns is-multiline my-0\">\n <ng-container *ngFor=\"let prop3 of prop2.properties; trackBy: trackByProperty\">\n <ng-container *ngTemplateOutlet=\"showProperty; context: {$implicit: prop3}\"></ng-container>\n </ng-container>\n </div>\n </div>\n </div>\n </ng-container>\n\n <button class=\"button is-dark is-outlined is-small\" type=\"button\"\n *ngIf=\"editable && !errorsEditable\"\n (click)=\"addArrayGroup(property)\"\n >\n <fa-icon icon=\"plus-circle\"></fa-icon>\n <span class=\"pl-2\" i18n=\"@@add\">Add</span>\n <span class=\"pl-1\">{{property.key | pluralize:1}}</span>\n </button>\n </div>\n </ng-container>\n </div>\n </div>\n </div>\n </ng-container>\n</ng-template>\n\n<ng-template #inputForm let-property>\n <ng-container *ngIf=\"property.key\">\n <ng-container [ngSwitch]=\"property.suggestions?.type\">\n <ng-container *ngSwitchCase=\"'select'\">\n <ng-container *ngTemplateOutlet=\"inputSelect; context: {$implicit: property}\"></ng-container>\n </ng-container>\n <ng-container *ngSwitchDefault>\n <ng-container *ngTemplateOutlet=\"inputInput; context: {$implicit: property}\"></ng-container>\n </ng-container>\n </ng-container>\n </ng-container>\n</ng-template>\n\n<ng-template #inputInput let-property>\n <div class=\"field mb-0\" [class.has-addons]=\"hasAddons(property)\">\n <div class=\"control is-expanded\"\n [class.has-icons-right]=\"property.loading\"\n >\n <input class=\"input is-small search-input\"\n [class.is-dark]=\"property.key === 'type'\"\n [class.is-link]=\"errorMode && property.changed\"\n [class.is-danger]=\"(!property.changed && property.hasError) || isRequired(property)\"\n [class.is-warning]=\"!property.changed && property.hasWarning\"\n\n [(ngModel)]=\"property.value\" #propertyModel=\"ngModel\"\n [type]=\"property.schema?.type === 'number' ? 'number' : 'string'\"\n [id]=\"property.id\"\n name=\"randomname\"\n [readonly]=\"!editable || !property.editable || property.schema?.internal\"\n [placeholder]=\"property.placeholder\"\n [appTagsInput]=\"{enabled: editable && property.editable && property.schema?.type === 'array', items: property.schema?.items, delimiter: ';', allowDuplicates: true, placeholder: property.placeholder}\"\n (change)=\"propertyChanged($event.target.value, property)\"\n\n [pattern]=\"property.schema?.pattern\"\n [required]=\"property.fullKey.endsWith('id') && property.fullKey !== 'id'\"\n [min]=\"property.schema?.minimum\"\n [max]=\"property.schema?.maximum\"\n\n [ngbTypeahead]=\"propertySuggest(property.fullKey, property.suggestions?.type)\"\n [resultTemplate]=\"suggestion\"\n [inputFormatter]=\"formatter\"\n [focusFirst]=\"true\"\n (focus)=\"editable && property.editable && typeaheadFocus($event)\"\n (selectItem)=\"suggestionSelected($event.item, property)\"\n >\n\n <span class=\"icon is-small is-right has-text-grey-dark\" [class.is-hidden]=\"!property.loading\">\n <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"sm\"></fa-icon>\n </span>\n </div>\n <ng-container *ngTemplateOutlet=\"inputAddons; context: {$implicit: property}\"></ng-container>\n </div>\n <p class=\"help is-danger-light\"\n *ngIf=\"!property.hasError && propertyModel.invalid\"\n >\n <span *bindOnce=\"propertyModel.errors\" [innerHTML]=\"formatPropertyError(propertyModel.errors, property)\"></span>\n </p>\n</ng-template>\n\n<ng-template #inputSelect let-property>\n <div class=\"field mb-0\" [class.has-addons]=\"hasAddons(property)\">\n <div class=\"control is-expanded\">\n <div class=\"select is-small is-fullwidth\"\n [class.is-link]=\"errorMode && property.changed\"\n [class.is-danger]=\"(!property.changed && property.hasError) || isRequired(property)\"\n [class.is-warning]=\"!property.changed && property.hasWarning\"\n >\n <select\n [(ngModel)]=\"property.value\" #propertyModel=\"ngModel\"\n [id]=\"property.id\"\n name=\"randomname\"\n [disabled]=\"!editable || !property.editable || property.schema?.internal\"\n (change)=\"propertyChanged($event.target.value, property)\"\n >\n <option value=\"\">Select</option>\n <ng-container *bindOnce=\"property.suggestions\">\n <option *ngFor=\"let value of property.suggestions.values; trackBy: trackByIndex\" [value]=\"value\">{{value}}</option>\n </ng-container>\n </select>\n </div>\n </div>\n <ng-container *ngTemplateOutlet=\"inputAddons; context: {$implicit: property}\"></ng-container>\n </div>\n</ng-template>\n\n<ng-template #removeFieldAddon let-property>\n <div class=\"control\" *ngIf=\"!errorsEditable && !property.isRequired\">\n <a class=\"button is-small\" title=\"Remove field\"\n [class.is-outlined]=\"!property.changed && (property.hasError || property.hasWarning)\"\n [class.is-danger]=\"!property.changed && property.hasError\"\n [class.is-warning]=\"!property.changed && property.hasWarning\"\n (click)=\"$event.stopPropagation(); propertyChanged(null, property)\"\n >\n <fa-icon icon=\"times\"></fa-icon>\n </a>\n </div>\n</ng-template>\n\n<ng-template #inputAddons let-property>\n <he-popover-confirm class=\"control\"\n *ngIf=\"enableAddError(property)\"\n position=\"left\"\n [content]=\"popupErrorForm\"\n (confirmed)=\"addError(property, $event)\"\n >\n <span class=\"button is-small\">\n <fa-icon icon=\"comments\"></fa-icon>\n </span>\n </he-popover-confirm>\n\n <ng-container *ngIf=\"editable && property.editable\">\n <ng-container [ngSwitch]=\"property.schemaType\">\n <ng-container *ngSwitchCase=\"SchemaType.Actor\">\n <ng-container *ngTemplateOutlet=\"actorAddons; context: {$implicit: property}\"></ng-container>\n </ng-container>\n <ng-container *ngSwitchCase=\"SchemaType.Cycle\">\n <ng-container *ngTemplateOutlet=\"cycleAddons; context: {$implicit: property}\"></ng-container>\n </ng-container>\n </ng-container>\n <ng-container *bindOnce=\"property\">\n <div class=\"control\" *ngIf=\"mapDrawingModes(property).length\">\n <button class=\"button is-small\" title=\"Pick on Map\"\n (click)=\"mapDrawingProperty = property\"\n >\n <fa-icon icon=\"map-marked-alt\"></fa-icon>\n </button>\n </div>\n <div class=\"control\" *ngIf=\"bibliographiesSearchKey(property)\">\n <button class=\"button is-small\" title=\"Advanced Search\"\n (click)=\"bibliographiesSearchProperty = property\"\n >\n <fa-icon icon=\"search\"></fa-icon>\n </button>\n </div>\n </ng-container>\n\n <ng-container *ngTemplateOutlet=\"removeFieldAddon; context: {$implicit: property}\"></ng-container>\n </ng-container>\n <div class=\"control\" *ngIf=\"addPropertyEnabled(property)\">\n <a class=\"button is-small is-danger\" title=\"Add field\"\n (click)=\"$event.stopPropagation(); addMissingProperty(property)\"\n >\n <fa-icon icon=\"plus-circle\"></fa-icon>\n <span class=\"pl-2\" i18n=\"@@add\">Add</span>\n </a>\n </div>\n <div class=\"control\" *ngIf=\"isRequired(property)\">\n <label class=\"button is-small is-danger\" [for]=\"property.id\"\n ngbTooltip=\"This field is required\" placement=\"top\"\n >\n <fa-icon icon=\"exclamation-triangle\"></fa-icon>\n </label>\n </div>\n <ng-container *ngIf=\"property.externalUrl?.url\">\n <div class=\"control\">\n <a class=\"button is-small\"\n [href]=\"property.externalUrl.url + (property.externalUrl.urlParamValue ? property.value : '')\"\n target=\"_blank\"\n [title]=\"property.externalUrl.title\"\n [ngClass]=\"{'is-dark is-outlined': property.key === 'type'}\"\n [attr.disabled]=\"property.externalUrl.urlParamValue && !property.value ? true : null\"\n >\n <fa-icon [icon]=\"property.externalUrl.icon || 'external-link-alt'\"></fa-icon>\n </a>\n </div>\n </ng-container>\n <ng-container *ngIf=\"unitConverterEnabled(property)\">\n <div class=\"control\">\n <button class=\"button is-small\" title=\"Open calculator\"\n [ngbPopover]=\"convertUnits\" autoClose=\"outside\"\n triggers=\"manual\" #p=\"ngbPopover\" placement=\"bottom\" container=\"body\"\n (click)=\"openUnitConverter(p, property)\"\n >\n <fa-icon icon=\"calculator\"></fa-icon>\n </button>\n </div>\n </ng-container>\n</ng-template>\n\n<ng-template #actorAddons let-property>\n <div class=\"control\" *ngIf=\"property.key === '@id'\">\n <button class=\"button is-small\" title=\"Add myself as Actor\"\n (click)=\"setUserActorId(property)\"\n >\n <fa-icon [icon]=\"['far', 'id-badge']\"></fa-icon>\n </button>\n </div>\n</ng-template>\n\n<ng-template #cycleAddons let-property>\n <div class=\"control\" *ngIf=\"property.key === 'cycleDuration'\">\n <button class=\"button is-small\" title=\"Calculate value from startDate and endDate\"\n (click)=\"calculateCycleDuration(property)\"\n [disabled]=\"!calculateCycleDurationEnabled(property)\"\n >\n <fa-icon icon=\"calculator\"></fa-icon>\n </button>\n </div>\n <div class=\"control\" *ngIf=\"property.key === 'startDate'\">\n <button class=\"button is-small\" title=\"Calculate value from endDate and cycleDuration\"\n (click)=\"calculateCycleStartDate(property)\"\n [disabled]=\"!calculateCycleStartDateEnabled(property)\"\n >\n <fa-icon icon=\"calculator\"></fa-icon>\n </button>\n </div>\n</ng-template>\n\n<ng-template #showNewProperty let-property>\n <header class=\"card-header\" *ngIf=\"editable && property.editable && (property.addPropertyEnabled || deepEditable); else padder\">\n <form class=\"py-3 px-4 is-flex-grow-1\" (submit)=\"addProperty(property)\" novalidate>\n <div class=\"field is-horizontal\">\n <div class=\"field-label is-small\">\n <label class=\"label\" [for]=\"property.id + '_new'\">\n <span i18n=\"@@files.form.newProperty\">Add new field</span>\n </label>\n </div>\n <div class=\"field-body\">\n <div class=\"field has-addons\">\n <div class=\"control is-expanded\">\n <input class=\"input is-small\"\n [(ngModel)]=\"property.newProperty\"\n [id]=\"property.id + '_new'\"\n name=\"randomname\"\n placeholder=\"Search and select field from results\"\n\n [ngbTypeahead]=\"suggestNewProperty(property.fullKey)\"\n [resultTemplate]=\"suggestion\"\n [inputFormatter]=\"formatter\"\n [focusFirst]=\"false\"\n [editable]=\"false\"\n (focus)=\"typeaheadFocus($event)\"\n >\n </div>\n <div class=\"control\">\n <button class=\"button is-small\" type=\"submit\"\n [disabled]=\"!property.newProperty || !property.newProperty.name\"\n >\n <fa-icon icon=\"plus\"></fa-icon>\n <span class=\"pl-2\" i18n=\"@@add\">Add</span>\n </button>\n </div>\n </div>\n </div>\n </div>\n </form>\n </header>\n</ng-template>\n\n<ng-template #propertyError let-property=\"property\" let-edit=\"edit\">\n <p class=\"help\"\n [class.is-danger]=\"property.hasError\"\n [class.is-warning]=\"property.hasWarning\"\n *ngIf=\"property.error\"\n >\n <span class=\"is-pre-wrap\" *bindOnce=\"property.error\" [innerHTML]=\"property.error.message\"></span>\n <a class=\"pl-2\"\n *ngIf=\"edit && errorsEditable && property.error.index >= 0\"\n (click)=\"editError(property)\"\n >\n <fa-icon class=\"pr-2\" icon=\"edit\"></fa-icon>\n <span i18n=\"@@edit\">Edit</span>\n </a>\n <a class=\"pl-2\"\n *ngIf=\"(property.hasWarning || errorsEditable) && property.error.index >= 0\"\n (click)=\"resolveError(property)\"\n >\n <fa-icon class=\"pr-2\" icon=\"check\"></fa-icon>\n <span i18n=\"@@resolved\">Resolved</span>\n </a>\n </p>\n</ng-template>\n\n<ng-template #propertyMap let-property>\n <div class=\"panel is-default\" *ngIf=\"showMap\">\n <div class=\"open-group panel-heading py-0\"\n (click)=\"mapVisible = !mapVisible\" pointer\n [class.is-open]=\"mapVisible\"\n >\n <div class=\"columns is-mobile is-vcentered mb-0\">\n <span class=\"column is-narrow py-1 my-0\">\n <fa-icon icon=\"angle-down\" [class.is-hidden]=\"!mapVisible\"></fa-icon>\n <fa-icon icon=\"angle-right\" [class.is-hidden]=\"mapVisible\"></fa-icon>\n </span>\n <span class=\"column py-1 my-0\">\n <span class=\"is-size-6\">View on Map</span>\n </span>\n </div>\n </div>\n <div class=\"panel-block is-block p-0\" [class.is-hidden]=\"!mapVisible\">\n <he-sites-maps [sites]=\"[node]\" [showNotice]=\"false\"></he-sites-maps>\n </div>\n </div>\n</ng-template>\n\n<ng-template #nodeErrorForm let-property>\n <ng-container *ngIf=\"property.newError && property.editable\">\n <div class=\"field has-addons\">\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 class=\"textarea is-small\"\n [(ngModel)]=\"property.newError.message\"\n placeholder=\"Enter your message here\"\n rows=\"1\"\n ></textarea>\n </div>\n <div class=\"control\">\n <button class=\"button is-small\"\n [disabled]=\"!property.newError.level || !property.newError.message\"\n (click)=\"addError(property, property.newError)\"\n >\n <fa-icon icon=\"plus-circle\"></fa-icon>\n <span class=\"pl-2\" i18n=\"@@add\">Add</span>\n </button>\n </div>\n </div>\n </ng-container>\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\"\n [(ngModel)]=\"data.message\"\n placeholder=\"Enter your message here\"\n >\n </div>\n </div>\n</ng-template>\n\n<ng-template #suggestion let-r=\"result\" let-t=\"term\">\n <ngb-highlight\n [title]=\"r.bibliography?.title || r.bibliography?.documentDOI || r.bibliography?.scopus || r.name\"\n [result]=\"r.bibliography?.title || r.bibliography?.documentDOI || r.bibliography?.scopus || r.name\"\n [term]=\"t\"\n ></ngb-highlight>\n</ng-template>\n\n<ng-template #padder>\n <div class=\"pt-1\"></div>\n</ng-template>\n\n<ng-template #convertUnits let-value=\"value\" let-term=\"term\" let-units=\"units\">\n <he-unit-converter [value]=\"value\" [term]=\"term\" [toUnits]=\"units\"></he-unit-converter>\n</ng-template>\n", styles: [".panel.is-default .panel-heading{background-color:#ededed;color:#363636}.card-toggle{left:0;position:absolute;top:0}@media screen and (max-width: 768px){.card-toggle{position:relative}}.card-toggle>fa-icon{display:inline-block;width:10px}.card{overflow:visible}.card .card{box-shadow:2px 2px #36363652,0 0 0 1px #36363652}.card .card>.card-header{box-shadow:0 2px 1px #36363652}.property-array-number{left:-4px;position:absolute;top:-12px}.property-array-number .tag.is-delete{border:1px solid rgba(54,54,54,.32)}.control>.button{height:100%}.is-danger-light{color:#f5758f}he-sites-maps{height:200px}\n"], components: [{ type: i3.FaIconComponent, selector: "fa-icon", inputs: ["classes", "icon", "title", "spin", "pulse", "mask", "styles", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "transform", "a11yRole"] }, { type: i4.NodeIconComponent, selector: "he-node-icon", inputs: ["type", "size"] }, { type: i5.MapsDrawingConfirmComponent, selector: "he-maps-drawing-confirm", inputs: ["value", "modes", "center", "zoom"], outputs: ["closed"] }, { type: i6.BibliographiesSearchConfirmComponent, selector: "he-bibliographies-search-confirm", inputs: ["search", "searchSources", "searchBibliographies", "searchBy"], outputs: ["closed"] }, { type: i7.SchemaVersionLinkComponent, selector: "he-schema-version-link", inputs: ["node", "showExternalLink", "linkClass", "text"] }, { type: i8.PopoverConfirmComponent, selector: "he-popover-confirm", inputs: ["message", "content", "position", "popoverClass"], outputs: ["confirmed"] }, { type: i9.SitesMapsComponent, selector: "he-sites-maps", inputs: ["loadPolygons", "sites", "nodes", "center", "zoom", "showNotice"] }, { type: i10.NgbHighlight, selector: "ngb-highlight", inputs: ["highlightClass", "result", "term"] }, { type: i11.UnitConverterComponent, selector: "he-unit-converter", inputs: ["term", "value", "fromUnits", "toUnits"] }], directives: [{ type: i12.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i12.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i12.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i10.NgbPopover, selector: "[ngbPopover]", inputs: ["animation", "autoClose", "placement", "triggers", "container", "disablePopover", "popoverClass", "openDelay", "closeDelay", "ngbPopover", "popoverTitle"], outputs: ["shown", "hidden"], exportAs: ["ngbPopover"] }, { type: i12.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { type: i12.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { type: i13.BindOnceDirective, selector: "[bindOnce]", inputs: ["bindOnce"] }, { type: i10.NgbTooltip, selector: "[ngbTooltip]", inputs: ["animation", "autoClose", "placement", "triggers", "container", "disableTooltip", "tooltipClass", "openDelay", "closeDelay", "ngbTooltip"], outputs: ["shown", "hidden"], exportAs: ["ngbTooltip"] }, { type: i12.NgSwitchDefault, selector: "[ngSwitchDefault]" }, { type: i14.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { type: i10.NgbTypeahead, selector: "input[ngbTypeahead]", inputs: ["autocomplete", "placement", "container", "editable", "focusFirst", "showHint", "inputFormatter", "ngbTypeahead", "resultFormatter", "resultTemplate"], outputs: ["selectItem"], exportAs: ["ngbTypeahead"] }, { type: i14.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i14.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { type: i15.TagsInputDirective, selector: "[appTagsInput]", inputs: ["appTagsInput"] }, { type: i14.PatternValidator, selector: "[pattern][formControlName],[pattern][formControl],[pattern][ngModel]", inputs: ["pattern"] }, { type: i14.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { type: i14.SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: ["compareWith"] }, { type: i14.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { type: i14.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { type: i12.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i14.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { type: i14.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { type: i14.NgForm, selector: "form:not([ngNoForm]):not([formGroup]),ng-form,[ngForm]", inputs: ["ngFormOptions"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }], pipes: { "pluralize": i16.PluralizePipe } });
|
|
409
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: FilesFormComponent, decorators: [{
|
|
410
|
+
type: Component,
|
|
411
|
+
args: [{
|
|
412
|
+
selector: 'he-files-form',
|
|
413
|
+
templateUrl: './files-form.component.html',
|
|
414
|
+
styleUrls: ['./files-form.component.scss']
|
|
415
|
+
}]
|
|
416
|
+
}], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i1.HeSearchService }, { type: i2.HeUsersService }]; }, propDecorators: { schemas: [{
|
|
417
|
+
type: Input
|
|
418
|
+
}], errors: [{
|
|
419
|
+
type: Input
|
|
420
|
+
}], node: [{
|
|
421
|
+
type: Input
|
|
422
|
+
}], nodeMap: [{
|
|
423
|
+
type: Input
|
|
424
|
+
}], editable: [{
|
|
425
|
+
type: Input
|
|
426
|
+
}], errorMode: [{
|
|
427
|
+
type: Input
|
|
428
|
+
}], deepEditable: [{
|
|
429
|
+
type: Input
|
|
430
|
+
}], errorsEditable: [{
|
|
431
|
+
type: Input
|
|
432
|
+
}], nodeChange: [{
|
|
433
|
+
type: Output
|
|
434
|
+
}], nodeErorrResolved: [{
|
|
435
|
+
type: Output
|
|
436
|
+
}], nodeErrorAdded: [{
|
|
437
|
+
type: Output
|
|
438
|
+
}] } });
|
|
439
|
+
//# 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":["/* eslint-disable complexity */\nimport { Component, OnInit, Input, Output, EventEmitter, ElementRef } from '@angular/core';\nimport { Observable, of } from 'rxjs';\nimport { debounceTime, distinctUntilChanged, map, mergeMap, switchMap, tap } from 'rxjs/operators';\nimport * as pluralize from 'pluralize';\nimport { moveItemInArray } from '@angular/cdk/drag-drop';\nconst get = require('lodash.get');\nimport { JSON as HestiaJson, SchemaType, NodeType, Bibliography, Source } from '@hestia-earth/schema';\nimport { definitions, definition } from '@hestia-earth/json-schema';\nimport { isEmpty } from '@hestia-earth/utils';\n\nimport {\n  IErrorProperty, INodeProperty,\n  formatPropertyError, propertyError, hasError, hasWarning,\n  formatSuggestion, formatLinkNodesSuggestions, suggestionType, defaultSuggestionType,\n  recursiveProperties, typeToNewProperty, propertyId,\n  isAddPropertyEnabled, nodeAvailableProperties, sortProperties,\n  groupChanged, parentProperty, siblingProperty, updateProperties, refreshPropertyKeys, keyToDataPath, findProperty,\n  calculateCycleDurationEnabled, calculateCycleDuration,\n  calculateCycleStartDateEnabled, calculateCycleStartDate, parseNewValue\n} from '../files-form.model';\nimport { IValidationError, errorHasError, errorHasWarning, filterError, formatError } from '../files-error.model';\nimport { HeUsersService } from '../../users/users.service';\nimport { HeSearchService } from '../../search/search.service';\nimport { matchType, matchPhraseQuery, matchPhrasePrefixQuery, matchBoolPrefixQuery } from '../../search/search.model';\nimport { safeJSONParse } from '../../common/utils';\n\nconst MIN_TYPEAHEAD_LENGTH = 2;\nconst populateTermFields = [\n  'termType', 'units',\n  'defaultProperties.term.name', 'defaultProperties.term.units', 'defaultProperties.value'\n];\n\ntype suggestFunc = (term: string, property: INodeProperty) => Observable<any[]>;\n\n@Component({\n  selector: 'he-files-form',\n  templateUrl: './files-form.component.html',\n  styleUrls: ['./files-form.component.scss']\n})\nexport class FilesFormComponent implements OnInit {\n  @Input()\n  private schemas?: definitions;\n  @Input()\n  private errors: IValidationError[] = [];\n  @Input()\n  public node?: HestiaJson<SchemaType>;\n  @Input()\n  public nodeMap: { [type in NodeType]?: string[] } = {};\n  @Input()\n  public editable = false;\n  @Input()\n  public errorMode = false;\n  @Input()\n  public deepEditable = true;\n  @Input()\n  public errorsEditable = false;\n\n  @Output()\n  private nodeChange = new EventEmitter<{ key: string; value: any }>();\n  @Output()\n  private nodeErorrResolved = new EventEmitter<number>();\n  @Output()\n  private nodeErrorAdded = new EventEmitter<IValidationError>();\n\n  public NodeType = NodeType;\n  public SchemaType = SchemaType;\n  public isOpen = true;\n  public mapVisible = false;\n  public pluralize = pluralize;\n  public formatPropertyError = formatPropertyError;\n  public mapDrawingProperty?: INodeProperty;\n  public bibliographiesSearchProperty?: INodeProperty;\n  public nodeProperty?: INodeProperty;\n  public properties: INodeProperty[] = [];\n\n  public formatter = (value: any) =>\n    typeof value === 'object' ? ('bibliography' in value ? value.bibliography?.title : value.name) : value;\n\n  public suggestNewProperty = (fullKey: string) => (text$: Observable<string>) =>\n    text$.pipe(\n      distinctUntilChanged(),\n      switchMap(\n        term => of(this.newProperties(findProperty(this.properties, fullKey) || this.nodeProperty, term))\n      )\n    );\n\n  private suggestExistingNode = (type: NodeType, term: string, uniqueKey = '@id') =>\n    term.length < MIN_TYPEAHEAD_LENGTH ?\n      of([]) :\n      this.searchService.suggest(term, type, [], {\n        bool: {\n          must: [matchType(type)],\n          should: [\n            matchPhraseQuery(term, 100, uniqueKey),\n            matchPhrasePrefixQuery(term, 20, uniqueKey),\n            matchBoolPrefixQuery(term, 10, uniqueKey)\n          ],\n          minimum_should_match: 1\n        }\n      }, 5, populateTermFields);\n\n  private suggestNode = (type: NodeType, suggestDefault?: suggestFunc) => (term: string, property: INodeProperty) =>\n    (property.suggestions?.isUniqueKey ? (\n      this.suggestExistingNode(type, term, property.suggestions.isUniqueKey)\n    ) : (\n      property.suggestions?.isLinkNode ?\n        of(formatLinkNodesSuggestions(this.nodeMap, type)) : (\n          suggestDefault ? suggestDefault(term, property) : of([])\n        )\n    )).pipe(\n      map((res: any) => res.map(formatSuggestion))\n    );\n\n  private suggester: {\n    [type in suggestionType]?: suggestFunc;\n  } = {\n    default: (term, property) => of(\n      (property?.suggestions?.values || [])\n        .map(v => `${v}`)\n        .filter(v => v.toLowerCase().includes(term.toLowerCase()))\n        .map(name => ({ name }))\n    ),\n    [NodeType.Actor]: this.suggestNode(NodeType.Actor),\n    [NodeType.Cycle]: this.suggestNode(NodeType.Cycle),\n    [NodeType.ImpactAssessment]: this.suggestNode(NodeType.ImpactAssessment),\n    [NodeType.Organisation]: this.suggestNode(NodeType.Organisation),\n    [NodeType.Site]: this.suggestNode(NodeType.Site),\n    [NodeType.Source]: this.suggestNode(NodeType.Source, (term, { key }) =>\n      term.length < MIN_TYPEAHEAD_LENGTH ?\n        of([]) :\n        this.searchService.suggestSource(term, 5, [`bibliography.${key}`], [`bibliography.${key}`])\n    ),\n    [NodeType.Term]: this.suggestNode(NodeType.Term, (term, { suggestions }) =>\n      term.length < MIN_TYPEAHEAD_LENGTH ?\n        of([]) :\n        this.searchService.suggest(term, NodeType.Term, suggestions?.queries || [], null, 5, populateTermFields)\n    )\n  };\n  public propertySuggest = (fullKey: string, suggestType: suggestionType) => (text$: Observable<string>) =>\n    text$.pipe(\n      debounceTime(suggestType === defaultSuggestionType ? 0 : 300),\n      distinctUntilChanged(),\n      mergeMap(term => {\n        const property = findProperty(this.properties, fullKey);\n        return property?.editable && suggestType in this.suggester ?\n          of(term).pipe(\n            tap(() => property.loading = true),\n            switchMap(() => this.suggester[suggestType]!(term, property)),\n            tap(() => property.loading = false)\n          ) :\n          of([]);\n      }),\n    );\n\n  constructor(\n    public ref: ElementRef,\n    private searchService: HeSearchService,\n    private usersService: HeUsersService\n  ) { }\n\n  ngOnInit() {\n    const schemaType = this.node!.type || this.node!['@type'];\n    const schema = this.schemas ? this.schemas[schemaType] : {} as definition;\n    const errors = this.errors.filter(filterError);\n    const error = propertyError(errors)('');\n    this.nodeProperty = this.schemas ? {\n      id: propertyId(),\n      schema,\n      schemaType,\n      editable: !!this.node!.type,\n      fullKey: '',\n      key: '',\n      value: {},\n      error: formatError(error),\n      hasError: errorHasError(error),\n      hasWarning: errorHasWarning(error),\n      suggestions: {},\n      newProperty: {},\n      newError: { level: 'warning' },\n      properties: [],\n      addPropertyEnabled: isAddPropertyEnabled(this.node!, schemaType, schema)\n    } : undefined;\n    const recurser = recursiveProperties(this.schemas!, errors, schema, this.deepEditable);\n    this.properties = recurser(this.node);\n  }\n\n  public trackByIndex(index: number, _value: any) {\n    return index;\n  }\n\n  public trackByProperty(_index: number, { id, fullKey }: INodeProperty) {\n    return [fullKey, id].join('_');\n  }\n\n  public hasAddons(property: INodeProperty) {\n    return property && (\n      this.enableAddError(property) ||\n      (this.editable && property.editable) ||\n      this.addPropertyEnabled(property) ||\n      this.isRequired(property) ||\n      property.externalUrl?.url ||\n      this.mapDrawingModes(property).length ||\n      this.unitConverterEnabled(property)\n    );\n  }\n\n  public isRequired({ value, isRequired }: INodeProperty) {\n    return isRequired && isEmpty(value);\n  }\n\n  // --- Errors\n\n  public enableAddError(property: INodeProperty) {\n    return this.errorsEditable && !property.hasError && !property.hasWarning;\n  }\n\n  public 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      message,\n      level: level || 'error',\n      dataPath: keyToDataPath(keys.join('.')),\n      params: {\n        missingProperty: property.key\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  public editError(property: INodeProperty) {\n    property.newError = property.error;\n  }\n\n  public 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  // ---\n\n  public typeaheadFocus(e: Event) {\n    if (e.bubbles) {\n      return;\n    }\n    e.stopPropagation();\n    setTimeout(() => {\n      const inputEvent = new Event('input');\n      e.target?.dispatchEvent(inputEvent);\n    }, 0);\n  }\n\n  public addPropertyEnabled(property: INodeProperty) {\n    return this.editable && !this.errorsEditable && property.addPropertyEnabled;\n  }\n\n  private newProperties(property: INodeProperty, field: string) {\n    const properties = nodeAvailableProperties(this.node!, property);\n    return Object.keys(properties)\n      .filter(prop => prop.toLowerCase().includes(field.toLowerCase()))\n      .map(name => ({ name, prop: properties[name] }));\n  }\n\n  private replacePropertyData(property: INodeProperty, data: any, editable?: boolean, replaceParent = true) {\n    const parent = replaceParent ? parentProperty(this.properties, property) : property;\n    const properties = recursiveProperties(\n      this.schemas!, [], this.schemas![this.node!.type], this.deepEditable, parent.fullKey\n    )(data);\n    updateProperties({ properties }, prop => {\n      prop.editable = typeof editable === 'undefined' ? prop.editable : editable;\n      return prop;\n    });\n    parent.editable = typeof editable === 'undefined' ? parent.editable : editable;\n    parent.properties = properties;\n    parent.value = data;\n    return this.propertyChanged(data, parent);\n  }\n\n  public suggestionSelected({ '@id': _id, name, type, id, ...data }, property: INodeProperty) {\n    return _id ?\n      // suggestions from existing nodes\n      this.replacePropertyData(property, { '@id': _id, '@type': property.suggestions?.type, name, ...data }, false) :\n      (\n        // suggestion from linked nodes\n        type && id ?\n          this.replacePropertyData(property, { type, id, ...data }, true) :\n          // suggestion by name => enum values\n          this.propertyChanged(name, property)\n      );\n  }\n\n  public addArrayGroup(array: INodeProperty) {\n    const prop = this.schemas![array.schemaType];\n    const value = get(this.node, array.fullKey);\n    const newProperty = typeToNewProperty(\n      this.schemas!, prop, prop, value, `${array.fullKey}[${array.properties.length}]`\n    );\n    array.properties.push(newProperty);\n    this.nodeChange.emit({ key: array.fullKey, value });\n    return groupChanged(this.properties, array.fullKey);\n  }\n\n  public duplicateArrayGroup(array: INodeProperty, property: INodeProperty) {\n    const index = +property.key;\n    const newProperty: INodeProperty = JSON.parse(JSON.stringify(property));\n    newProperty.id = propertyId();\n    const value = [...get(this.node, array.fullKey)];\n    value.splice(index, 0, safeJSONParse(newProperty.value));\n    array.properties.splice(index, 0, newProperty);\n    // need to change key on every properties recursively\n    refreshPropertyKeys(array);\n    this.nodeChange.emit({ key: array.fullKey, value });\n    return groupChanged(this.properties, array.fullKey);\n  }\n\n  public removeArrayGroup(array: INodeProperty, { key }: INodeProperty) {\n    const index = +key;\n    const value = [...get(this.node, array.fullKey)];\n    value.splice(index, 1);\n    array.properties.splice(index, 1);\n    // need to change key on every properties recursively\n    refreshPropertyKeys(array);\n    this.nodeChange.emit({ key: array.fullKey, value });\n    // if no more values, remove parent group entirely\n    return array.properties.length ? null : this.propertyChanged(null, array);\n  }\n\n  private moveArrayGroupToPosition(array: INodeProperty, currentIndex: number, newIndex: number) {\n    const value = [...get(this.node, array.fullKey)];\n    moveItemInArray(value, currentIndex, newIndex);\n    moveItemInArray(array.properties, currentIndex, newIndex);\n    // need to change key on every properties recursively\n    refreshPropertyKeys(array);\n    this.nodeChange.emit({ key: array.fullKey, value });\n    return groupChanged(this.properties, array.fullKey);\n  }\n\n  public moveArrayGroupUp(array: INodeProperty, { key }: INodeProperty) {\n    const index = +key;\n    return this.moveArrayGroupToPosition(array, index, index - 1);\n  }\n\n  public moveArrayGroupDown(array: INodeProperty, { key }: INodeProperty) {\n    const index = +key;\n    return this.moveArrayGroupToPosition(array, index, index + 1);\n  }\n\n  public propertyChanged(value: any, property: INodeProperty) {\n    groupChanged(this.properties, property.fullKey, value);\n    property.addPropertyEnabled = isAddPropertyEnabled(\n      this.node!, property.schemaType, property.schema!, property.fullKey\n    );\n\n    if (value === null) {\n      value = undefined;\n    }\n    else {\n      property.value = value;\n      property.changed = true;\n    }\n\n    this.nodeChange.emit({ key: property.fullKey, value: parseNewValue(property.schema!, value) });\n  }\n\n  private addDefaultProperty(property: INodeProperty, key: string, def: definition) {\n    const node = property.key ? get(this.node, property.fullKey) : this.node;\n    const fullKey = property.key ? `${property.fullKey}.${key}` : key;\n    const properties = property.key ? property.properties : this.properties;\n    const newProperty = typeToNewProperty(this.schemas!, def, this.schemas![property.schemaType], node, fullKey);\n    const existingPropertyIndex = properties.findIndex(v => v.fullKey === fullKey);\n    existingPropertyIndex >= 0 ? (properties[existingPropertyIndex] = newProperty) : properties.push(newProperty);\n    properties.sort(sortProperties);\n    return this.propertyChanged(newProperty.value, newProperty);\n  }\n\n  public addMissingProperty(property: INodeProperty) {\n    const parent = parentProperty(this.properties, property);\n    const newProperty = parent === property ? this.nodeProperty! : parent;\n    const newProperties = this.newProperties(newProperty, property.key);\n    newProperty.newProperty = newProperties[0];\n    return newProperty.newProperty ? this.addProperty(newProperty) : null;\n  }\n\n  public addProperty(property: INodeProperty) {\n    const { name, prop } = property.newProperty!;\n    property.newProperty = {};\n    return name === '@id' ?\n      this.replacePropertyData(property, { '@id': '', type: property.schemaType }, undefined, false) :\n      this.addDefaultProperty(property, name!, prop!);\n  }\n\n  private updatePropertyCoordinates(property: INodeProperty, value: google.maps.LatLngLiteral) {\n    const latitude = siblingProperty(this.properties, property, 'latitude');\n    const longitude = siblingProperty(this.properties, property, 'longitude');\n    return [\n      latitude ? this.propertyChanged(value.lat, latitude) : null,\n      longitude ? this.propertyChanged(value.lng, longitude) : null\n    ];\n  }\n\n  // Map\n\n  public get showMap() {\n    return (!this.editable || this.errorMode) && [SchemaType.Site, SchemaType.Organisation].includes(this.node!.type);\n  }\n\n  public mapDrawingModes({ key }: INodeProperty) {\n    return ['boundary'].includes(key) ? [\n      google.maps.drawing.OverlayType.POLYGON\n    ] : ([\n      'latitude', 'longitude'\n    ].includes(key) ? [\n      google.maps.drawing.OverlayType.MARKER\n    ] : []);\n  }\n\n  public onMapDrawingClosed(newValue) {\n    const property = this.mapDrawingProperty;\n    this.mapDrawingProperty = undefined;\n    return ['latitude', 'longitude'].includes(property!.key) ?\n      this.updatePropertyCoordinates(property!, newValue) :\n      this.propertyChanged(newValue, property!);\n  }\n\n  // Bibliography Search\n\n  public bibliographiesSearchKey({ key, schemaType }: INodeProperty) {\n    const validSchema = [SchemaType.Bibliography].includes(schemaType);\n    const valieKey = ['title', 'documentDOI', 'scopus'].includes(key);\n    return validSchema && valieKey ? key : null;\n  }\n\n  private updateBibliography(value: Partial<Bibliography>) {\n    const parent = parentProperty(this.properties, this.bibliographiesSearchProperty!);\n    const newValue = this.bibliographiesSearchSources ? {\n      ...parent.value,\n      bibliography: value\n    } : value;\n    this.replacePropertyData(this.bibliographiesSearchProperty!, newValue, true);\n    return undefined;\n  }\n\n  private updateSource(value: Partial<Source>) {\n    // cannot replace a top-level source\n    this.bibliographiesSearchSources && this.replacePropertyData(this.bibliographiesSearchProperty!, {\n      '@type': NodeType.Source,\n      ...value\n    }, !('@id' in value));\n    return undefined;\n  }\n\n  public get bibliographiesSearchSources() {\n    return this.bibliographiesSearchProperty!.fullKey.split('.').length > 2;\n  }\n\n  public onBibliographiesSearchClosed(value?: Partial<Source> | Partial<Bibliography>) {\n    this.bibliographiesSearchProperty = value ? (\n      value.type === SchemaType.Bibliography ?\n        this.updateBibliography(value) :\n        this.updateSource(value as Partial<Source>)\n    ) : undefined;\n  }\n\n  // Actor.@id\n\n  public async setUserActorId(property: INodeProperty) {\n    const { actorId, name } = await this.usersService.loggedInUser;\n    return this.replacePropertyData(property, { '@id': actorId, '@type': NodeType.Actor, name }, false);\n  }\n\n  // Cycle.cycleDuration\n\n  public calculateCycleDurationEnabled(property: INodeProperty) {\n    return calculateCycleDurationEnabled(this.properties, property);\n  }\n\n  public calculateCycleDuration(property: INodeProperty) {\n    const duration = calculateCycleDuration(this.properties, property);\n    return this.propertyChanged(duration, property);\n  }\n\n  // Cycle.startDate\n\n  public calculateCycleStartDateEnabled(property: INodeProperty) {\n    return calculateCycleStartDateEnabled(this.properties, property);\n  }\n\n  public calculateCycleStartDate(property: INodeProperty) {\n    const startDate = calculateCycleStartDate(this.properties, property);\n    return this.propertyChanged(startDate, property);\n  }\n\n  // Units converter\n\n  public unitConverterEnabled(property: INodeProperty) {\n    return this.editable && property.editable && ['value', 'min', 'max', 'sd'].includes(property.key);\n  }\n\n  public openUnitConverter(popover, property: INodeProperty) {\n    const value = property.value;\n    const termProperty = siblingProperty(this.properties, property, 'term');\n    const term = safeJSONParse(termProperty?.value);\n    const units = term?.units;\n    return popover.isOpen() ? popover.close() : popover.open({ value, term, units });\n  }\n}\n","<div class=\"card\">\n  <div class=\"card-toggle p-4\" (click)=\"isOpen = !isOpen\" pointer>\n    <fa-icon icon=\"angle-down\" [class.is-hidden]=\"!isOpen\"></fa-icon>\n    <fa-icon icon=\"angle-right\" [class.is-hidden]=\"isOpen\"></fa-icon>\n    <span *ngIf=\"nodeProperty\" class=\"is-px-2\"\n      [class.has-text-danger]=\"nodeProperty.hasError\"\n      [class.has-text-warning]=\"nodeProperty.hasWarning\"\n    >\n      <he-node-icon [type]=\"nodeProperty.schemaType\"></he-node-icon>\n    </span>\n  </div>\n\n  <ng-container *ngIf=\"editable && isOpen && nodeProperty\">\n    <ng-container *ngTemplateOutlet=\"showNewProperty; context: {$implicit: nodeProperty}\"></ng-container>\n  </ng-container>\n\n  <div class=\"card-content\">\n    <ng-container *ngIf=\"isOpen\">\n      <div class=\"pb-3 mb-2\" *ngIf=\"errorsEditable\">\n        <ng-container *ngTemplateOutlet=\"nodeErrorForm; context: {$implicit: nodeProperty}\"></ng-container>\n      </div>\n\n      <div class=\"mb-4\" *ngIf=\"nodeProperty?.error\">\n        <ng-container *ngTemplateOutlet=\"propertyError; context: {property: nodeProperty, edit: true}\"></ng-container>\n      </div>\n    </ng-container>\n\n    <div class=\"columns is-multiline\">\n      <ng-container *ngFor=\"let property of properties; trackBy: trackByProperty\">\n        <ng-container *ngTemplateOutlet=\"showProperty; context: {$implicit: property}\"></ng-container>\n      </ng-container>\n    </div>\n\n    <ng-container *ngIf=\"isOpen\">\n      <ng-container *ngTemplateOutlet=\"propertyMap; context: {$implicit: nodeProperty}\"></ng-container>\n    </ng-container>\n  </div>\n</div>\n\n<he-maps-drawing-confirm *ngIf=\"!!mapDrawingProperty\"\n  [value]=\"mapDrawingProperty.value\" [modes]=\"mapDrawingModes(mapDrawingProperty)\"\n  (closed)=\"onMapDrawingClosed($event)\"\n></he-maps-drawing-confirm>\n\n<he-bibliographies-search-confirm *ngIf=\"!!bibliographiesSearchProperty\"\n  [search]=\"bibliographiesSearchProperty.value\"\n  [searchBy]=\"bibliographiesSearchKey(bibliographiesSearchProperty)\"\n  [searchSources]=\"bibliographiesSearchSources\"\n  (closed)=\"onBibliographiesSearchClosed($event)\"\n></he-bibliographies-search-confirm>\n\n<ng-template #labelDescription let-property>\n  <span\n    class=\"trigger-popover\"\n    [ngbPopover]=\"property.schema?.description\" [autoClose]=\"'outside'\"\n    triggers=\"hover\" placement=\"right\" container=\"body\"\n  >\n    <span>{{property.key}}</span>\n  </span>\n</ng-template>\n\n<ng-template #labelDefault let-property>\n  <span>{{property.key}}</span>\n</ng-template>\n\n<ng-template #showProperty let-property>\n  <ng-container *ngIf=\"(isOpen || property.closedVisible) && !property.isHidden\">\n    <div class=\"column is-6\"\n      [id]=\"property.fullKey + '_' + property.id\"\n      [class.is-12]=\"property.properties.length || !property.key\"\n      [ngSwitch]=\"!!property.properties.length\"\n    >\n      <div class=\"columns is-multiline is-variable is-1\" *ngSwitchCase=\"false\">\n        <div class=\"column is-3 py-1\" *ngIf=\"property.key\">\n          <label class=\"label has-text-right-tablet has-text-ellipsis\"\n            *bindOnce=\"property\"\n            [for]=\"property.id\"\n          >\n            <ng-container\n              *ngTemplateOutlet=\"property.schema?.description && editable && property.editable ? labelDescription : labelDefault; context: {$implicit: property}\">\n            </ng-container>\n          </label>\n        </div>\n\n        <div class=\"column is-9 py-1\" [class.is-12]=\"!property.key\">\n          <div class=\"field\">\n            <ng-container *ngTemplateOutlet=\"inputForm; context: {$implicit: property}\"></ng-container>\n\n            <ng-container *ngIf=\"(editable || errorsEditable) && (property.hasError || property.hasWarning)\">\n              <ng-container *ngTemplateOutlet=\"propertyError; context: {property: property, edit: false}\"></ng-container>\n            </ng-container>\n          </div>\n        </div>\n      </div>\n\n      <div class=\"panel\" *ngSwitchCase=\"true\"\n        [class.is-link]=\"errorMode && property.changed\"\n        [class.is-danger]=\"!property.changed && property.hasError\"\n        [class.is-warning]=\"!property.changed && property.hasWarning\"\n        [class.is-default]=\"!property.changed && !property.hasError && !property.hasWarning\"\n      >\n        <div class=\"open-group panel-heading py-0\"\n          (click)=\"property.isOpen = !property.isOpen\" pointer\n          [class.is-open]=\"property.isOpen\"\n          [class.has-text-white]=\"(errorMode && property.changed) || property.hasError || property.hasWarning\"\n        >\n          <div class=\"columns is-mobile is-vcentered\">\n            <span class=\"column is-narrow py-1 my-0\">\n              <fa-icon icon=\"angle-down\" [class.is-hidden]=\"!property.isOpen\"></fa-icon>\n              <fa-icon icon=\"angle-right\" [class.is-hidden]=\"property.isOpen\"></fa-icon>\n            </span>\n            <span class=\"column py-1 my-0\">\n              <span class=\"is-size-6\">{{property.key}}</span>\n            </span>\n            <span *ngIf=\"property.schemaType\" class=\"column is-narrow py-1 my-0\">\n              <span class=\"tags mb-0 has-addons\">\n                <span class=\"tag mb-0 is-light\">Type</span>\n                <span class=\"tag mb-0 is-white\">\n                  <he-schema-version-link linkClass=\"is-small\" [node]=\"{'@type': property.schemaType}\">\n                    <span>{{property.schemaType}}</span>\n                  </he-schema-version-link>\n                </span>\n              </span>\n            </span>\n            <he-popover-confirm class=\"column is-narrow py-1 my-0 px-0\"\n              *ngIf=\"editable && !errorsEditable\"\n              ngbTooltip=\"Remove group\" placement=\"top\"\n              [message]=\"'<p>This will remove the group completely.</p>' + (property.isRequired ? '<p><u>Warning: this field is required.</u></p>' : '') + '<p>Do you confirm?</p>'\"\n              popoverClass=\"px-3\"\n              (confirmed)=\"propertyChanged(null, property)\"\n            >\n              <fa-icon icon=\"times\" size=\"sm\"></fa-icon>\n            </he-popover-confirm>\n          </div>\n        </div>\n        <div class=\"panel-block is-block p-0\" *ngIf=\"property.isOpen\">\n          <ng-container [ngSwitch]=\"property.isArray\">\n            <div class=\"px-3 pt-4\" *ngIf=\"property.error\">\n              <ng-container *ngTemplateOutlet=\"propertyError; context: {property: property, edit: true}\"></ng-container>\n            </div>\n\n            <div class=\"mt-3\" *ngSwitchCase=\"false\">\n              <p class=\"help py-1 px-2\" *ngIf=\"editable && !property.editable\">\n                To change the {{property.key}}, please delete it first, then add the field again\n              </p>\n              <ng-container *ngIf=\"editable && property.editable\">\n                <ng-container [ngSwitch]=\"property.schema?.title\">\n                  <p class=\"help py-1 px-2\" *ngSwitchCase=\"'Bibliography'\">\n                    Search by Title or Document DOI to auto-populate the fields using the Mendeley catalogue\n                  </p>\n                </ng-container>\n              </ng-container>\n\n              <ng-container *ngTemplateOutlet=\"showNewProperty; context: {$implicit: property}\"></ng-container>\n\n              <div class=\"px-3 mt-1\" *ngIf=\"errorsEditable\">\n                <ng-container *ngTemplateOutlet=\"nodeErrorForm; context: {$implicit: property}\"></ng-container>\n              </div>\n\n              <div class=\"property-group py-2 px-3 mt-2\">\n                <div class=\"columns is-multiline mb-0\">\n                  <ng-container *ngFor=\"let prop2 of property.properties; trackBy: trackByProperty\">\n                    <ng-container *ngTemplateOutlet=\"showProperty; context: {$implicit: prop2}\"></ng-container>\n                  </ng-container>\n                </div>\n              </div>\n            </div>\n\n            <div class=\"py-2 px-3 mt-2\" *ngSwitchCase=\"true\">\n              <div class=\"mt-1\" *ngIf=\"errorsEditable\">\n                <ng-container *ngTemplateOutlet=\"nodeErrorForm; context: {$implicit: property}\"></ng-container>\n              </div>\n\n              <ng-container *ngFor=\"let prop2 of property.properties; trackBy: trackByProperty\">\n                <div class=\"card p-0 my-4\" *ngIf=\"prop2.key\"\n                  [id]=\"prop2.fullKey + '_' + prop2.id\"\n                >\n                  <div class=\"property-array-number\">\n                    <div class=\"tags has-addons\">\n                      <span class=\"tag is-dark\">{{prop2.key}}</span>\n                      <ng-container *ngIf=\"editable && !errorsEditable && property.editable\">\n                        <span class=\"tag is-info\" pointer\n                          (click)=\"duplicateArrayGroup(property, prop2)\"\n                          [ngbTooltip]=\"'Duplicate ' + pluralize(property.key, 1)\" placement=\"top\"\n                        >\n                          <fa-icon icon=\"clone\" size=\"sm\"></fa-icon>\n                        </span>\n                        <span class=\"tag is-light\" pointer\n                          *ngIf=\"prop2.key !== '0'\"\n                          (click)=\"moveArrayGroupUp(property, prop2)\"\n                          ngbTooltip=\"Move Up\" placement=\"top\"\n                        >\n                          <fa-icon icon=\"long-arrow-alt-up\" size=\"sm\"></fa-icon>\n                        </span>\n                        <span class=\"tag is-light\" pointer\n                          *ngIf=\"prop2.key !== property.properties.length - 1\"\n                          (click)=\"moveArrayGroupDown(property, prop2)\"\n                          ngbTooltip=\"Move Down\" placement=\"top\"\n                        >\n                          <fa-icon icon=\"long-arrow-alt-down\" size=\"sm\"></fa-icon>\n                        </span>\n                        <he-popover-confirm class=\"tag is-delete\"\n                          [ngbTooltip]=\"'Remove ' + pluralize(property.key, 1)\" placement=\"top\"\n                          message=\"This will remove the group completely. Do you confirm?\" position=\"right\"\n                          (confirmed)=\"removeArrayGroup(property, prop2)\"\n                        ></he-popover-confirm>\n                      </ng-container>\n                    </div>\n                  </div>\n\n                  <ng-container *ngTemplateOutlet=\"showNewProperty; context: {$implicit: prop2}\"></ng-container>\n\n                  <div class=\"px-4 mt-2\" *ngIf=\"errorsEditable\">\n                    <ng-container *ngTemplateOutlet=\"nodeErrorForm; context: {$implicit: prop2}\"></ng-container>\n                  </div>\n\n                  <div class=\"px-4 mt-2\" *ngIf=\"prop2.error\">\n                    <ng-container *ngTemplateOutlet=\"propertyError; context: {property: prop2, edit: true}\"></ng-container>\n                  </div>\n\n                  <div class=\"property-group card-content p-3\">\n                    <div class=\"columns is-multiline my-0\">\n                      <ng-container *ngFor=\"let prop3 of prop2.properties; trackBy: trackByProperty\">\n                        <ng-container *ngTemplateOutlet=\"showProperty; context: {$implicit: prop3}\"></ng-container>\n                      </ng-container>\n                    </div>\n                  </div>\n                </div>\n              </ng-container>\n\n              <button class=\"button is-dark is-outlined is-small\" type=\"button\"\n                *ngIf=\"editable && !errorsEditable\"\n                (click)=\"addArrayGroup(property)\"\n              >\n                <fa-icon icon=\"plus-circle\"></fa-icon>\n                <span class=\"pl-2\" i18n=\"@@add\">Add</span>\n                <span class=\"pl-1\">{{property.key | pluralize:1}}</span>\n              </button>\n            </div>\n          </ng-container>\n        </div>\n      </div>\n    </div>\n  </ng-container>\n</ng-template>\n\n<ng-template #inputForm let-property>\n  <ng-container *ngIf=\"property.key\">\n    <ng-container [ngSwitch]=\"property.suggestions?.type\">\n      <ng-container *ngSwitchCase=\"'select'\">\n        <ng-container *ngTemplateOutlet=\"inputSelect; context: {$implicit: property}\"></ng-container>\n      </ng-container>\n      <ng-container *ngSwitchDefault>\n        <ng-container *ngTemplateOutlet=\"inputInput; context: {$implicit: property}\"></ng-container>\n      </ng-container>\n    </ng-container>\n  </ng-container>\n</ng-template>\n\n<ng-template #inputInput let-property>\n  <div class=\"field mb-0\" [class.has-addons]=\"hasAddons(property)\">\n    <div class=\"control is-expanded\"\n      [class.has-icons-right]=\"property.loading\"\n    >\n      <input class=\"input is-small search-input\"\n        [class.is-dark]=\"property.key === 'type'\"\n        [class.is-link]=\"errorMode && property.changed\"\n        [class.is-danger]=\"(!property.changed && property.hasError) || isRequired(property)\"\n        [class.is-warning]=\"!property.changed && property.hasWarning\"\n\n        [(ngModel)]=\"property.value\" #propertyModel=\"ngModel\"\n        [type]=\"property.schema?.type === 'number' ? 'number' : 'string'\"\n        [id]=\"property.id\"\n        name=\"randomname\"\n        [readonly]=\"!editable || !property.editable || property.schema?.internal\"\n        [placeholder]=\"property.placeholder\"\n        [appTagsInput]=\"{enabled: editable && property.editable && property.schema?.type === 'array', items: property.schema?.items, delimiter: ';', allowDuplicates: true, placeholder: property.placeholder}\"\n        (change)=\"propertyChanged($event.target.value, property)\"\n\n        [pattern]=\"property.schema?.pattern\"\n        [required]=\"property.fullKey.endsWith('id') && property.fullKey !== 'id'\"\n        [min]=\"property.schema?.minimum\"\n        [max]=\"property.schema?.maximum\"\n\n        [ngbTypeahead]=\"propertySuggest(property.fullKey, property.suggestions?.type)\"\n        [resultTemplate]=\"suggestion\"\n        [inputFormatter]=\"formatter\"\n        [focusFirst]=\"true\"\n        (focus)=\"editable && property.editable && typeaheadFocus($event)\"\n        (selectItem)=\"suggestionSelected($event.item, property)\"\n      >\n\n      <span class=\"icon is-small is-right has-text-grey-dark\" [class.is-hidden]=\"!property.loading\">\n        <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"sm\"></fa-icon>\n      </span>\n    </div>\n    <ng-container *ngTemplateOutlet=\"inputAddons; context: {$implicit: property}\"></ng-container>\n  </div>\n  <p class=\"help is-danger-light\"\n    *ngIf=\"!property.hasError && propertyModel.invalid\"\n  >\n    <span *bindOnce=\"propertyModel.errors\" [innerHTML]=\"formatPropertyError(propertyModel.errors, property)\"></span>\n  </p>\n</ng-template>\n\n<ng-template #inputSelect let-property>\n  <div class=\"field mb-0\" [class.has-addons]=\"hasAddons(property)\">\n    <div class=\"control is-expanded\">\n      <div class=\"select is-small is-fullwidth\"\n        [class.is-link]=\"errorMode && property.changed\"\n        [class.is-danger]=\"(!property.changed && property.hasError) || isRequired(property)\"\n        [class.is-warning]=\"!property.changed && property.hasWarning\"\n      >\n        <select\n          [(ngModel)]=\"property.value\" #propertyModel=\"ngModel\"\n          [id]=\"property.id\"\n          name=\"randomname\"\n          [disabled]=\"!editable || !property.editable || property.schema?.internal\"\n          (change)=\"propertyChanged($event.target.value, property)\"\n        >\n          <option value=\"\">Select</option>\n          <ng-container *bindOnce=\"property.suggestions\">\n            <option *ngFor=\"let value of property.suggestions.values; trackBy: trackByIndex\" [value]=\"value\">{{value}}</option>\n          </ng-container>\n        </select>\n      </div>\n    </div>\n    <ng-container *ngTemplateOutlet=\"inputAddons; context: {$implicit: property}\"></ng-container>\n  </div>\n</ng-template>\n\n<ng-template #removeFieldAddon let-property>\n  <div class=\"control\" *ngIf=\"!errorsEditable && !property.isRequired\">\n    <a class=\"button is-small\" title=\"Remove field\"\n      [class.is-outlined]=\"!property.changed && (property.hasError || property.hasWarning)\"\n      [class.is-danger]=\"!property.changed && property.hasError\"\n      [class.is-warning]=\"!property.changed && property.hasWarning\"\n      (click)=\"$event.stopPropagation(); propertyChanged(null, property)\"\n    >\n      <fa-icon icon=\"times\"></fa-icon>\n    </a>\n  </div>\n</ng-template>\n\n<ng-template #inputAddons let-property>\n  <he-popover-confirm class=\"control\"\n    *ngIf=\"enableAddError(property)\"\n    position=\"left\"\n    [content]=\"popupErrorForm\"\n    (confirmed)=\"addError(property, $event)\"\n  >\n    <span class=\"button is-small\">\n      <fa-icon icon=\"comments\"></fa-icon>\n    </span>\n  </he-popover-confirm>\n\n  <ng-container *ngIf=\"editable && property.editable\">\n    <ng-container [ngSwitch]=\"property.schemaType\">\n      <ng-container *ngSwitchCase=\"SchemaType.Actor\">\n        <ng-container *ngTemplateOutlet=\"actorAddons; context: {$implicit: property}\"></ng-container>\n      </ng-container>\n      <ng-container *ngSwitchCase=\"SchemaType.Cycle\">\n        <ng-container *ngTemplateOutlet=\"cycleAddons; context: {$implicit: property}\"></ng-container>\n      </ng-container>\n    </ng-container>\n    <ng-container *bindOnce=\"property\">\n      <div class=\"control\" *ngIf=\"mapDrawingModes(property).length\">\n        <button class=\"button is-small\" title=\"Pick on Map\"\n          (click)=\"mapDrawingProperty = property\"\n        >\n          <fa-icon icon=\"map-marked-alt\"></fa-icon>\n        </button>\n      </div>\n      <div class=\"control\" *ngIf=\"bibliographiesSearchKey(property)\">\n        <button class=\"button is-small\" title=\"Advanced Search\"\n          (click)=\"bibliographiesSearchProperty = property\"\n        >\n          <fa-icon icon=\"search\"></fa-icon>\n        </button>\n      </div>\n    </ng-container>\n\n    <ng-container *ngTemplateOutlet=\"removeFieldAddon; context: {$implicit: property}\"></ng-container>\n  </ng-container>\n  <div class=\"control\" *ngIf=\"addPropertyEnabled(property)\">\n    <a class=\"button is-small is-danger\" title=\"Add field\"\n      (click)=\"$event.stopPropagation(); addMissingProperty(property)\"\n    >\n      <fa-icon icon=\"plus-circle\"></fa-icon>\n      <span class=\"pl-2\" i18n=\"@@add\">Add</span>\n    </a>\n  </div>\n  <div class=\"control\" *ngIf=\"isRequired(property)\">\n    <label class=\"button is-small is-danger\" [for]=\"property.id\"\n      ngbTooltip=\"This field is required\" placement=\"top\"\n    >\n      <fa-icon icon=\"exclamation-triangle\"></fa-icon>\n    </label>\n  </div>\n  <ng-container *ngIf=\"property.externalUrl?.url\">\n    <div class=\"control\">\n      <a class=\"button is-small\"\n        [href]=\"property.externalUrl.url + (property.externalUrl.urlParamValue ? property.value : '')\"\n        target=\"_blank\"\n        [title]=\"property.externalUrl.title\"\n        [ngClass]=\"{'is-dark is-outlined': property.key === 'type'}\"\n        [attr.disabled]=\"property.externalUrl.urlParamValue && !property.value ? true : null\"\n      >\n        <fa-icon [icon]=\"property.externalUrl.icon || 'external-link-alt'\"></fa-icon>\n      </a>\n    </div>\n  </ng-container>\n  <ng-container *ngIf=\"unitConverterEnabled(property)\">\n    <div class=\"control\">\n      <button class=\"button is-small\" title=\"Open calculator\"\n        [ngbPopover]=\"convertUnits\" autoClose=\"outside\"\n        triggers=\"manual\" #p=\"ngbPopover\" placement=\"bottom\" container=\"body\"\n        (click)=\"openUnitConverter(p, property)\"\n      >\n        <fa-icon icon=\"calculator\"></fa-icon>\n      </button>\n    </div>\n  </ng-container>\n</ng-template>\n\n<ng-template #actorAddons let-property>\n  <div class=\"control\" *ngIf=\"property.key === '@id'\">\n    <button class=\"button is-small\" title=\"Add myself as Actor\"\n      (click)=\"setUserActorId(property)\"\n    >\n      <fa-icon [icon]=\"['far', 'id-badge']\"></fa-icon>\n    </button>\n  </div>\n</ng-template>\n\n<ng-template #cycleAddons let-property>\n  <div class=\"control\" *ngIf=\"property.key === 'cycleDuration'\">\n    <button class=\"button is-small\" title=\"Calculate value from startDate and endDate\"\n      (click)=\"calculateCycleDuration(property)\"\n      [disabled]=\"!calculateCycleDurationEnabled(property)\"\n    >\n      <fa-icon icon=\"calculator\"></fa-icon>\n    </button>\n  </div>\n  <div class=\"control\" *ngIf=\"property.key === 'startDate'\">\n    <button class=\"button is-small\" title=\"Calculate value from endDate and cycleDuration\"\n      (click)=\"calculateCycleStartDate(property)\"\n      [disabled]=\"!calculateCycleStartDateEnabled(property)\"\n    >\n      <fa-icon icon=\"calculator\"></fa-icon>\n    </button>\n  </div>\n</ng-template>\n\n<ng-template #showNewProperty let-property>\n  <header class=\"card-header\" *ngIf=\"editable && property.editable && (property.addPropertyEnabled || deepEditable); else padder\">\n    <form class=\"py-3 px-4 is-flex-grow-1\" (submit)=\"addProperty(property)\" novalidate>\n      <div class=\"field is-horizontal\">\n        <div class=\"field-label is-small\">\n          <label class=\"label\" [for]=\"property.id + '_new'\">\n            <span i18n=\"@@files.form.newProperty\">Add new field</span>\n          </label>\n        </div>\n        <div class=\"field-body\">\n          <div class=\"field has-addons\">\n            <div class=\"control is-expanded\">\n              <input class=\"input is-small\"\n                [(ngModel)]=\"property.newProperty\"\n                [id]=\"property.id + '_new'\"\n                name=\"randomname\"\n                placeholder=\"Search and select field from results\"\n\n                [ngbTypeahead]=\"suggestNewProperty(property.fullKey)\"\n                [resultTemplate]=\"suggestion\"\n                [inputFormatter]=\"formatter\"\n                [focusFirst]=\"false\"\n                [editable]=\"false\"\n                (focus)=\"typeaheadFocus($event)\"\n              >\n            </div>\n            <div class=\"control\">\n              <button class=\"button is-small\" type=\"submit\"\n                [disabled]=\"!property.newProperty || !property.newProperty.name\"\n              >\n                <fa-icon icon=\"plus\"></fa-icon>\n                <span class=\"pl-2\" i18n=\"@@add\">Add</span>\n              </button>\n            </div>\n          </div>\n        </div>\n      </div>\n    </form>\n  </header>\n</ng-template>\n\n<ng-template #propertyError let-property=\"property\" let-edit=\"edit\">\n  <p class=\"help\"\n    [class.is-danger]=\"property.hasError\"\n    [class.is-warning]=\"property.hasWarning\"\n    *ngIf=\"property.error\"\n  >\n    <span class=\"is-pre-wrap\" *bindOnce=\"property.error\" [innerHTML]=\"property.error.message\"></span>\n    <a class=\"pl-2\"\n      *ngIf=\"edit && errorsEditable && property.error.index >= 0\"\n      (click)=\"editError(property)\"\n    >\n      <fa-icon class=\"pr-2\" icon=\"edit\"></fa-icon>\n      <span i18n=\"@@edit\">Edit</span>\n    </a>\n    <a class=\"pl-2\"\n      *ngIf=\"(property.hasWarning || errorsEditable) && property.error.index >= 0\"\n      (click)=\"resolveError(property)\"\n    >\n      <fa-icon class=\"pr-2\" icon=\"check\"></fa-icon>\n      <span i18n=\"@@resolved\">Resolved</span>\n    </a>\n  </p>\n</ng-template>\n\n<ng-template #propertyMap let-property>\n  <div class=\"panel is-default\" *ngIf=\"showMap\">\n    <div class=\"open-group panel-heading py-0\"\n      (click)=\"mapVisible = !mapVisible\" pointer\n      [class.is-open]=\"mapVisible\"\n    >\n      <div class=\"columns is-mobile is-vcentered mb-0\">\n        <span class=\"column is-narrow py-1 my-0\">\n          <fa-icon icon=\"angle-down\" [class.is-hidden]=\"!mapVisible\"></fa-icon>\n          <fa-icon icon=\"angle-right\" [class.is-hidden]=\"mapVisible\"></fa-icon>\n        </span>\n        <span class=\"column py-1 my-0\">\n          <span class=\"is-size-6\">View on Map</span>\n        </span>\n      </div>\n    </div>\n    <div class=\"panel-block is-block p-0\" [class.is-hidden]=\"!mapVisible\">\n      <he-sites-maps [sites]=\"[node]\" [showNotice]=\"false\"></he-sites-maps>\n    </div>\n  </div>\n</ng-template>\n\n<ng-template #nodeErrorForm let-property>\n  <ng-container *ngIf=\"property.newError && property.editable\">\n    <div class=\"field has-addons\">\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 class=\"textarea is-small\"\n          [(ngModel)]=\"property.newError.message\"\n          placeholder=\"Enter your message here\"\n          rows=\"1\"\n        ></textarea>\n      </div>\n      <div class=\"control\">\n        <button class=\"button is-small\"\n          [disabled]=\"!property.newError.level || !property.newError.message\"\n          (click)=\"addError(property, property.newError)\"\n        >\n          <fa-icon icon=\"plus-circle\"></fa-icon>\n          <span class=\"pl-2\" i18n=\"@@add\">Add</span>\n        </button>\n      </div>\n    </div>\n  </ng-container>\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\"\n        [(ngModel)]=\"data.message\"\n        placeholder=\"Enter your message here\"\n      >\n    </div>\n  </div>\n</ng-template>\n\n<ng-template #suggestion let-r=\"result\" let-t=\"term\">\n  <ngb-highlight\n    [title]=\"r.bibliography?.title || r.bibliography?.documentDOI || r.bibliography?.scopus || r.name\"\n    [result]=\"r.bibliography?.title || r.bibliography?.documentDOI || r.bibliography?.scopus || r.name\"\n    [term]=\"t\"\n  ></ngb-highlight>\n</ng-template>\n\n<ng-template #padder>\n  <div class=\"pt-1\"></div>\n</ng-template>\n\n<ng-template #convertUnits let-value=\"value\" let-term=\"term\" let-units=\"units\">\n  <he-unit-converter [value]=\"value\" [term]=\"term\" [toUnits]=\"units\"></he-unit-converter>\n</ng-template>\n"]}
|