@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.
Files changed (207) hide show
  1. package/README.md +24 -0
  2. package/auth/auth.service.d.ts +10 -0
  3. package/auth/public-api.d.ts +1 -0
  4. package/bibliographies/bibliographies-search-confirm/bibliographies-search-confirm.component.d.ts +40 -0
  5. package/bibliographies/bibliographies.module.d.ts +10 -0
  6. package/bibliographies/public-api.d.ts +2 -0
  7. package/bundles/hestia-earth-ui-components.umd.js +7570 -0
  8. package/bundles/hestia-earth-ui-components.umd.js.map +1 -0
  9. package/common/bind-once.directive.d.ts +10 -0
  10. package/common/blank-node-diffs/blank-node-diffs.component.d.ts +18 -0
  11. package/common/blank-node-state/blank-node-state.component.d.ts +36 -0
  12. package/common/blank-node-state-notice/blank-node-state-notice.component.d.ts +9 -0
  13. package/common/blank-node-value-delta/blank-node-value-delta.component.d.ts +13 -0
  14. package/common/click-outside.directive.d.ts +10 -0
  15. package/common/clipboard/clipboard.component.d.ts +16 -0
  16. package/common/common.module.d.ts +37 -0
  17. package/common/common.service.d.ts +11 -0
  18. package/common/default.pipe.d.ts +7 -0
  19. package/common/ellipsis.pipe.d.ts +7 -0
  20. package/common/get.pipe.d.ts +7 -0
  21. package/common/key-to-label.pipe.d.ts +7 -0
  22. package/common/keys.pipe.d.ts +7 -0
  23. package/common/link-key-value/link-key-value.component.d.ts +16 -0
  24. package/common/logs-utils.d.ts +17 -0
  25. package/common/maps-drawing-confirm/maps-drawing-confirm.component.d.ts +28 -0
  26. package/common/maps-utils.d.ts +70 -0
  27. package/common/pluralize.pipe.d.ts +7 -0
  28. package/common/popover/popover.component.d.ts +16 -0
  29. package/common/popover-confirm/popover-confirm.component.d.ts +15 -0
  30. package/common/precision.pipe.d.ts +7 -0
  31. package/common/public-api.d.ts +30 -0
  32. package/common/schema-version-link/schema-version-link.component.d.ts +10 -0
  33. package/common/skeleton-text/skeleton-text.component.d.ts +11 -0
  34. package/common/social-tags/social-tags.component.d.ts +12 -0
  35. package/common/tags-input.directive.d.ts +20 -0
  36. package/common/times.pipe.d.ts +7 -0
  37. package/common/toast/toast.component.d.ts +19 -0
  38. package/common/toast.service.d.ts +16 -0
  39. package/common/unit-converter/unit-converter.component.d.ts +25 -0
  40. package/common/utils.d.ts +88 -0
  41. package/cycles/cycles-activity/cycles-activity.component.d.ts +31 -0
  42. package/cycles/cycles-activity-logs/cycles-activity-logs.component.d.ts +25 -0
  43. package/cycles/cycles-completeness/cycles-completeness.component.d.ts +14 -0
  44. package/cycles/cycles-emissions/cycles-emissions.component.d.ts +40 -0
  45. package/cycles/cycles-emissions-chart/cycles-emissions-chart.component.d.ts +18 -0
  46. package/cycles/cycles-emissions-logs/cycles-emissions-logs.component.d.ts +23 -0
  47. package/cycles/cycles-functional-unit-measure/cycles-functional-unit-measure.component.d.ts +8 -0
  48. package/cycles/cycles-result/cycles-result.component.d.ts +13 -0
  49. package/cycles/cycles-suggest-form/cycles-suggest-form.component.d.ts +24 -0
  50. package/cycles/cycles.model.d.ts +7 -0
  51. package/cycles/cycles.module.d.ts +19 -0
  52. package/cycles/public-api.d.ts +10 -0
  53. package/engine/aggregation-engine.service.d.ts +71 -0
  54. package/engine/engine.service.d.ts +61 -0
  55. package/engine/public-api.d.ts +2 -0
  56. package/esm2015/auth/auth.service.js +19 -0
  57. package/esm2015/auth/public-api.js +2 -0
  58. package/esm2015/bibliographies/bibliographies-search-confirm/bibliographies-search-confirm.component.js +100 -0
  59. package/esm2015/bibliographies/bibliographies.module.js +30 -0
  60. package/esm2015/bibliographies/public-api.js +3 -0
  61. package/esm2015/common/bind-once.directive.js +23 -0
  62. package/esm2015/common/blank-node-diffs/blank-node-diffs.component.js +47 -0
  63. package/esm2015/common/blank-node-state/blank-node-state.component.js +90 -0
  64. package/esm2015/common/blank-node-state-notice/blank-node-state-notice.component.js +27 -0
  65. package/esm2015/common/blank-node-value-delta/blank-node-value-delta.component.js +41 -0
  66. package/esm2015/common/click-outside.directive.js +28 -0
  67. package/esm2015/common/clipboard/clipboard.component.js +69 -0
  68. package/esm2015/common/common.module.js +144 -0
  69. package/esm2015/common/common.service.js +29 -0
  70. package/esm2015/common/default.pipe.js +23 -0
  71. package/esm2015/common/ellipsis.pipe.js +17 -0
  72. package/esm2015/common/get.pipe.js +17 -0
  73. package/esm2015/common/key-to-label.pipe.js +18 -0
  74. package/esm2015/common/keys.pipe.js +20 -0
  75. package/esm2015/common/link-key-value/link-key-value.component.js +55 -0
  76. package/esm2015/common/logs-utils.js +49 -0
  77. package/esm2015/common/maps-drawing-confirm/maps-drawing-confirm.component.js +89 -0
  78. package/esm2015/common/maps-utils.js +101 -0
  79. package/esm2015/common/pluralize.pipe.js +17 -0
  80. package/esm2015/common/popover/popover.component.js +37 -0
  81. package/esm2015/common/popover-confirm/popover-confirm.component.js +38 -0
  82. package/esm2015/common/precision.pipe.js +20 -0
  83. package/esm2015/common/public-api.js +31 -0
  84. package/esm2015/common/schema-version-link/schema-version-link.component.js +39 -0
  85. package/esm2015/common/skeleton-text/skeleton-text.component.js +44 -0
  86. package/esm2015/common/social-tags/social-tags.component.js +37 -0
  87. package/esm2015/common/tags-input.directive.js +74 -0
  88. package/esm2015/common/times.pipe.js +16 -0
  89. package/esm2015/common/toast/toast.component.js +37 -0
  90. package/esm2015/common/toast.service.js +36 -0
  91. package/esm2015/common/unit-converter/unit-converter.component.js +73 -0
  92. package/esm2015/common/utils.js +245 -0
  93. package/esm2015/cycles/cycles-activity/cycles-activity.component.js +92 -0
  94. package/esm2015/cycles/cycles-activity-logs/cycles-activity-logs.component.js +48 -0
  95. package/esm2015/cycles/cycles-completeness/cycles-completeness.component.js +41 -0
  96. package/esm2015/cycles/cycles-emissions/cycles-emissions.component.js +121 -0
  97. package/esm2015/cycles/cycles-emissions-chart/cycles-emissions-chart.component.js +93 -0
  98. package/esm2015/cycles/cycles-emissions-logs/cycles-emissions-logs.component.js +63 -0
  99. package/esm2015/cycles/cycles-functional-unit-measure/cycles-functional-unit-measure.component.js +22 -0
  100. package/esm2015/cycles/cycles-result/cycles-result.component.js +92 -0
  101. package/esm2015/cycles/cycles-suggest-form/cycles-suggest-form.component.js +83 -0
  102. package/esm2015/cycles/cycles.model.js +2 -0
  103. package/esm2015/cycles/cycles.module.js +66 -0
  104. package/esm2015/cycles/public-api.js +11 -0
  105. package/esm2015/engine/aggregation-engine.service.js +126 -0
  106. package/esm2015/engine/engine.service.js +78 -0
  107. package/esm2015/engine/public-api.js +3 -0
  108. package/esm2015/files/files-error.model.js +180 -0
  109. package/esm2015/files/files-form/files-form.component.js +439 -0
  110. package/esm2015/files/files-form.model.js +548 -0
  111. package/esm2015/files/files.module.js +42 -0
  112. package/esm2015/files/public-api.js +3 -0
  113. package/esm2015/fontawesome/fontawesome.module.js +36 -0
  114. package/esm2015/fontawesome/public-api.js +2 -0
  115. package/esm2015/hestia-earth-ui-components.js +5 -0
  116. package/esm2015/impact-assessments/impact-assessments-indicator-breakdown-chart/impact-assessments-indicator-breakdown-chart.component.js +161 -0
  117. package/esm2015/impact-assessments/impact-assessments-indicators-chart/impact-assessments-indicators-chart.component.js +142 -0
  118. package/esm2015/impact-assessments/impact-assessments-products/impact-assessments-products.component.js +252 -0
  119. package/esm2015/impact-assessments/impact-assessments-products-logs/impact-assessments-products-logs.component.js +68 -0
  120. package/esm2015/impact-assessments/impact-assessments.model.js +2 -0
  121. package/esm2015/impact-assessments/impact-assessments.module.js +47 -0
  122. package/esm2015/impact-assessments/public-api.js +7 -0
  123. package/esm2015/mendeley/mendeley.service.js +35 -0
  124. package/esm2015/mendeley/public-api.js +2 -0
  125. package/esm2015/node/node-csv-export-confirm/node-csv-export-confirm.component.js +65 -0
  126. package/esm2015/node/node-csv-select-headers/node-csv-select-headers.component.js +119 -0
  127. package/esm2015/node/node-csv.service.js +24 -0
  128. package/esm2015/node/node-diffs/node-diffs.component.js +67 -0
  129. package/esm2015/node/node-diffs/node-diffs.model.js +6 -0
  130. package/esm2015/node/node-icon/node-icon.component.js +37 -0
  131. package/esm2015/node/node-link/node-link.component.js +33 -0
  132. package/esm2015/node/node-logs-file/node-logs-file.component.js +63 -0
  133. package/esm2015/node/node-logs-models/node-logs-models.component.js +188 -0
  134. package/esm2015/node/node-missing-lookup-factors/node-missing-lookup-factors.component.js +34 -0
  135. package/esm2015/node/node-value-details/node-value-details.component.js +46 -0
  136. package/esm2015/node/node.module.js +67 -0
  137. package/esm2015/node/node.service.js +113 -0
  138. package/esm2015/node/public-api.js +13 -0
  139. package/esm2015/public-api.js +16 -0
  140. package/esm2015/schema/public-api.js +2 -0
  141. package/esm2015/schema/schema.service.js +167 -0
  142. package/esm2015/search/public-api.js +4 -0
  143. package/esm2015/search/search.model.js +204 -0
  144. package/esm2015/search/search.module.js +13 -0
  145. package/esm2015/search/search.service.js +99 -0
  146. package/esm2015/sites/public-api.js +6 -0
  147. package/esm2015/sites/sites-maps/sites-maps.component.js +162 -0
  148. package/esm2015/sites/sites-measurements/sites-measurements.component.js +78 -0
  149. package/esm2015/sites/sites-measurements-logs/sites-measurements-logs.component.js +63 -0
  150. package/esm2015/sites/sites.model.js +10 -0
  151. package/esm2015/sites/sites.module.js +42 -0
  152. package/esm2015/terms/public-api.js +2 -0
  153. package/esm2015/terms/terms.model.js +184 -0
  154. package/esm2015/users/public-api.js +2 -0
  155. package/esm2015/users/users.service.js +17 -0
  156. package/fesm2015/hestia-earth-ui-components.js +5920 -0
  157. package/fesm2015/hestia-earth-ui-components.js.map +1 -0
  158. package/files/files-error.model.d.ts +40 -0
  159. package/files/files-form/files-form.component.d.ts +94 -0
  160. package/files/files-form.model.d.ts +113 -0
  161. package/files/files.module.d.ts +13 -0
  162. package/files/public-api.d.ts +2 -0
  163. package/fontawesome/fontawesome.module.d.ts +10 -0
  164. package/fontawesome/public-api.d.ts +1 -0
  165. package/hestia-earth-ui-components.d.ts +5 -0
  166. package/impact-assessments/impact-assessments-indicator-breakdown-chart/impact-assessments-indicator-breakdown-chart.component.d.ts +28 -0
  167. package/impact-assessments/impact-assessments-indicators-chart/impact-assessments-indicators-chart.component.d.ts +25 -0
  168. package/impact-assessments/impact-assessments-products/impact-assessments-products.component.d.ts +77 -0
  169. package/impact-assessments/impact-assessments-products-logs/impact-assessments-products-logs.component.d.ts +25 -0
  170. package/impact-assessments/impact-assessments.model.d.ts +6 -0
  171. package/impact-assessments/impact-assessments.module.d.ts +15 -0
  172. package/impact-assessments/public-api.d.ts +6 -0
  173. package/mendeley/mendeley.service.d.ts +25 -0
  174. package/mendeley/public-api.d.ts +1 -0
  175. package/node/node-csv-export-confirm/node-csv-export-confirm.component.d.ts +32 -0
  176. package/node/node-csv-select-headers/node-csv-select-headers.component.d.ts +42 -0
  177. package/node/node-csv.service.d.ts +9 -0
  178. package/node/node-diffs/node-diffs.component.d.ts +23 -0
  179. package/node/node-diffs/node-diffs.model.d.ts +4 -0
  180. package/node/node-icon/node-icon.component.d.ts +8 -0
  181. package/node/node-link/node-link.component.d.ts +11 -0
  182. package/node/node-logs-file/node-logs-file.component.d.ts +31 -0
  183. package/node/node-logs-models/node-logs-models.component.d.ts +72 -0
  184. package/node/node-missing-lookup-factors/node-missing-lookup-factors.component.d.ts +14 -0
  185. package/node/node-value-details/node-value-details.component.d.ts +17 -0
  186. package/node/node.module.d.ts +20 -0
  187. package/node/node.service.d.ts +70 -0
  188. package/node/public-api.d.ts +12 -0
  189. package/package.json +53 -0
  190. package/public-api.d.ts +15 -0
  191. package/schema/public-api.d.ts +1 -0
  192. package/schema/schema.service.d.ts +51 -0
  193. package/search/public-api.d.ts +3 -0
  194. package/search/search.model.d.ts +243 -0
  195. package/search/search.module.d.ts +7 -0
  196. package/search/search.service.d.ts +129 -0
  197. package/sites/public-api.d.ts +5 -0
  198. package/sites/sites-maps/sites-maps.component.d.ts +35 -0
  199. package/sites/sites-measurements/sites-measurements.component.d.ts +31 -0
  200. package/sites/sites-measurements-logs/sites-measurements-logs.component.d.ts +23 -0
  201. package/sites/sites.model.d.ts +10 -0
  202. package/sites/sites.module.d.ts +13 -0
  203. package/styles.scss +197 -0
  204. package/terms/public-api.d.ts +1 -0
  205. package/terms/terms.model.d.ts +21 -0
  206. package/users/public-api.d.ts +1 -0
  207. 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"]}