geonetwork-ui 2.4.0-dev.71a43b38 → 2.4.0-dev.8118addf

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 (179) hide show
  1. package/esm2022/libs/api/metadata-converter/src/index.mjs +2 -1
  2. package/esm2022/libs/api/metadata-converter/src/lib/gn4/atomic-operations.mjs +2 -1
  3. package/esm2022/libs/api/metadata-converter/src/lib/gn4/types/metadata.model.mjs +1 -1
  4. package/esm2022/libs/api/metadata-converter/src/lib/iso19139/iso19139.converter.mjs +5 -5
  5. package/esm2022/libs/api/metadata-converter/src/lib/iso19139/read-parts.mjs +30 -2
  6. package/esm2022/libs/api/metadata-converter/src/lib/iso19139/utils/geometry.mjs +31 -0
  7. package/esm2022/libs/api/metadata-converter/src/lib/iso19139/write-parts.mjs +23 -1
  8. package/esm2022/libs/api/metadata-converter/src/lib/xml-utils.mjs +6 -1
  9. package/esm2022/libs/api/repository/src/lib/gn4/gn4-repository.mjs +12 -1
  10. package/esm2022/libs/common/domain/src/lib/model/record/metadata.model.mjs +1 -1
  11. package/esm2022/libs/common/domain/src/lib/repository/records-repository.interface.mjs +1 -1
  12. package/esm2022/libs/data-access/gn4/src/openapi/api/records.api.service.mjs +2 -6
  13. package/esm2022/libs/feature/editor/src/lib/+state/editor.actions.mjs +2 -1
  14. package/esm2022/libs/feature/editor/src/lib/+state/editor.effects.mjs +3 -3
  15. package/esm2022/libs/feature/editor/src/lib/+state/editor.facade.mjs +7 -2
  16. package/esm2022/libs/feature/editor/src/lib/+state/editor.models.mjs +1 -1
  17. package/esm2022/libs/feature/editor/src/lib/+state/editor.reducer.mjs +7 -3
  18. package/esm2022/libs/feature/editor/src/lib/+state/editor.selectors.mjs +16 -6
  19. package/esm2022/libs/feature/editor/src/lib/components/record-form/form-field/form-field.component.mjs +3 -3
  20. package/esm2022/libs/feature/editor/src/lib/components/record-form/form-field/index.mjs +1 -2
  21. package/esm2022/libs/feature/editor/src/lib/components/record-form/record-form.component.mjs +11 -7
  22. package/esm2022/libs/feature/editor/src/lib/expressions.mjs +1 -1
  23. package/esm2022/libs/feature/editor/src/lib/fields.config.mjs +143 -60
  24. package/esm2022/libs/feature/editor/src/lib/models/editor-config.model.mjs +2 -0
  25. package/esm2022/libs/feature/editor/src/lib/models/index.mjs +2 -1
  26. package/esm2022/libs/feature/editor/src/lib/services/editor.service.mjs +4 -3
  27. package/esm2022/libs/feature/map/src/lib/utils/map-utils.service.mjs +3 -4
  28. package/esm2022/libs/feature/search/src/lib/results-table/results-table-container.component.mjs +9 -3
  29. package/esm2022/libs/ui/elements/src/lib/metadata-quality/metadata-quality.component.mjs +5 -11
  30. package/esm2022/libs/ui/elements/src/lib/metadata-quality-item/metadata-quality-item.component.mjs +3 -3
  31. package/esm2022/libs/ui/elements/src/lib/ui-elements.module.mjs +5 -2
  32. package/esm2022/libs/ui/elements/src/lib/user-preview/user-preview.component.mjs +3 -3
  33. package/esm2022/libs/ui/search/src/lib/results-table/action-menu/action-menu.component.mjs +29 -0
  34. package/esm2022/libs/ui/search/src/lib/results-table/results-table.component.mjs +15 -7
  35. package/esm2022/libs/ui/widgets/src/index.mjs +2 -1
  36. package/esm2022/libs/ui/widgets/src/lib/popover/popover.component.mjs +68 -0
  37. package/esm2022/translations/de.json +31 -1
  38. package/esm2022/translations/en.json +31 -1
  39. package/esm2022/translations/es.json +30 -0
  40. package/esm2022/translations/fr.json +31 -1
  41. package/esm2022/translations/it.json +31 -1
  42. package/esm2022/translations/nl.json +30 -0
  43. package/esm2022/translations/pt.json +30 -0
  44. package/fesm2022/geonetwork-ui.mjs +617 -117
  45. package/fesm2022/geonetwork-ui.mjs.map +1 -1
  46. package/libs/api/metadata-converter/src/index.d.ts +1 -0
  47. package/libs/api/metadata-converter/src/index.d.ts.map +1 -1
  48. package/libs/api/metadata-converter/src/lib/gn4/atomic-operations.d.ts.map +1 -1
  49. package/libs/api/metadata-converter/src/lib/gn4/types/metadata.model.d.ts +1 -0
  50. package/libs/api/metadata-converter/src/lib/gn4/types/metadata.model.d.ts.map +1 -1
  51. package/libs/api/metadata-converter/src/lib/iso19139/iso19139.converter.d.ts.map +1 -1
  52. package/libs/api/metadata-converter/src/lib/iso19139/read-parts.d.ts +8 -1
  53. package/libs/api/metadata-converter/src/lib/iso19139/read-parts.d.ts.map +1 -1
  54. package/libs/api/metadata-converter/src/lib/iso19139/utils/geometry.d.ts +5 -0
  55. package/libs/api/metadata-converter/src/lib/iso19139/utils/geometry.d.ts.map +1 -0
  56. package/libs/api/metadata-converter/src/lib/iso19139/write-parts.d.ts +3 -1
  57. package/libs/api/metadata-converter/src/lib/iso19139/write-parts.d.ts.map +1 -1
  58. package/libs/api/metadata-converter/src/lib/xml-utils.d.ts +1 -0
  59. package/libs/api/metadata-converter/src/lib/xml-utils.d.ts.map +1 -1
  60. package/libs/api/repository/src/lib/gn4/gn4-repository.d.ts +1 -0
  61. package/libs/api/repository/src/lib/gn4/gn4-repository.d.ts.map +1 -1
  62. package/libs/common/domain/src/lib/model/record/metadata.model.d.ts +2 -1
  63. package/libs/common/domain/src/lib/model/record/metadata.model.d.ts.map +1 -1
  64. package/libs/common/domain/src/lib/repository/records-repository.interface.d.ts +9 -0
  65. package/libs/common/domain/src/lib/repository/records-repository.interface.d.ts.map +1 -1
  66. package/libs/data-access/gn4/src/openapi/api/records.api.service.d.ts.map +1 -1
  67. package/libs/feature/editor/src/lib/+state/editor.actions.d.ts +5 -0
  68. package/libs/feature/editor/src/lib/+state/editor.actions.d.ts.map +1 -1
  69. package/libs/feature/editor/src/lib/+state/editor.facade.d.ts +4 -4
  70. package/libs/feature/editor/src/lib/+state/editor.facade.d.ts.map +1 -1
  71. package/libs/feature/editor/src/lib/+state/editor.models.d.ts +8 -0
  72. package/libs/feature/editor/src/lib/+state/editor.models.d.ts.map +1 -1
  73. package/libs/feature/editor/src/lib/+state/editor.reducer.d.ts +4 -3
  74. package/libs/feature/editor/src/lib/+state/editor.reducer.d.ts.map +1 -1
  75. package/libs/feature/editor/src/lib/+state/editor.selectors.d.ts +4 -8
  76. package/libs/feature/editor/src/lib/+state/editor.selectors.d.ts.map +1 -1
  77. package/libs/feature/editor/src/lib/components/record-form/form-field/form-field.component.d.ts +1 -1
  78. package/libs/feature/editor/src/lib/components/record-form/form-field/form-field.component.d.ts.map +1 -1
  79. package/libs/feature/editor/src/lib/components/record-form/form-field/index.d.ts +0 -1
  80. package/libs/feature/editor/src/lib/components/record-form/form-field/index.d.ts.map +1 -1
  81. package/libs/feature/editor/src/lib/components/record-form/record-form.component.d.ts +5 -7
  82. package/libs/feature/editor/src/lib/components/record-form/record-form.component.d.ts.map +1 -1
  83. package/libs/feature/editor/src/lib/expressions.d.ts +1 -1
  84. package/libs/feature/editor/src/lib/expressions.d.ts.map +1 -1
  85. package/libs/feature/editor/src/lib/fields.config.d.ts +38 -2
  86. package/libs/feature/editor/src/lib/fields.config.d.ts.map +1 -1
  87. package/libs/feature/editor/src/lib/models/editor-config.model.d.ts +36 -0
  88. package/libs/feature/editor/src/lib/models/editor-config.model.d.ts.map +1 -0
  89. package/libs/feature/editor/src/lib/models/index.d.ts +1 -0
  90. package/libs/feature/editor/src/lib/models/index.d.ts.map +1 -1
  91. package/libs/feature/editor/src/lib/services/editor.service.d.ts +2 -2
  92. package/libs/feature/editor/src/lib/services/editor.service.d.ts.map +1 -1
  93. package/libs/feature/map/src/lib/utils/map-utils.service.d.ts.map +1 -1
  94. package/libs/feature/search/src/lib/results-table/results-table-container.component.d.ts +3 -1
  95. package/libs/feature/search/src/lib/results-table/results-table-container.component.d.ts.map +1 -1
  96. package/libs/ui/elements/src/lib/downloads-list/downloads-list.component.d.ts +1 -1
  97. package/libs/ui/elements/src/lib/metadata-quality/metadata-quality.component.d.ts +0 -3
  98. package/libs/ui/elements/src/lib/metadata-quality/metadata-quality.component.d.ts.map +1 -1
  99. package/libs/ui/elements/src/lib/record-api-form/record-api-form.component.d.ts +1 -1
  100. package/libs/ui/elements/src/lib/ui-elements.module.d.ts +7 -6
  101. package/libs/ui/elements/src/lib/ui-elements.module.d.ts.map +1 -1
  102. package/libs/ui/search/src/lib/results-table/action-menu/action-menu.component.d.ts +11 -0
  103. package/libs/ui/search/src/lib/results-table/action-menu/action-menu.component.d.ts.map +1 -0
  104. package/libs/ui/search/src/lib/results-table/results-table.component.d.ts +4 -2
  105. package/libs/ui/search/src/lib/results-table/results-table.component.d.ts.map +1 -1
  106. package/libs/ui/widgets/src/index.d.ts +1 -0
  107. package/libs/ui/widgets/src/index.d.ts.map +1 -1
  108. package/libs/ui/widgets/src/lib/popover/popover.component.d.ts +19 -0
  109. package/libs/ui/widgets/src/lib/popover/popover.component.d.ts.map +1 -0
  110. package/package.json +1 -1
  111. package/src/libs/api/metadata-converter/src/index.ts +1 -0
  112. package/src/libs/api/metadata-converter/src/lib/fixtures/geo2france.records.ts +5 -1
  113. package/src/libs/api/metadata-converter/src/lib/fixtures/geocat-ch.records.ts +37 -12
  114. package/src/libs/api/metadata-converter/src/lib/fixtures/metawal.records.ts +5 -1
  115. package/src/libs/api/metadata-converter/src/lib/gn4/atomic-operations.ts +1 -0
  116. package/src/libs/api/metadata-converter/src/lib/gn4/types/metadata.model.ts +1 -0
  117. package/src/libs/api/metadata-converter/src/lib/iso19139/iso19139.converter.ts +4 -2
  118. package/src/libs/api/metadata-converter/src/lib/iso19139/read-parts.ts +72 -2
  119. package/src/libs/api/metadata-converter/src/lib/iso19139/utils/geometry.ts +39 -0
  120. package/src/libs/api/metadata-converter/src/lib/iso19139/write-parts.ts +70 -1
  121. package/src/libs/api/metadata-converter/src/lib/xml-utils.ts +8 -0
  122. package/src/libs/api/repository/src/lib/gn4/gn4-repository.ts +20 -0
  123. package/src/libs/common/domain/src/lib/model/record/metadata.model.ts +2 -1
  124. package/src/libs/common/domain/src/lib/repository/records-repository.interface.ts +12 -0
  125. package/src/libs/common/fixtures/src/index.ts +2 -0
  126. package/src/libs/common/fixtures/src/lib/editor/editor.fixtures.ts +156 -0
  127. package/src/libs/common/fixtures/src/lib/editor/index.ts +1 -0
  128. package/src/libs/data-access/gn4/src/openapi/api/records.api.service.ts +1 -5
  129. package/src/libs/data-access/gn4/src/spec.yaml +0 -8
  130. package/src/libs/feature/editor/src/lib/+state/editor.actions.ts +5 -0
  131. package/src/libs/feature/editor/src/lib/+state/editor.effects.ts +2 -2
  132. package/src/libs/feature/editor/src/lib/+state/editor.facade.ts +9 -1
  133. package/src/libs/feature/editor/src/lib/+state/editor.models.ts +11 -0
  134. package/src/libs/feature/editor/src/lib/+state/editor.reducer.ts +11 -5
  135. package/src/libs/feature/editor/src/lib/+state/editor.selectors.ts +22 -8
  136. package/src/libs/feature/editor/src/lib/components/record-form/form-field/form-field.component.html +9 -9
  137. package/src/libs/feature/editor/src/lib/components/record-form/form-field/form-field.component.ts +2 -1
  138. package/src/libs/feature/editor/src/lib/components/record-form/form-field/index.ts +0 -1
  139. package/src/libs/feature/editor/src/lib/components/record-form/record-form.component.html +51 -11
  140. package/src/libs/feature/editor/src/lib/components/record-form/record-form.component.ts +15 -8
  141. package/src/libs/feature/editor/src/lib/expressions.ts +1 -1
  142. package/src/libs/feature/editor/src/lib/fields.config.ts +175 -61
  143. package/src/libs/feature/editor/src/lib/models/editor-config.model.ts +53 -0
  144. package/src/libs/feature/editor/src/lib/models/index.ts +1 -0
  145. package/src/libs/feature/editor/src/lib/services/editor.service.ts +8 -4
  146. package/src/libs/feature/map/src/lib/utils/map-utils.service.ts +2 -3
  147. package/src/libs/feature/search/src/lib/results-table/results-table-container.component.html +1 -0
  148. package/src/libs/feature/search/src/lib/results-table/results-table-container.component.ts +5 -0
  149. package/src/libs/ui/elements/src/lib/metadata-quality/metadata-quality.component.html +14 -20
  150. package/src/libs/ui/elements/src/lib/metadata-quality/metadata-quality.component.ts +0 -10
  151. package/src/libs/ui/elements/src/lib/metadata-quality-item/metadata-quality-item.component.html +1 -1
  152. package/src/libs/ui/elements/src/lib/ui-elements.module.ts +2 -1
  153. package/src/libs/ui/elements/src/lib/user-preview/user-preview.component.html +1 -1
  154. package/src/libs/ui/search/src/lib/results-table/action-menu/action-menu.component.css +0 -0
  155. package/src/libs/ui/search/src/lib/results-table/action-menu/action-menu.component.html +17 -0
  156. package/src/libs/ui/search/src/lib/results-table/action-menu/action-menu.component.ts +22 -0
  157. package/src/libs/ui/search/src/lib/results-table/results-table.component.html +9 -0
  158. package/src/libs/ui/search/src/lib/results-table/results-table.component.ts +26 -12
  159. package/src/libs/ui/widgets/src/index.ts +1 -0
  160. package/src/libs/ui/widgets/src/lib/popover/popover.component.css +0 -0
  161. package/src/libs/ui/widgets/src/lib/popover/popover.component.html +3 -0
  162. package/src/libs/ui/widgets/src/lib/popover/popover.component.ts +85 -0
  163. package/tailwind.base.css +2 -1
  164. package/translations/de.json +31 -1
  165. package/translations/en.json +31 -1
  166. package/translations/es.json +30 -0
  167. package/translations/fr.json +31 -1
  168. package/translations/it.json +31 -1
  169. package/translations/nl.json +30 -0
  170. package/translations/pt.json +30 -0
  171. package/translations/sk.json +31 -1
  172. package/esm2022/libs/feature/editor/src/lib/components/record-form/form-field/form-field.model.mjs +0 -2
  173. package/esm2022/libs/feature/editor/src/lib/models/fields.model.mjs +0 -2
  174. package/libs/feature/editor/src/lib/components/record-form/form-field/form-field.model.d.ts +0 -26
  175. package/libs/feature/editor/src/lib/components/record-form/form-field/form-field.model.d.ts.map +0 -1
  176. package/libs/feature/editor/src/lib/models/fields.model.d.ts +0 -15
  177. package/libs/feature/editor/src/lib/models/fields.model.d.ts.map +0 -1
  178. package/src/libs/feature/editor/src/lib/components/record-form/form-field/form-field.model.ts +0 -43
  179. package/src/libs/feature/editor/src/lib/models/fields.model.ts +0 -29
@@ -1,5 +1,6 @@
1
1
  import { createFeatureSelector, createSelector } from '@ngrx/store'
2
2
  import { EDITOR_FEATURE_KEY, EditorState } from './editor.reducer'
3
+ import { EditorSectionWithValues } from './editor.models'
3
4
 
4
5
  export const selectEditorState =
5
6
  createFeatureSelector<EditorState>(EDITOR_FEATURE_KEY)
@@ -34,16 +35,29 @@ export const selectRecordAlreadySavedOnce = createSelector(
34
35
  (state: EditorState) => state.alreadySavedOnce
35
36
  )
36
37
 
37
- export const selectRecordFieldsConfig = createSelector(
38
+ export const selectEditorConfig = createSelector(
38
39
  selectEditorState,
39
- (state: EditorState) => state.fieldsConfig
40
+ (state: EditorState) => state.editorConfig
40
41
  )
41
42
 
42
- export const selectRecordFields = createSelector(
43
+ export const selectCurrentPage = createSelector(
43
44
  selectEditorState,
44
- (state: EditorState) =>
45
- state.fieldsConfig.map((fieldConfig) => ({
46
- config: fieldConfig,
47
- value: state.record?.[fieldConfig.model] ?? null,
48
- }))
45
+ (state: EditorState) => state.currentPage
46
+ )
47
+
48
+ export const selectRecordSections = createSelector(
49
+ selectEditorState,
50
+ (state: EditorState) => {
51
+ const currentPage = state.editorConfig.pages[state.currentPage]
52
+ if (!currentPage) {
53
+ return [] as EditorSectionWithValues[]
54
+ }
55
+ return currentPage.sections.map((section) => ({
56
+ ...section,
57
+ fieldsWithValues: section.fields.map((fieldConfig) => ({
58
+ config: fieldConfig,
59
+ value: state.record?.[fieldConfig.model] ?? null,
60
+ })),
61
+ })) as EditorSectionWithValues[]
62
+ }
49
63
  )
@@ -4,8 +4,8 @@
4
4
  </ng-container>
5
5
  <ng-template #withGenericWrapper>
6
6
  <gn-ui-form-field-wrapper
7
- [label]="config.labelKey | translate"
8
- [hint]="config.hintKey | translate"
7
+ [label]="config.labelKey! | translate"
8
+ [hint]="config.hintKey! | translate"
9
9
  >
10
10
  <ng-container *ngTemplateOutlet="fieldContent"></ng-container>
11
11
  </gn-ui-form-field-wrapper>
@@ -15,14 +15,14 @@
15
15
  <ng-template #fieldContent>
16
16
  <ng-container *ngIf="isTitle">
17
17
  <div class="flex justify-between items-center gap-3">
18
- <h2
18
+ <span
19
19
  #titleInput
20
- class="grow text-3xl font-normal"
20
+ class="grow font-title text-3xl font-normal"
21
21
  [gnUiEditableLabel]="true"
22
22
  (editableLabelChanged)="formControl.setValue($event)"
23
23
  >
24
24
  {{ formControl.value }}
25
- </h2>
25
+ </span>
26
26
  <span
27
27
  class="material-symbols-outlined gn-ui-icon-small m-2 cursor-pointer"
28
28
  (click)="focusTitleInput()"
@@ -30,7 +30,7 @@
30
30
  >
31
31
  <span
32
32
  class="material-symbols-outlined gn-ui-icon-small m-2"
33
- [matTooltip]="config.hintKey | translate"
33
+ [matTooltip]="config.hintKey! | translate"
34
34
  matTooltipPosition="above"
35
35
  >
36
36
  help
@@ -41,14 +41,14 @@
41
41
  <gn-ui-form-field-rich
42
42
  class="h-[8rem]"
43
43
  [control]="formControl"
44
- [label]="config.labelKey | translate"
45
- [hint]="config.hintKey | translate"
44
+ [label]="config.labelKey! | translate"
45
+ [hint]="config.hintKey! | translate"
46
46
  ></gn-ui-form-field-rich>
47
47
  </ng-container>
48
48
  <ng-container *ngIf="isLicenses">
49
49
  <gn-ui-form-field-license
50
50
  [control]="formControl"
51
- [label]="config.labelKey | translate"
51
+ [label]="config.labelKey! | translate"
52
52
  ></gn-ui-form-field-license>
53
53
  </ng-container>
54
54
  <ng-container *ngIf="isResourceUpdated">
@@ -25,10 +25,10 @@ import { FormFieldObjectComponent } from './form-field-object/form-field-object.
25
25
  import { FormFieldRichComponent } from './form-field-rich/form-field-rich.component'
26
26
  import { FormFieldSimpleComponent } from './form-field-simple/form-field-simple.component'
27
27
  import { FormFieldSpatialExtentComponent } from './form-field-spatial-extent/form-field-spatial-extent.component'
28
- import { FormFieldConfig } from './form-field.model'
29
28
  import { FormFieldUpdateFrequencyComponent } from './form-field-update-frequency/form-field-update-frequency.component'
30
29
  import { CatalogRecordKeys } from '../../../../../../../../libs/common/domain/src/lib/model/record'
31
30
  import { FormFieldKeywordsComponent } from './form-field-keywords/form-field-keywords.component'
31
+ import { FormFieldConfig } from '../../../models'
32
32
 
33
33
  @Component({
34
34
  selector: 'gn-ui-form-field',
@@ -65,6 +65,7 @@ export class FormFieldComponent {
65
65
  emitEvent: false,
66
66
  })
67
67
  }
68
+
68
69
  @Output() valueChange: Observable<unknown>
69
70
 
70
71
  @ViewChild('titleInput') titleInput: ElementRef
@@ -9,4 +9,3 @@ export * from './form-field-object/form-field-object.component'
9
9
  export * from './form-field-array/form-field-array.component'
10
10
  export * from './form-field-spatial-extent/form-field-spatial-extent.component'
11
11
  export * from './form-field.component'
12
- export * from './form-field.model'
@@ -1,11 +1,51 @@
1
- <div class="flex flex-col gap-6 p-6">
2
- <ng-container *ngFor="let field of fields$ | async; trackBy: fieldTracker">
3
- <gn-ui-form-field
4
- *ngIf="field.config.formFieldConfig && !field.config.hidden"
5
- [model]="field.config.model"
6
- [config]="field.config.formFieldConfig"
7
- [value]="field.value"
8
- (valueChange)="handleFieldValueChange(field, $event)"
9
- ></gn-ui-form-field>
10
- </ng-container>
11
- </div>
1
+ <ng-container>
2
+ <div class="flex flex-col gap-6 p-8">
3
+ <ng-container
4
+ *ngFor="
5
+ let section of facade.currentSections$ | async;
6
+ trackBy: sectionTracker
7
+ "
8
+ >
9
+ <ng-container *ngIf="!section.hidden">
10
+ <div class="flex flex-col gap-8 border p-8 rounded-[8px] shadow">
11
+ <div
12
+ class="flex flex-col gap-2"
13
+ [ngClass]="section.labelKey ? 'mb-4' : 'hidden'"
14
+ >
15
+ <div
16
+ *ngIf="section.labelKey"
17
+ class="text-2xl font-petrona text-secondary"
18
+ translate
19
+ >
20
+ {{ section.labelKey }}
21
+ </div>
22
+ <div
23
+ *ngIf="section.descriptionKey"
24
+ class="text-secondary-lightest"
25
+ translate
26
+ >
27
+ {{ section.descriptionKey }}
28
+ </div>
29
+ </div>
30
+ <ng-container
31
+ *ngFor="
32
+ let field of section.fieldsWithValues;
33
+ trackBy: fieldTracker
34
+ "
35
+ >
36
+ <ng-container *ngIf="!field.config.hidden">
37
+ <gn-ui-form-field
38
+ [model]="field.config.model!"
39
+ [config]="field.config.formFieldConfig"
40
+ [value]="field.value"
41
+ (valueChange)="
42
+ handleFieldValueChange(field.config.model!, $event)
43
+ "
44
+ ></gn-ui-form-field>
45
+ </ng-container>
46
+ </ng-container>
47
+ </div>
48
+ </ng-container>
49
+ </ng-container>
50
+ </div>
51
+ </ng-container>
@@ -1,8 +1,13 @@
1
1
  import { CommonModule } from '@angular/common'
2
2
  import { ChangeDetectionStrategy, Component } from '@angular/core'
3
3
  import { EditorFacade } from '../../+state/editor.facade'
4
- import { EditorFieldState, EditorFieldValue } from '../../models/fields.model'
4
+ import { EditorFieldValue } from '../../models'
5
5
  import { FormFieldComponent } from './form-field'
6
+ import { TranslateModule } from '@ngx-translate/core'
7
+ import {
8
+ EditorFieldWithValue,
9
+ EditorSectionWithValues,
10
+ } from '../../+state/editor.models'
6
11
 
7
12
  @Component({
8
13
  selector: 'gn-ui-record-form',
@@ -10,21 +15,23 @@ import { FormFieldComponent } from './form-field'
10
15
  styleUrls: ['./record-form.component.css'],
11
16
  changeDetection: ChangeDetectionStrategy.OnPush,
12
17
  standalone: true,
13
- imports: [CommonModule, FormFieldComponent],
18
+ imports: [CommonModule, FormFieldComponent, TranslateModule],
14
19
  })
15
20
  export class RecordFormComponent {
16
- fields$ = this.facade.recordFields$
17
-
18
21
  constructor(public facade: EditorFacade) {}
19
22
 
20
- handleFieldValueChange(field: EditorFieldState, newValue: EditorFieldValue) {
21
- if (!field.config.model) {
23
+ handleFieldValueChange(model: string, newValue: EditorFieldValue) {
24
+ if (!model) {
22
25
  return
23
26
  }
24
- this.facade.updateRecordField(field.config.model, newValue)
27
+ this.facade.updateRecordField(model, newValue)
25
28
  }
26
29
 
27
- fieldTracker(index: number, field: EditorFieldState) {
30
+ fieldTracker(index: number, field: EditorFieldWithValue): any {
28
31
  return field.config.model
29
32
  }
33
+
34
+ sectionTracker(index: number, section: EditorSectionWithValues): any {
35
+ return section.labelKey
36
+ }
30
37
  }
@@ -1,4 +1,4 @@
1
- import { EditorFieldState, EditorFieldValue } from './models/fields.model'
1
+ import { EditorFieldState, EditorFieldValue } from './models/'
2
2
 
3
3
  export type ExpressionEvaluator = (field: EditorFieldState) => EditorFieldValue
4
4
 
@@ -1,71 +1,185 @@
1
1
  import { marker } from '@biesbjerg/ngx-translate-extract-marker'
2
- import { EditorFieldsConfig } from './models/fields.model'
3
-
4
- export const DEFAULT_FIELDS: EditorFieldsConfig = [
5
- {
6
- model: 'title',
7
- formFieldConfig: {
8
- labelKey: marker('editor.record.form.metadata.title'),
9
- type: 'text',
10
- },
2
+ import {
3
+ EditorConfig,
4
+ EditorField,
5
+ EditorSection,
6
+ } from './models/editor-config.model'
7
+
8
+ /**
9
+ * This file contains the configuration of the fields that will be displayed in the editor.
10
+ * To add a new field, you need to create a new EditorField object in the fields part of this file.
11
+ * Then add it to the corresponding section in the sections part of this file.
12
+ * Finally, add the section to the corresponding page in the pages part of this file.
13
+ */
14
+
15
+ /************************************************************
16
+ *************** FIELDS *****************
17
+ ************************************************************
18
+ */
19
+
20
+ export const RECORD_LICENSE_FIELD: EditorField = {
21
+ model: 'licenses',
22
+ formFieldConfig: {
23
+ labelKey: marker('editor.record.form.field.license'),
11
24
  },
12
- {
13
- model: 'abstract',
14
- formFieldConfig: {
15
- labelKey: marker('editor.record.form.abstract'),
16
- type: 'rich',
17
- },
25
+ }
26
+
27
+ export const RECORD_KEYWORDS_FIELD: EditorField = {
28
+ model: 'keywords',
29
+ formFieldConfig: {
30
+ labelKey: marker('editor.record.form.field.keywords'),
18
31
  },
19
- {
20
- model: 'uniqueIdentifier',
21
- formFieldConfig: {
22
- labelKey: marker('editor.record.form.unique.identifier'),
23
- type: 'text',
24
- locked: true,
25
- },
32
+ }
33
+
34
+ export const RECORD_UNIQUE_IDENTIFIER_FIELD: EditorField = {
35
+ model: 'uniqueIdentifier',
36
+ formFieldConfig: {
37
+ labelKey: marker('editor.record.form.field.uniqueIdentifier'),
38
+ locked: true,
26
39
  },
27
- {
28
- model: 'recordUpdated',
29
- formFieldConfig: {
30
- labelKey: marker('editor.record.form.record.updated'),
31
- type: 'text',
32
- locked: true,
33
- },
34
- onSaveProcess: '${dateNow()}',
40
+ }
41
+
42
+ export const RECORD_RESOURCE_UPDATED_FIELD: EditorField = {
43
+ model: 'resourceUpdated',
44
+ formFieldConfig: {
45
+ labelKey: marker('editor.record.form.field.resourceUpdated'),
35
46
  },
36
- {
37
- model: 'licenses',
38
- formFieldConfig: {
39
- labelKey: marker('editor.record.form.license'),
40
- type: 'list',
41
- },
47
+ }
48
+
49
+ export const RECORD_UPDATED_FIELD: EditorField = {
50
+ model: 'recordUpdated',
51
+ formFieldConfig: {
52
+ labelKey: marker('editor.record.form.field.recordUpdated'),
53
+ locked: true,
42
54
  },
43
- {
44
- model: 'resourceUpdated',
45
- formFieldConfig: {
46
- labelKey: marker('editor.record.form.resourceUpdated'),
47
- type: 'date',
48
- },
55
+ onSaveProcess: '${dateNow()}',
56
+ }
57
+
58
+ export const RECORD_UPDATE_FREQUENCY_FIELD: EditorField = {
59
+ model: 'updateFrequency',
60
+ formFieldConfig: {
61
+ labelKey: marker('editor.record.form.field.updateFrequency'),
49
62
  },
50
- {
51
- model: 'updateFrequency',
52
- formFieldConfig: {
53
- labelKey: marker('editor.record.form.updateFrequency'),
54
- type: 'text',
55
- },
63
+ }
64
+
65
+ export const RECORD_TEMPORAL_EXTENTS_FIELD: EditorField = {
66
+ model: 'temporalExtents',
67
+ formFieldConfig: {
68
+ labelKey: marker('editor.record.form.field.temporalExtents'),
56
69
  },
57
- {
58
- model: 'temporalExtents',
59
- formFieldConfig: {
60
- labelKey: marker('editor.record.form.temporalExtents'),
61
- type: 'list',
62
- },
70
+ }
71
+
72
+ export const RECORD_TITLE_FIELD: EditorField = {
73
+ model: 'title',
74
+ formFieldConfig: {
75
+ labelKey: marker('editor.record.form.field.title'),
63
76
  },
64
- {
65
- model: 'keywords',
66
- formFieldConfig: {
67
- labelKey: marker('editor.record.form.keywords'),
68
- type: 'list',
69
- },
77
+ }
78
+
79
+ export const RECORD_ABSTRACT_FIELD: EditorField = {
80
+ model: 'abstract',
81
+ formFieldConfig: {
82
+ labelKey: marker('editor.record.form.field.abstract'),
70
83
  },
71
- ]
84
+ }
85
+
86
+ /************************************************************
87
+ *************** SECTIONS *****************
88
+ ************************************************************
89
+ */
90
+
91
+ export const TITLE_SECTION: EditorSection = {
92
+ hidden: false,
93
+ fields: [RECORD_TITLE_FIELD, RECORD_ABSTRACT_FIELD],
94
+ }
95
+
96
+ export const ABOUT_SECTION: EditorSection = {
97
+ labelKey: marker('editor.record.form.section.about.label'),
98
+ descriptionKey: marker('editor.record.form.section.about.description'),
99
+ hidden: false,
100
+ fields: [
101
+ RECORD_UNIQUE_IDENTIFIER_FIELD,
102
+ RECORD_RESOURCE_UPDATED_FIELD,
103
+ RECORD_UPDATED_FIELD,
104
+ RECORD_UPDATE_FREQUENCY_FIELD,
105
+ RECORD_TEMPORAL_EXTENTS_FIELD,
106
+ ],
107
+ }
108
+
109
+ export const GEOGRAPHICAL_COVERAGE_SECTION: EditorSection = {
110
+ labelKey: marker('editor.record.form.section.geographicalCoverage.label'),
111
+ hidden: false,
112
+ fields: [],
113
+ }
114
+
115
+ export const ASSOCIATED_RESOURCES_SECTION: EditorSection = {
116
+ labelKey: marker('editor.record.form.section.associatedResources.label'),
117
+ descriptionKey: marker(
118
+ 'editor.record.form.section.associatedResources.description'
119
+ ),
120
+ hidden: false,
121
+ fields: [],
122
+ }
123
+
124
+ export const ANNEXES_SECTION: EditorSection = {
125
+ labelKey: marker('editor.record.form.section.annexes.label'),
126
+ hidden: false,
127
+ fields: [],
128
+ }
129
+
130
+ export const CLASSIFICATION_SECTION: EditorSection = {
131
+ labelKey: marker('editor.record.form.section.classification.label'),
132
+ descriptionKey: marker(
133
+ 'editor.record.form.section.classification.description'
134
+ ),
135
+ hidden: false,
136
+ fields: [RECORD_KEYWORDS_FIELD],
137
+ }
138
+
139
+ export const USE_AND_ACCESS_CONDITIONS_SECTION: EditorSection = {
140
+ labelKey: marker('editor.record.form.section.useAndAccessConditions.label'),
141
+ hidden: false,
142
+ fields: [RECORD_LICENSE_FIELD],
143
+ }
144
+
145
+ export const DATA_MANAGERS_SECTION: EditorSection = {
146
+ labelKey: marker('editor.record.form.section.dataManagers.label'),
147
+ descriptionKey: marker('editor.record.form.section.dataManagers.description'),
148
+ hidden: false,
149
+ fields: [],
150
+ }
151
+
152
+ export const DATA_POINT_OF_CONTACT_SECTION: EditorSection = {
153
+ labelKey: marker('editor.record.form.section.dataPointOfContact.label'),
154
+ descriptionKey: marker(
155
+ 'editor.record.form.section.dataPointOfContact.description'
156
+ ),
157
+ hidden: false,
158
+ fields: [],
159
+ }
160
+
161
+ /************************************************************
162
+ *************** PAGES *****************
163
+ ************************************************************
164
+ */
165
+ export const DEFAULT_CONFIGURATION: EditorConfig = {
166
+ pages: [
167
+ {
168
+ labelKey: marker('editor.record.form.page.description'),
169
+ sections: [TITLE_SECTION, ABOUT_SECTION, GEOGRAPHICAL_COVERAGE_SECTION],
170
+ },
171
+ {
172
+ labelKey: marker('editor.record.form.page.ressources'),
173
+ sections: [ASSOCIATED_RESOURCES_SECTION, ANNEXES_SECTION],
174
+ },
175
+ {
176
+ labelKey: marker('editor.record.form.page.accessAndContact'),
177
+ sections: [
178
+ CLASSIFICATION_SECTION,
179
+ USE_AND_ACCESS_CONDITIONS_SECTION,
180
+ DATA_MANAGERS_SECTION,
181
+ DATA_POINT_OF_CONTACT_SECTION,
182
+ ],
183
+ },
184
+ ],
185
+ }
@@ -0,0 +1,53 @@
1
+ import { CatalogRecordKeys } from '../../../../../../libs/common/domain/src/lib/model/record'
2
+
3
+ // Expressions should be enclosed in `${}` to be recognized as such
4
+ // eg. ${dateNow()}
5
+ export type EditorFieldExpression = `$\{${string}}`
6
+
7
+ export type EditorFieldValue = string | number | boolean | unknown
8
+
9
+ export interface FormFieldConfig {
10
+ labelKey?: string
11
+ hintKey?: string
12
+ tooltipKey?: string
13
+ required?: boolean
14
+ locked?: boolean
15
+ invalid?: boolean
16
+ invalidHintKey?: string
17
+ }
18
+
19
+ export interface EditorField {
20
+ // configuration of the form field used as presentation
21
+ formFieldConfig: FormFieldConfig
22
+
23
+ // name of the target field in the record; will not change the record directly if not defined
24
+ model?: CatalogRecordKeys
25
+
26
+ // a hidden field won't show but can still be used to modify the record
27
+ // FIXME: currently this is redundant with an absence of formFieldConfig but necessary for clarity
28
+ hidden?: boolean
29
+
30
+ // the result of this expression will replace the field value on save
31
+ onSaveProcess?: EditorFieldExpression
32
+ }
33
+
34
+ export interface EditorSection {
35
+ labelKey?: string
36
+ descriptionKey?: string
37
+ hidden: boolean
38
+ fields: EditorField[]
39
+ }
40
+
41
+ export interface EditorFieldPage {
42
+ labelKey?: string
43
+ sections: EditorSection[]
44
+ }
45
+
46
+ export interface EditorConfig {
47
+ pages: EditorFieldPage[]
48
+ }
49
+
50
+ export interface EditorFieldState {
51
+ model: string
52
+ value: EditorFieldValue
53
+ }
@@ -1,2 +1,3 @@
1
1
  export * from './wizard-field.model'
2
2
  export * from './wizard-field.type'
3
+ export * from './editor-config.model'
@@ -2,7 +2,7 @@ import { Injectable } from '@angular/core'
2
2
  import { Observable, switchMap } from 'rxjs'
3
3
  import { map, tap } from 'rxjs/operators'
4
4
  import { CatalogRecord } from '../../../../../../libs/common/domain/src/lib/model/record'
5
- import { EditorFieldsConfig } from '../models/fields.model'
5
+ import { EditorConfig } from '../models/'
6
6
  import { evaluate } from '../expressions'
7
7
  import { RecordsRepositoryInterface } from '../../../../../../libs/common/domain/src/lib/repository/records-repository.interface'
8
8
 
@@ -15,17 +15,21 @@ export class EditorService {
15
15
  // returns the record as it was when saved, alongside its source
16
16
  saveRecord(
17
17
  record: CatalogRecord,
18
- fieldsConfig: EditorFieldsConfig,
18
+ fieldsConfig: EditorConfig,
19
19
  generateNewUniqueIdentifier = false
20
20
  ): Observable<[CatalogRecord, string]> {
21
21
  const savedRecord = { ...record }
22
22
 
23
+ const fields = fieldsConfig.pages.flatMap((page) =>
24
+ page.sections.flatMap((section) => section.fields)
25
+ )
26
+
23
27
  // run onSave processes
24
- for (const field of fieldsConfig) {
28
+ for (const field of fields) {
25
29
  if (field.onSaveProcess && field.model) {
26
30
  const evaluator = evaluate(field.onSaveProcess)
27
31
  savedRecord[field.model] = evaluator({
28
- config: field,
32
+ model: field.model,
29
33
  value: record[field.model],
30
34
  })
31
35
  }
@@ -220,11 +220,10 @@ export class MapUtilsService {
220
220
  if (!('spatialExtents' in record) || record.spatialExtents.length === 0) {
221
221
  return null
222
222
  }
223
- // transform an array of geojson geometries into a bbox
223
+ // extend all the spatial extents bbox into an including bbox
224
224
  const totalExtent = record.spatialExtents.reduce(
225
225
  (prev, curr) => {
226
- const geom = GEOJSON.readGeometry(curr.geometry)
227
- return extend(prev, geom.getExtent())
226
+ return extend(prev, curr.bbox)
228
227
  },
229
228
  [Infinity, Infinity, -Infinity, -Infinity]
230
229
  )
@@ -4,6 +4,7 @@
4
4
  [selectedRecordsIdentifiers]="selectedRecords$ | async"
5
5
  [sortOrder]="sortBy$ | async"
6
6
  (recordClick)="handleRecordClick($event)"
7
+ (duplicateRecord)="handleDuplicateRecord($event)"
7
8
  (recordsSelectedChange)="handleRecordsSelectedChange($event[0], $event[1])"
8
9
  (sortByChange)="handleSortByChange($event[0], $event[1])"
9
10
  ></gn-ui-results-table>
@@ -16,6 +16,7 @@ import { CommonModule } from '@angular/common'
16
16
  })
17
17
  export class ResultsTableContainerComponent {
18
18
  @Output() recordClick = new EventEmitter<CatalogRecord>()
19
+ @Output() duplicateRecord = new EventEmitter<CatalogRecord>()
19
20
 
20
21
  records$ = this.searchFacade.results$
21
22
  selectedRecords$ = this.selectionService.selectedRecordsIdentifiers$
@@ -35,6 +36,10 @@ export class ResultsTableContainerComponent {
35
36
  this.recordClick.emit(item as CatalogRecord)
36
37
  }
37
38
 
39
+ handleDuplicateRecord(item: unknown) {
40
+ this.duplicateRecord.emit(item as CatalogRecord)
41
+ }
42
+
38
43
  handleSortByChange(col: string, order: 'asc' | 'desc') {
39
44
  this.searchService.setSortBy([order, col])
40
45
  }