geonetwork-ui 2.8.0-dev.50da356b9 → 2.8.0-dev.5186798f7

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 (159) hide show
  1. package/esm2022/libs/api/metadata-converter/src/lib/dcat-ap/dcat-ap.converter.mjs +3 -3
  2. package/esm2022/libs/api/metadata-converter/src/lib/gn4/gn4.field.mapper.mjs +24 -1
  3. package/esm2022/libs/api/metadata-converter/src/lib/iso19139/iso19139.converter.mjs +7 -7
  4. package/esm2022/libs/api/metadata-converter/src/lib/iso19139/read-parts.mjs +3 -2
  5. package/esm2022/libs/api/metadata-converter/src/lib/iso19139/write-parts.mjs +4 -3
  6. package/esm2022/libs/api/repository/src/lib/gn4/elasticsearch/constant.mjs +1 -2
  7. package/esm2022/libs/common/domain/src/lib/model/dataviz/dataviz-configuration.model.mjs +1 -1
  8. package/esm2022/libs/common/domain/src/lib/model/record/metadata.model.mjs +1 -1
  9. package/esm2022/libs/feature/dataviz/src/lib/service/data.service.mjs +26 -8
  10. package/esm2022/libs/feature/editor/src/lib/components/record-form/form-field/form-field.component.mjs +23 -3
  11. package/esm2022/libs/feature/editor/src/lib/fields.config.mjs +2 -2
  12. package/esm2022/libs/feature/record/src/index.mjs +2 -1
  13. package/esm2022/libs/feature/record/src/lib/data-view-permalink/data-view-permalink.component.mjs +13 -4
  14. package/esm2022/libs/feature/record/src/lib/data-view-web-component/data-view-web-component.component.mjs +23 -8
  15. package/esm2022/libs/feature/record/src/lib/stac-view/stac-view.component.mjs +138 -0
  16. package/esm2022/libs/feature/record/src/lib/state/mdview.facade.mjs +14 -1
  17. package/esm2022/libs/feature/search/src/lib/constants.mjs +1 -2
  18. package/esm2022/libs/ui/elements/src/index.mjs +3 -1
  19. package/esm2022/libs/ui/elements/src/lib/downloads-list/downloads-list.component.mjs +1 -1
  20. package/esm2022/libs/ui/elements/src/lib/metadata-doi/metadata-doi.component.mjs +37 -0
  21. package/esm2022/libs/ui/elements/src/lib/metadata-info/metadata-info.component.mjs +5 -9
  22. package/esm2022/libs/ui/elements/src/lib/stac-items-result-grid/stac-items-result-grid.component.mjs +18 -0
  23. package/esm2022/libs/ui/elements/src/lib/user-feedback-item/user-feedback-item.component.mjs +5 -5
  24. package/esm2022/libs/ui/inputs/src/index.mjs +2 -1
  25. package/esm2022/libs/ui/inputs/src/lib/date-picker/date-picker.component.mjs +21 -7
  26. package/esm2022/libs/ui/inputs/src/lib/date-range-inputs/date-range-inputs.component.mjs +35 -0
  27. package/esm2022/libs/ui/layout/src/lib/paginable.interface.mjs +1 -1
  28. package/esm2022/libs/ui/layout/src/lib/previous-next-buttons/previous-next-buttons.component.mjs +13 -5
  29. package/esm2022/libs/util/i18n/src/lib/date-locales.mjs +33 -0
  30. package/esm2022/libs/util/shared/src/index.mjs +2 -1
  31. package/esm2022/libs/util/shared/src/lib/humanize-date.directive.mjs +33 -0
  32. package/esm2022/libs/util/shared/src/lib/services/date.service.mjs +19 -2
  33. package/esm2022/translations/de.json +14 -4
  34. package/esm2022/translations/en.json +14 -4
  35. package/esm2022/translations/es.json +10 -0
  36. package/esm2022/translations/fr.json +14 -4
  37. package/esm2022/translations/it.json +15 -5
  38. package/esm2022/translations/nl.json +10 -0
  39. package/esm2022/translations/pt.json +10 -0
  40. package/esm2022/translations/sk.json +11 -1
  41. package/fesm2022/geonetwork-ui-date-locales-MYnkDJ5h.mjs +35 -0
  42. package/fesm2022/geonetwork-ui-date-locales-MYnkDJ5h.mjs.map +1 -0
  43. package/fesm2022/geonetwork-ui.mjs +522 -151
  44. package/fesm2022/geonetwork-ui.mjs.map +1 -1
  45. package/libs/api/metadata-converter/src/lib/gn4/gn4.field.mapper.d.ts.map +1 -1
  46. package/libs/api/metadata-converter/src/lib/iso19139/iso19139.converter.d.ts.map +1 -1
  47. package/libs/api/metadata-converter/src/lib/iso19139/read-parts.d.ts +5 -1
  48. package/libs/api/metadata-converter/src/lib/iso19139/read-parts.d.ts.map +1 -1
  49. package/libs/api/metadata-converter/src/lib/iso19139/write-parts.d.ts.map +1 -1
  50. package/libs/api/repository/src/lib/gn4/elasticsearch/constant.d.ts.map +1 -1
  51. package/libs/common/domain/src/lib/model/dataviz/dataviz-configuration.model.d.ts +1 -1
  52. package/libs/common/domain/src/lib/model/dataviz/dataviz-configuration.model.d.ts.map +1 -1
  53. package/libs/common/domain/src/lib/model/record/metadata.model.d.ts +6 -1
  54. package/libs/common/domain/src/lib/model/record/metadata.model.d.ts.map +1 -1
  55. package/libs/feature/dataviz/src/lib/service/data.service.d.ts +2 -1
  56. package/libs/feature/dataviz/src/lib/service/data.service.d.ts.map +1 -1
  57. package/libs/feature/editor/src/lib/components/record-form/form-field/form-field.component.d.ts +2 -0
  58. package/libs/feature/editor/src/lib/components/record-form/form-field/form-field.component.d.ts.map +1 -1
  59. package/libs/feature/record/src/index.d.ts +1 -0
  60. package/libs/feature/record/src/index.d.ts.map +1 -1
  61. package/libs/feature/record/src/lib/data-view-permalink/data-view-permalink.component.d.ts +3 -2
  62. package/libs/feature/record/src/lib/data-view-permalink/data-view-permalink.component.d.ts.map +1 -1
  63. package/libs/feature/record/src/lib/data-view-web-component/data-view-web-component.component.d.ts +3 -2
  64. package/libs/feature/record/src/lib/data-view-web-component/data-view-web-component.component.d.ts.map +1 -1
  65. package/libs/feature/record/src/lib/stac-view/stac-view.component.d.ts +35 -0
  66. package/libs/feature/record/src/lib/stac-view/stac-view.component.d.ts.map +1 -0
  67. package/libs/feature/record/src/lib/state/mdview.facade.d.ts +28 -24
  68. package/libs/feature/record/src/lib/state/mdview.facade.d.ts.map +1 -1
  69. package/libs/feature/search/src/lib/constants.d.ts.map +1 -1
  70. package/libs/ui/elements/src/index.d.ts +2 -0
  71. package/libs/ui/elements/src/index.d.ts.map +1 -1
  72. package/libs/ui/elements/src/lib/metadata-doi/metadata-doi.component.d.ts +8 -0
  73. package/libs/ui/elements/src/lib/metadata-doi/metadata-doi.component.d.ts.map +1 -0
  74. package/libs/ui/elements/src/lib/metadata-info/metadata-info.component.d.ts +0 -2
  75. package/libs/ui/elements/src/lib/metadata-info/metadata-info.component.d.ts.map +1 -1
  76. package/libs/ui/elements/src/lib/stac-items-result-grid/stac-items-result-grid.component.d.ts +8 -0
  77. package/libs/ui/elements/src/lib/stac-items-result-grid/stac-items-result-grid.component.d.ts.map +1 -0
  78. package/libs/ui/inputs/src/index.d.ts +1 -0
  79. package/libs/ui/inputs/src/index.d.ts.map +1 -1
  80. package/libs/ui/inputs/src/lib/date-picker/date-picker.component.d.ts +5 -0
  81. package/libs/ui/inputs/src/lib/date-picker/date-picker.component.d.ts.map +1 -1
  82. package/libs/ui/inputs/src/lib/date-range-inputs/date-range-inputs.component.d.ts +12 -0
  83. package/libs/ui/inputs/src/lib/date-range-inputs/date-range-inputs.component.d.ts.map +1 -0
  84. package/libs/ui/layout/src/lib/paginable.interface.d.ts +3 -3
  85. package/libs/ui/layout/src/lib/paginable.interface.d.ts.map +1 -1
  86. package/libs/ui/layout/src/lib/previous-next-buttons/previous-next-buttons.component.d.ts +2 -1
  87. package/libs/ui/layout/src/lib/previous-next-buttons/previous-next-buttons.component.d.ts.map +1 -1
  88. package/libs/util/i18n/src/lib/date-locales.d.ts +5 -0
  89. package/libs/util/i18n/src/lib/date-locales.d.ts.map +1 -0
  90. package/libs/util/shared/src/index.d.ts +1 -0
  91. package/libs/util/shared/src/index.d.ts.map +1 -1
  92. package/libs/util/shared/src/lib/humanize-date.directive.d.ts +15 -0
  93. package/libs/util/shared/src/lib/humanize-date.directive.d.ts.map +1 -0
  94. package/libs/util/shared/src/lib/services/date.service.d.ts +4 -0
  95. package/libs/util/shared/src/lib/services/date.service.d.ts.map +1 -1
  96. package/package.json +1 -1
  97. package/src/libs/api/metadata-converter/src/lib/dcat-ap/dcat-ap.converter.ts +2 -2
  98. package/src/libs/api/metadata-converter/src/lib/fixtures/generic.records.ts +1 -1
  99. package/src/libs/api/metadata-converter/src/lib/fixtures/geo2france.records.service+eaux-usees.ts +1 -1
  100. package/src/libs/api/metadata-converter/src/lib/fixtures/geo2france.records.ts +5 -2
  101. package/src/libs/api/metadata-converter/src/lib/fixtures/geocat-ch.records.ts +1 -1
  102. package/src/libs/api/metadata-converter/src/lib/fixtures/georhena.records.ts +1 -1
  103. package/src/libs/api/metadata-converter/src/lib/fixtures/metawal.records.ts +2 -2
  104. package/src/libs/api/metadata-converter/src/lib/fixtures/wallonie.records.reuse.ts +1 -1
  105. package/src/libs/api/metadata-converter/src/lib/fixtures/wallonie.records.service+napitswallonia.ts +1 -1
  106. package/src/libs/api/metadata-converter/src/lib/gn4/gn4.field.mapper.ts +26 -0
  107. package/src/libs/api/metadata-converter/src/lib/iso19139/iso19139.converter.ts +13 -6
  108. package/src/libs/api/metadata-converter/src/lib/iso19139/read-parts.ts +6 -2
  109. package/src/libs/api/metadata-converter/src/lib/iso19139/write-parts.ts +4 -2
  110. package/src/libs/api/repository/src/lib/gn4/elasticsearch/constant.ts +0 -1
  111. package/src/libs/common/domain/src/lib/model/dataviz/dataviz-configuration.model.ts +1 -1
  112. package/src/libs/common/domain/src/lib/model/record/metadata.model.ts +8 -1
  113. package/src/libs/feature/dataviz/src/lib/service/data.service.ts +30 -5
  114. package/src/libs/feature/editor/src/lib/components/record-form/form-field/form-field.component.html +3 -3
  115. package/src/libs/feature/editor/src/lib/components/record-form/form-field/form-field.component.ts +30 -0
  116. package/src/libs/feature/editor/src/lib/fields.config.ts +1 -1
  117. package/src/libs/feature/record/src/index.ts +1 -0
  118. package/src/libs/feature/record/src/lib/data-view-permalink/data-view-permalink.component.ts +7 -1
  119. package/src/libs/feature/record/src/lib/data-view-web-component/data-view-web-component.component.ts +23 -4
  120. package/src/libs/feature/record/src/lib/stac-view/stac-view.component.css +3 -0
  121. package/src/libs/feature/record/src/lib/stac-view/stac-view.component.html +68 -0
  122. package/src/libs/feature/record/src/lib/stac-view/stac-view.component.ts +190 -0
  123. package/src/libs/feature/record/src/lib/state/mdview.facade.ts +30 -1
  124. package/src/libs/feature/search/src/lib/constants.ts +0 -1
  125. package/src/libs/ui/elements/src/index.ts +2 -0
  126. package/src/libs/ui/elements/src/lib/metadata-doi/metadata-doi.component.css +0 -0
  127. package/src/libs/ui/elements/src/lib/metadata-doi/metadata-doi.component.html +31 -0
  128. package/src/libs/ui/elements/src/lib/metadata-doi/metadata-doi.component.ts +30 -0
  129. package/src/libs/ui/elements/src/lib/metadata-info/metadata-info.component.html +67 -21
  130. package/src/libs/ui/elements/src/lib/metadata-info/metadata-info.component.ts +2 -9
  131. package/src/libs/ui/elements/src/lib/stac-items-result-grid/stac-items-result-grid.component.css +0 -0
  132. package/src/libs/ui/elements/src/lib/stac-items-result-grid/stac-items-result-grid.component.html +13 -0
  133. package/src/libs/ui/elements/src/lib/stac-items-result-grid/stac-items-result-grid.component.ts +15 -0
  134. package/src/libs/ui/elements/src/lib/user-feedback-item/user-feedback-item.component.html +1 -1
  135. package/src/libs/ui/elements/src/lib/user-feedback-item/user-feedback-item.component.ts +2 -2
  136. package/src/libs/ui/inputs/src/index.ts +1 -0
  137. package/src/libs/ui/inputs/src/lib/date-picker/date-picker.component.ts +17 -1
  138. package/src/libs/ui/inputs/src/lib/date-range-inputs/date-range-inputs.component.css +0 -0
  139. package/src/libs/ui/inputs/src/lib/date-range-inputs/date-range-inputs.component.html +15 -0
  140. package/src/libs/ui/inputs/src/lib/date-range-inputs/date-range-inputs.component.ts +41 -0
  141. package/src/libs/ui/layout/src/lib/paginable.interface.ts +3 -3
  142. package/src/libs/ui/layout/src/lib/previous-next-buttons/previous-next-buttons.component.html +12 -6
  143. package/src/libs/ui/layout/src/lib/previous-next-buttons/previous-next-buttons.component.ts +4 -1
  144. package/src/libs/util/i18n/src/lib/date-locales.ts +63 -0
  145. package/src/libs/util/shared/src/index.ts +1 -0
  146. package/src/libs/util/shared/src/lib/humanize-date.directive.ts +35 -0
  147. package/src/libs/util/shared/src/lib/services/date.service.ts +27 -1
  148. package/translations/de.json +14 -4
  149. package/translations/en.json +14 -4
  150. package/translations/es.json +10 -0
  151. package/translations/fr.json +14 -4
  152. package/translations/it.json +15 -5
  153. package/translations/nl.json +10 -0
  154. package/translations/pt.json +10 -0
  155. package/translations/sk.json +11 -1
  156. package/esm2022/libs/ui/elements/src/lib/user-feedback-item/time-since.pipe.mjs +0 -59
  157. package/libs/ui/elements/src/lib/user-feedback-item/time-since.pipe.d.ts +0 -11
  158. package/libs/ui/elements/src/lib/user-feedback-item/time-since.pipe.d.ts.map +0 -1
  159. package/src/libs/ui/elements/src/lib/user-feedback-item/time-since.pipe.ts +0 -54
@@ -0,0 +1,190 @@
1
+ import { CommonModule } from '@angular/common'
2
+ import { ChangeDetectionStrategy, Component, OnInit } from '@angular/core'
3
+ import {
4
+ DatasetRecord,
5
+ DatasetTemporalExtent,
6
+ } from '../../../../../../libs/common/domain/src/lib/model/record'
7
+ import { StacItemsResultGridComponent } from '../../../../../../libs/ui/elements/src'
8
+ import {
9
+ ButtonComponent,
10
+ DateRangeInputsComponent,
11
+ } from '../../../../../../libs/ui/inputs/src'
12
+ import { NgIconComponent, provideIcons } from '@ng-icons/core'
13
+ import { matDeleteOutline } from '@ng-icons/material-icons/outline'
14
+ import { TranslateDirective, TranslateService } from '@ngx-translate/core'
15
+ import { DataService } from '../../../../../../libs/feature/dataviz/src'
16
+ import {
17
+ BehaviorSubject,
18
+ catchError,
19
+ combineLatest,
20
+ debounceTime,
21
+ from,
22
+ map,
23
+ Observable,
24
+ of,
25
+ switchMap,
26
+ take,
27
+ tap,
28
+ } from 'rxjs'
29
+ import { GetCollectionItemsOptions, StacItem } from '@camptocamp/ogc-client'
30
+ import { MdViewFacade } from '../state'
31
+ import { PreviousNextButtonsComponent } from '../../../../../../libs/ui/layout/src'
32
+ import { FetchError } from '../../../../../../libs/util/data-fetcher/src'
33
+ import { PopupAlertComponent } from '../../../../../../libs/ui/widgets/src'
34
+
35
+ const STAC_ITEMS_PER_PAGE = 12
36
+ const DEBOUNCE_TIME_MS = 500
37
+
38
+ @Component({
39
+ selector: 'gn-ui-stac-view',
40
+ templateUrl: './stac-view.component.html',
41
+ styleUrls: ['./stac-view.component.css'],
42
+ changeDetection: ChangeDetectionStrategy.OnPush,
43
+ standalone: true,
44
+ imports: [
45
+ CommonModule,
46
+ NgIconComponent,
47
+ TranslateDirective,
48
+ StacItemsResultGridComponent,
49
+ DateRangeInputsComponent,
50
+ PreviousNextButtonsComponent,
51
+ PopupAlertComponent,
52
+ ButtonComponent,
53
+ ],
54
+ viewProviders: [provideIcons({ matDeleteOutline })],
55
+ })
56
+ export class StacViewComponent implements OnInit {
57
+ isFilterModified = false
58
+ error = null
59
+
60
+ initialTemporalExtent: DatasetTemporalExtent | null = null
61
+ currentTemporalExtent$ = new BehaviorSubject<DatasetTemporalExtent | null>(
62
+ null
63
+ )
64
+
65
+ initialPageUrl: string
66
+ previousPageUrl: string
67
+ nextPageUrl: string
68
+ currentPageUrl$ = new BehaviorSubject<string | null>(null)
69
+
70
+ items$: Observable<StacItem[]> = combineLatest([
71
+ this.currentPageUrl$,
72
+ this.currentTemporalExtent$,
73
+ ]).pipe(
74
+ debounceTime(DEBOUNCE_TIME_MS),
75
+ switchMap(([currentPageUrl, temporalExtent]) => {
76
+ this.error = null
77
+ const options: GetCollectionItemsOptions = {
78
+ limit: STAC_ITEMS_PER_PAGE,
79
+ }
80
+ if (temporalExtent && (temporalExtent.start || temporalExtent.end)) {
81
+ options.datetime = {
82
+ ...(temporalExtent.start && { start: temporalExtent.start }),
83
+ ...(temporalExtent.end && { end: temporalExtent.end }),
84
+ }
85
+ }
86
+ return from(
87
+ this.dataService.getItemsFromStacApi(currentPageUrl, options)
88
+ ).pipe(
89
+ tap((stacDocument) => {
90
+ this.previousPageUrl =
91
+ stacDocument.links.find((link) => link.rel === 'previous')?.href ||
92
+ null
93
+ this.nextPageUrl =
94
+ stacDocument.links.find((link) => link.rel === 'next')?.href || null
95
+ }),
96
+ map((stacDocument) => stacDocument.features),
97
+ catchError((err) => {
98
+ this.handleError(err)
99
+ return of([])
100
+ })
101
+ )
102
+ })
103
+ )
104
+
105
+ constructor(
106
+ private dataService: DataService,
107
+ private metadataViewFacade: MdViewFacade,
108
+ private translateService: TranslateService
109
+ ) {}
110
+
111
+ ngOnInit() {
112
+ this.metadataViewFacade.metadata$
113
+ .pipe(
114
+ take(1),
115
+ map((metadata) => {
116
+ const temporalExtents =
117
+ metadata?.kind === 'dataset'
118
+ ? (metadata as DatasetRecord).temporalExtents
119
+ : []
120
+
121
+ return temporalExtents.length > 0
122
+ ? temporalExtents[0]
123
+ : ({
124
+ start: null,
125
+ end: null,
126
+ } as DatasetTemporalExtent)
127
+ })
128
+ )
129
+ .subscribe((extent) => {
130
+ this.initialTemporalExtent = extent
131
+ this.currentTemporalExtent$.next(extent)
132
+ })
133
+
134
+ this.metadataViewFacade.stacLinks$
135
+ .pipe(
136
+ take(1),
137
+ map((links) => (links && links.length > 0 ? links[0] : null))
138
+ )
139
+ .subscribe((link) => {
140
+ if (link) {
141
+ this.initialPageUrl = link.url.href
142
+ this.currentPageUrl$.next(link.url.href)
143
+ }
144
+ })
145
+ }
146
+
147
+ onTemporalExtentChange(extent: DatasetTemporalExtent | null) {
148
+ this.currentTemporalExtent$.next(extent)
149
+ // make sure to use url without pagination token when temporal filter changes
150
+ this.currentPageUrl$.next(this.initialPageUrl)
151
+ this.isFilterModified = true
152
+ }
153
+
154
+ onResetFilters() {
155
+ this.currentTemporalExtent$.next(this.initialTemporalExtent)
156
+ this.isFilterModified = false
157
+ }
158
+
159
+ handleError(error: FetchError | Error | string) {
160
+ if (error instanceof FetchError) {
161
+ this.error = this.translateService.instant(
162
+ `dataset.error.${error.type}`,
163
+ {
164
+ info: error.info,
165
+ }
166
+ )
167
+ console.warn(error.message)
168
+ } else if (error instanceof Error) {
169
+ this.error = this.translateService.instant(error.message)
170
+ console.warn(error.stack || error)
171
+ } else {
172
+ this.error = this.translateService.instant(error)
173
+ console.warn(error)
174
+ }
175
+ }
176
+
177
+ // Paginable API
178
+ get isFirstPage() {
179
+ return this.previousPageUrl == null
180
+ }
181
+ get isLastPage() {
182
+ return this.nextPageUrl == null
183
+ }
184
+ goToNextPage() {
185
+ this.currentPageUrl$.next(this.nextPageUrl)
186
+ }
187
+ goToPrevPage() {
188
+ this.currentPageUrl$.next(this.previousPageUrl)
189
+ }
190
+ }
@@ -20,7 +20,7 @@ import {
20
20
  } from '../../../../../../libs/common/domain/src/lib/model/record'
21
21
  import { AvatarServiceInterface } from '../../../../../../libs/api/repository/src'
22
22
  import { OgcApiRecord } from '@camptocamp/ogc-client'
23
- import { from, of, Observable } from 'rxjs'
23
+ import { from, of } from 'rxjs'
24
24
  import { DataService } from '../../../../../../libs/feature/dataviz/src'
25
25
 
26
26
  @Injectable()
@@ -89,6 +89,25 @@ export class MdViewFacade {
89
89
  shareReplay(1)
90
90
  )
91
91
 
92
+ resourceDoi$ = this.metadata$.pipe(
93
+ map((record) => {
94
+ if (!record?.resourceIdentifiers?.length) return null
95
+ const doiIdentifier = record.resourceIdentifiers.find(
96
+ (id) =>
97
+ id.codeSpace?.toLowerCase().includes('doi.org') ||
98
+ id.code.startsWith('10.')
99
+ )
100
+
101
+ if (!doiIdentifier) return null
102
+
103
+ return {
104
+ code: doiIdentifier.code,
105
+ url: doiIdentifier.url ? doiIdentifier.url : null,
106
+ }
107
+ }),
108
+ shareReplay(1)
109
+ )
110
+
92
111
  apiLinks$ = this.allLinks$.pipe(
93
112
  map((links) =>
94
113
  links.filter((link) => this.linkClassifier.hasUsage(link, LinkUsage.API))
@@ -105,6 +124,16 @@ export class MdViewFacade {
105
124
  shareReplay(1)
106
125
  )
107
126
 
127
+ stacLinks$ = this.allLinks$.pipe(
128
+ map((links) =>
129
+ links.filter(
130
+ (link) =>
131
+ link.type === 'service' && link.accessServiceProtocol === 'stac'
132
+ )
133
+ ),
134
+ shareReplay(1)
135
+ )
136
+
108
137
  downloadLinks$ = this.allLinks$.pipe(
109
138
  map((links) =>
110
139
  links.filter((link) =>
@@ -11,7 +11,6 @@ export const FIELDS_SUMMARY: FieldName[] = [
11
11
  'resourceAbstractObject',
12
12
  'overview',
13
13
  'logo',
14
- 'codelist_status_text',
15
14
  'link',
16
15
  'linkProtocol',
17
16
  'contactForResource*.organisation*',
@@ -14,6 +14,7 @@ export * from './lib/markdown-editor/markdown-editor.component'
14
14
  export * from './lib/markdown-parser/markdown-parser.component'
15
15
  export * from './lib/metadata-catalog/metadata-catalog.component'
16
16
  export * from './lib/metadata-contact/metadata-contact.component'
17
+ export * from './lib/metadata-doi/metadata-doi.component'
17
18
  export * from './lib/metadata-info/metadata-info.component'
18
19
  export * from './lib/metadata-quality-item/metadata-quality-item.component'
19
20
  export * from './lib/metadata-quality/metadata-quality.component'
@@ -26,3 +27,4 @@ export * from './lib/application-banner/application-banner.component'
26
27
  export * from './lib/internal-link-card/internal-link-card.component'
27
28
  export * from './lib/service-capabilities/service-capabilities.component'
28
29
  export * from './lib/record-feature-catalog/feature-catalog-list/feature-catalog-list.component'
30
+ export * from './lib/stac-items-result-grid/stac-items-result-grid.component'
@@ -0,0 +1,31 @@
1
+ <div
2
+ class="border border-gray-300 rounded-lg py-4 px-5 text-black flex justify-between items-center gap-4"
3
+ >
4
+ <div class="overflow-hidden flex-1">
5
+ <p class="text-base font-medium mb-3">DOI</p>
6
+ <p
7
+ class="text-base font-medium overflow-hidden text-ellipsis whitespace-nowrap"
8
+ [title]="code"
9
+ >
10
+ {{ code }}
11
+ </p>
12
+ </div>
13
+ <div class="flex gap-2 items-start">
14
+ <gn-ui-copy-text-button
15
+ [text]="code"
16
+ [displayText]="false"
17
+ [tooltipText]="'record.metadata.doi.copy' | translate"
18
+ class="[&>div]:flex [&>div]:items-center [&>div]:justify-center [&_button]:w-[40px] [&_button]:h-[32px] [&_button]:flex [&_button]:items-center [&_button]:justify-center [&_button]:hover:bg-gray-100 [&_button]:rounded-lg [&_button]:transition-colors [&_button]:border [&_button]:border-gray-300 [&_button]:px-2 [&_button]:py-1 [&_ng-icon]:w-5 [&_ng-icon]:h-5"
19
+ ></gn-ui-copy-text-button>
20
+ <a
21
+ *ngIf="link"
22
+ [href]="link"
23
+ target="_blank"
24
+ rel="noopener noreferrer"
25
+ class="w-[40px] h-[32px] flex items-center justify-center hover:bg-gray-100 rounded-lg transition-colors border border-gray-300 px-2 py-1"
26
+ [matTooltip]="'record.metadata.doi.open' | translate"
27
+ >
28
+ <ng-icon name="matOpenInNew" size="20"></ng-icon>
29
+ </a>
30
+ </div>
31
+ </div>
@@ -0,0 +1,30 @@
1
+ import { Component, Input } from '@angular/core'
2
+ import { CommonModule } from '@angular/common'
3
+ import { NgIcon, provideIcons } from '@ng-icons/core'
4
+ import { MatTooltipModule } from '@angular/material/tooltip'
5
+ import { matOpenInNew } from '@ng-icons/material-icons/baseline'
6
+ import { TranslatePipe } from '@ngx-translate/core'
7
+ import { CopyTextButtonComponent } from '../../../../../../libs/ui/inputs/src'
8
+
9
+ @Component({
10
+ selector: 'gn-ui-metadata-doi',
11
+ standalone: true,
12
+ imports: [
13
+ CommonModule,
14
+ MatTooltipModule,
15
+ NgIcon,
16
+ TranslatePipe,
17
+ CopyTextButtonComponent,
18
+ ],
19
+ templateUrl: './metadata-doi.component.html',
20
+ styleUrl: './metadata-doi.component.css',
21
+ viewProviders: [
22
+ provideIcons({
23
+ matOpenInNew,
24
+ }),
25
+ ],
26
+ })
27
+ export class MetadataDoiComponent {
28
+ @Input() code!: string
29
+ @Input() link?: string
30
+ }
@@ -3,7 +3,11 @@
3
3
  ghostClass="h-[178px]"
4
4
  [showContent]="fieldReady('abstract')"
5
5
  >
6
- <gn-ui-max-lines [maxLines]="6" *ngIf="metadata.abstract">
6
+ <gn-ui-max-lines
7
+ [maxLines]="6"
8
+ *ngIf="metadata.abstract"
9
+ data-test="metadata-info-abstract"
10
+ >
7
11
  <div class="mb-6">
8
12
  <gn-ui-markdown-parser
9
13
  [textContent]="metadata.abstract"
@@ -101,19 +105,31 @@
101
105
  <gn-ui-expandable-panel
102
106
  *ngIf="
103
107
  (metadata.kind === 'dataset' && metadata.lineage) ||
108
+ resourceContact ||
109
+ metadata.resourceCreated ||
110
+ metadata.resourcePublished ||
104
111
  metadata.resourceUpdated ||
105
- metadata.updateFrequency ||
106
- (metadata.kind === 'dataset' && metadata.status)
112
+ (metadata.kind === 'dataset' && metadata.updateFrequency) ||
113
+ metadata.otherLanguages?.length ||
114
+ (metadata.kind === 'dataset' && temporalExtent)
107
115
  "
108
116
  [title]="'record.metadata.details' | translate"
109
117
  data-test="details-panel"
110
118
  >
111
- <div *ngIf="metadata.lineage" class="text-gray-900 flex flex-col mt-4 gap-2">
119
+ <div
120
+ *ngIf="metadata.kind === 'dataset' && metadata.lineage"
121
+ class="text-gray-900 flex flex-col mt-4 gap-2"
122
+ data-test="details-panel-lineage"
123
+ >
112
124
  <p class="whitespace-pre-line break-words text-gray-900" gnUiLinkify>
113
125
  {{ metadata.lineage }}
114
126
  </p>
115
127
  </div>
116
- <div class="flex flex-row gap-6 mt-5 mb-8" *ngIf="resourceContact">
128
+ <div
129
+ class="flex flex-row gap-6 mt-5 mb-8 resource-contact"
130
+ *ngIf="resourceContact"
131
+ data-test="details-panel-resource-contact"
132
+ >
117
133
  <div
118
134
  *ngIf="resourceContact.organization?.logoUrl?.href"
119
135
  class="flex items-center justify-center border-solid border border-gray-300 rounded-md bg-white h-32 overflow-hidden"
@@ -165,19 +181,40 @@
165
181
  <div
166
182
  class="py-6 px-6 rounded bg-gray-100 grid grid-cols-2 gap-y-6 gap-x-[20px] text-gray-700"
167
183
  >
168
- <div *ngIf="metadata.resourceCreated">
184
+ <div
185
+ *ngIf="metadata.resourceCreated"
186
+ data-test="details-panel-resource-created"
187
+ >
169
188
  <p class="text-sm" translate>record.metadata.creation</p>
170
- <p class="text-primary font-medium mt-1">
171
- {{ formatDate(metadata.resourceCreated) }}
172
- </p>
189
+ <p
190
+ class="text-primary font-medium mt-1 resource-created"
191
+ [gnUiHumanizeDate]="metadata.resourceCreated"
192
+ ></p>
173
193
  </div>
174
- <div *ngIf="metadata.resourcePublished">
194
+ <div
195
+ *ngIf="metadata.resourcePublished"
196
+ data-test="details-panel-resource-published"
197
+ >
175
198
  <p class="text-sm" translate>record.metadata.publication</p>
176
- <p class="text-primary font-medium mt-1">
177
- {{ formatDate(metadata.resourcePublished) }}
178
- </p>
199
+ <p
200
+ class="text-primary font-medium mt-1 resource-published"
201
+ [gnUiHumanizeDate]="metadata.resourcePublished"
202
+ ></p>
203
+ </div>
204
+ <div
205
+ *ngIf="metadata.resourceUpdated"
206
+ data-test="details-panel-resource-updated"
207
+ >
208
+ <p class="text-sm" translate>record.metadata.update</p>
209
+ <p
210
+ class="text-primary font-medium mt-1 resource-updated"
211
+ [gnUiHumanizeDate]="metadata.resourceUpdated"
212
+ ></p>
179
213
  </div>
180
- <div *ngIf="updateFrequency">
214
+ <div
215
+ *ngIf="metadata.kind === 'dataset' && metadata.updateFrequency"
216
+ data-test="details-panel-update-frequency"
217
+ >
181
218
  <p class="text-sm" translate>record.metadata.updateFrequency</p>
182
219
  <p
183
220
  class="text-primary font-medium mt-1 updateFrequency"
@@ -187,11 +224,14 @@
187
224
  {{ updateFrequency }}
188
225
  </p>
189
226
  </div>
190
- <div *ngIf="metadata.otherLanguages?.length">
227
+ <div
228
+ *ngIf="metadata.otherLanguages?.length"
229
+ data-test="details-panel-other-languages"
230
+ >
191
231
  <p class="text-sm mb-1" translate>record.metadata.languages</p>
192
232
  <div class="flex flex-row gap-1 flex-wrap">
193
233
  <p
194
- class="text-primary font-medium"
234
+ class="text-primary font-medium other-languages"
195
235
  translate
196
236
  *ngFor="let language of metadata.otherLanguages"
197
237
  >
@@ -199,9 +239,14 @@
199
239
  </p>
200
240
  </div>
201
241
  </div>
202
- <div *ngIf="temporalExtent">
242
+ <div
243
+ *ngIf="metadata.kind === 'dataset' && temporalExtent"
244
+ data-test="details-panel-temporal-extent"
245
+ >
203
246
  <p class="text-sm" translate>record.metadata.temporalExtent</p>
204
- <div class="flex flex-row gap-1 mb-1 text-primary font-medium">
247
+ <div
248
+ class="flex flex-row gap-1 mb-1 text-primary font-medium temporal-extent"
249
+ >
205
250
  <p
206
251
  *ngIf="temporalExtent.start && temporalExtent.end"
207
252
  translate
@@ -252,9 +297,10 @@
252
297
  <div class="flex flex-col gap-4 mr-4 py-5 rounded text-gray-700">
253
298
  <div *ngIf="metadata.recordUpdated">
254
299
  <p class="text-sm" translate>record.metadata.updatedOn</p>
255
- <p class="text-primary font-medium">
256
- {{ metadata.recordUpdated && formatDateTime(metadata.recordUpdated) }}
257
- </p>
300
+ <p
301
+ class="text-primary font-medium"
302
+ [gnUiHumanizeDate]="metadata.recordUpdated"
303
+ ></p>
258
304
  </div>
259
305
  <div *ngIf="metadata.landingPage">
260
306
  <p class="text-sm" translate>record.metadata.sheet</p>
@@ -26,7 +26,7 @@ import { matOpenInNew } from '@ng-icons/material-icons/baseline'
26
26
  import { matMailOutline } from '@ng-icons/material-icons/outline'
27
27
  import { ThumbnailComponent } from '../thumbnail/thumbnail.component'
28
28
  import { GnUiLinkifyDirective } from './linkify.directive'
29
-
29
+ import { GnUiHumanizeDateDirective } from '../../../../../../libs/util/shared/src'
30
30
  import { CommonModule } from '@angular/common'
31
31
  import { SpatialExtentComponent } from '../../../../../../libs/ui/map/src'
32
32
 
@@ -49,6 +49,7 @@ import { SpatialExtentComponent } from '../../../../../../libs/ui/map/src'
49
49
  CopyTextButtonComponent,
50
50
  NgIcon,
51
51
  GnUiLinkifyDirective,
52
+ GnUiHumanizeDateDirective,
52
53
  SpatialExtentComponent,
53
54
  ],
54
55
  viewProviders: [
@@ -146,12 +147,4 @@ export class MetadataInfoComponent {
146
147
  onKeywordClick(keyword: Keyword) {
147
148
  this.keyword.emit(keyword)
148
149
  }
149
-
150
- formatDate(date: Date): string {
151
- return this.dateService.formatDate(date)
152
- }
153
-
154
- formatDateTime(date: Date): string {
155
- return this.dateService.formatDateTime(date)
156
- }
157
150
  }
@@ -0,0 +1,13 @@
1
+ <div class="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 gap-[10px]">
2
+ <div *ngFor="let item of items">
3
+ <div
4
+ data-cy="stac-item-card"
5
+ class="h-[173px] flex flex-col justify-between gap-4 bg-white rounded-lg border border-gray-300 px-5 py-3 mx-auto"
6
+ >
7
+ <h2 class="text-xl text-black line-clamp-4 break-words">
8
+ {{ item.id }}
9
+ </h2>
10
+ <p class="truncate">{{ item.properties.datetime }}</p>
11
+ </div>
12
+ </div>
13
+ </div>
@@ -0,0 +1,15 @@
1
+ import { CommonModule } from '@angular/common'
2
+ import { ChangeDetectionStrategy, Component, Input } from '@angular/core'
3
+ import { StacItem } from '@camptocamp/ogc-client'
4
+
5
+ @Component({
6
+ selector: 'gn-ui-stac-items-result-grid',
7
+ standalone: true,
8
+ imports: [CommonModule],
9
+ templateUrl: './stac-items-result-grid.component.html',
10
+ styleUrl: './stac-items-result-grid.component.css',
11
+ changeDetection: ChangeDetectionStrategy.OnPush,
12
+ })
13
+ export class StacItemsResultGridComponent {
14
+ @Input() items: StacItem[] = []
15
+ }
@@ -13,7 +13,7 @@
13
13
  </div>
14
14
  <div class="p-4 flex flex-col">
15
15
  <span>{{ userFeedbackParent.authorName }}</span>
16
- <span> {{ userFeedbackParent.date | timeSince }}</span>
16
+ <span [gnUiHumanizeDate]="userFeedbackParent.date"></span>
17
17
  </div>
18
18
  </div>
19
19
  <div data-cy="commentText" class="mt-4 whitespace-pre-line">
@@ -11,13 +11,13 @@ import {
11
11
  UserFeedbackViewModel,
12
12
  } from '../../../../../../libs/common/domain/src/lib/model/record'
13
13
  import { UserModel } from '../../../../../../libs/common/domain/src/lib/model/user'
14
- import { TimeSincePipe } from './time-since.pipe'
15
14
  import { CommonModule } from '@angular/common'
16
15
  import { ButtonComponent, TextAreaComponent } from '../../../../../../libs/ui/inputs/src'
17
16
  import { TranslatePipe } from '@ngx-translate/core'
18
17
  import { SpinningLoaderComponent } from '../../../../../../libs/ui/widgets/src'
19
18
  import { NgIcon, provideIcons } from '@ng-icons/core'
20
19
  import { matSendOutline } from '@ng-icons/material-icons/outline'
20
+ import { GnUiHumanizeDateDirective } from '../../../../../../libs/util/shared/src'
21
21
 
22
22
  @Component({
23
23
  selector: 'gn-ui-user-feedback-item',
@@ -27,11 +27,11 @@ import { matSendOutline } from '@ng-icons/material-icons/outline'
27
27
  standalone: true,
28
28
  imports: [
29
29
  CommonModule,
30
- TimeSincePipe,
31
30
  TextAreaComponent,
32
31
  TranslatePipe,
33
32
  ButtonComponent,
34
33
  SpinningLoaderComponent,
34
+ GnUiHumanizeDateDirective,
35
35
  NgIcon,
36
36
  ],
37
37
  viewProviders: [
@@ -24,3 +24,4 @@ export * from './lib/text-input/text-input.component'
24
24
  export * from './lib/url-input/url-input.component'
25
25
  export * from './lib/viewport-intersector/viewport-intersector.component'
26
26
  export * from './lib/search-feature-catalog/search-feature-catalog.component'
27
+ export * from './lib/date-range-inputs/date-range-inputs.component'
@@ -5,7 +5,11 @@ import {
5
5
  Input,
6
6
  Output,
7
7
  } from '@angular/core'
8
- import { MatNativeDateModule } from '@angular/material/core'
8
+ import {
9
+ DateAdapter,
10
+ MAT_DATE_LOCALE,
11
+ MatNativeDateModule,
12
+ } from '@angular/material/core'
9
13
  import { MatDatepickerModule } from '@angular/material/datepicker'
10
14
  import { ButtonComponent } from '../button/button.component'
11
15
  import {
@@ -14,6 +18,7 @@ import {
14
18
  provideNgIconsConfig,
15
19
  } from '@ng-icons/core'
16
20
  import { iconoirCalendar } from '@ng-icons/iconoir'
21
+ import { TranslateService } from '@ngx-translate/core'
17
22
 
18
23
  @Component({
19
24
  selector: 'gn-ui-date-picker',
@@ -32,9 +37,20 @@ import { iconoirCalendar } from '@ng-icons/iconoir'
32
37
  provideNgIconsConfig({
33
38
  size: '1.5rem',
34
39
  }),
40
+ {
41
+ provide: MAT_DATE_LOCALE,
42
+ useFactory: (locale: string) => locale,
43
+ },
35
44
  ],
36
45
  })
37
46
  export class DatePickerComponent {
38
47
  @Input() date: Date
39
48
  @Output() dateChange = new EventEmitter<Date>()
49
+
50
+ constructor(
51
+ private dateAdapter: DateAdapter<Date>,
52
+ private translate: TranslateService
53
+ ) {
54
+ this.dateAdapter.setLocale(this.translate.currentLang)
55
+ }
40
56
  }
@@ -0,0 +1,15 @@
1
+ <div class="m-8">
2
+ <p class="mb-4" translate>daterange.filter.period</p>
3
+ <p translate>daterange.filter.from</p>
4
+ <gn-ui-date-picker
5
+ id="start-date-picker"
6
+ [date]="temporalExtent?.start"
7
+ (dateChange)="onStartDateChange($event)"
8
+ ></gn-ui-date-picker>
9
+ <p class="mt-4" translate>daterange.filter.to</p>
10
+ <gn-ui-date-picker
11
+ id="end-date-picker"
12
+ [date]="temporalExtent?.end"
13
+ (dateChange)="onEndDateChange($event)"
14
+ ></gn-ui-date-picker>
15
+ </div>