geonetwork-ui 2.6.0-dev.9d3ad45e2 → 2.6.0-dev.a0bd52a21

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 (183) hide show
  1. package/esm2022/libs/api/metadata-converter/src/lib/gn4/gn4.field.mapper.mjs +39 -4
  2. package/esm2022/libs/api/metadata-converter/src/lib/gn4/types/metadata.model.mjs +1 -1
  3. package/esm2022/libs/api/metadata-converter/src/lib/iso19115-3/read-parts.mjs +8 -4
  4. package/esm2022/libs/api/metadata-converter/src/lib/iso19115-3/write-parts.mjs +5 -2
  5. package/esm2022/libs/api/metadata-converter/src/lib/iso19139/read-parts.mjs +3 -3
  6. package/esm2022/libs/api/metadata-converter/src/lib/iso19139/write-parts.mjs +2 -2
  7. package/esm2022/libs/api/repository/src/lib/gn4/elasticsearch/constant.mjs +2 -1
  8. package/esm2022/libs/api/repository/src/lib/gn4/elasticsearch/elasticsearch.service.mjs +3 -3
  9. package/esm2022/libs/api/repository/src/lib/gn4/gn4-repository.mjs +75 -15
  10. package/esm2022/libs/api/repository/src/lib/gn4/platform/gn4-platform.service.mjs +7 -1
  11. package/esm2022/libs/api/repository/src/lib/gn4/settings/gn4-settings.service.mjs +2 -1
  12. package/esm2022/libs/common/domain/src/lib/model/record/metadata.model.mjs +1 -1
  13. package/esm2022/libs/common/domain/src/lib/repository/records-repository.interface.mjs +1 -1
  14. package/esm2022/libs/feature/editor/src/lib/components/multilingual-panel/multilingual-panel.component.mjs +266 -11
  15. package/esm2022/libs/feature/map/src/lib/add-layer-from-catalog/add-layer-from-catalog.component.mjs +1 -1
  16. package/esm2022/libs/feature/record/src/lib/state/mdview.actions.mjs +3 -1
  17. package/esm2022/libs/feature/record/src/lib/state/mdview.effects.mjs +7 -1
  18. package/esm2022/libs/feature/record/src/lib/state/mdview.facade.mjs +3 -1
  19. package/esm2022/libs/feature/record/src/lib/state/mdview.reducer.mjs +7 -1
  20. package/esm2022/libs/feature/record/src/lib/state/mdview.selectors.mjs +3 -1
  21. package/esm2022/libs/feature/search/src/lib/constants.mjs +3 -2
  22. package/esm2022/libs/feature/search/src/lib/fuzzy-search/fuzzy-search.component.mjs +5 -3
  23. package/esm2022/libs/feature/search/src/lib/results-table/results-table-container.component.mjs +13 -10
  24. package/esm2022/libs/ui/elements/src/lib/internal-link-card/internal-link-card.component.mjs +19 -34
  25. package/esm2022/libs/ui/elements/src/lib/internal-link-card-contact/internal-link-card-contact.component.mjs +59 -0
  26. package/esm2022/libs/ui/elements/src/lib/metadata-quality/metadata-quality.component.mjs +35 -10
  27. package/esm2022/libs/ui/elements/src/lib/metadata-quality-item/metadata-quality-item.component.mjs +5 -1
  28. package/esm2022/libs/ui/elements/src/lib/record-feature-catalog/feature-catalog-list/feature-catalog-list.component.mjs +50 -29
  29. package/esm2022/libs/ui/elements/src/lib/ui-elements.module.mjs +7 -3
  30. package/esm2022/libs/ui/inputs/src/lib/autocomplete/autocomplete.component.mjs +3 -3
  31. package/esm2022/libs/ui/layout/src/index.mjs +2 -1
  32. package/esm2022/libs/ui/layout/src/lib/cell-popin/cell-popin.component.mjs +110 -0
  33. package/esm2022/libs/ui/layout/src/lib/interactive-table/interactive-table.component.mjs +20 -10
  34. package/esm2022/libs/ui/layout/src/lib/truncated-text/truncated-text.component.mjs +15 -49
  35. package/esm2022/libs/ui/layout/src/lib/ui-layout.module.mjs +2 -2
  36. package/esm2022/libs/ui/search/src/index.mjs +2 -1
  37. package/esm2022/libs/ui/search/src/lib/results-table/action-menu/action-menu.component.mjs +21 -12
  38. package/esm2022/libs/ui/search/src/lib/results-table/results-table.component.mjs +39 -18
  39. package/esm2022/libs/ui/search/src/lib/ui-search.module.mjs +10 -4
  40. package/esm2022/libs/util/i18n/src/lib/i18n.constants.mjs +42 -1
  41. package/esm2022/libs/util/i18n/src/lib/language-codes.mjs +24 -2
  42. package/esm2022/translations/de.json +41 -6
  43. package/esm2022/translations/en.json +41 -6
  44. package/esm2022/translations/es.json +40 -5
  45. package/esm2022/translations/fr.json +41 -6
  46. package/esm2022/translations/it.json +40 -5
  47. package/esm2022/translations/nl.json +40 -5
  48. package/esm2022/translations/pt.json +40 -5
  49. package/fesm2022/geonetwork-ui.mjs +1591 -719
  50. package/fesm2022/geonetwork-ui.mjs.map +1 -1
  51. package/libs/api/metadata-converter/src/lib/gn4/gn4.field.mapper.d.ts.map +1 -1
  52. package/libs/api/metadata-converter/src/lib/gn4/types/metadata.model.d.ts +0 -1
  53. package/libs/api/metadata-converter/src/lib/gn4/types/metadata.model.d.ts.map +1 -1
  54. package/libs/api/metadata-converter/src/lib/iso19115-3/read-parts.d.ts +1 -0
  55. package/libs/api/metadata-converter/src/lib/iso19115-3/read-parts.d.ts.map +1 -1
  56. package/libs/api/metadata-converter/src/lib/iso19115-3/write-parts.d.ts.map +1 -1
  57. package/libs/api/metadata-converter/src/lib/iso19139/read-parts.d.ts.map +1 -1
  58. package/libs/api/metadata-converter/src/lib/iso19139/write-parts.d.ts.map +1 -1
  59. package/libs/api/repository/src/lib/gn4/elasticsearch/constant.d.ts.map +1 -1
  60. package/libs/api/repository/src/lib/gn4/elasticsearch/elasticsearch.service.d.ts +1 -1
  61. package/libs/api/repository/src/lib/gn4/elasticsearch/elasticsearch.service.d.ts.map +1 -1
  62. package/libs/api/repository/src/lib/gn4/gn4-repository.d.ts +13 -2
  63. package/libs/api/repository/src/lib/gn4/gn4-repository.d.ts.map +1 -1
  64. package/libs/api/repository/src/lib/gn4/platform/gn4-platform.service.d.ts +2 -0
  65. package/libs/api/repository/src/lib/gn4/platform/gn4-platform.service.d.ts.map +1 -1
  66. package/libs/api/repository/src/lib/gn4/settings/gn4-settings.service.d.ts +1 -0
  67. package/libs/api/repository/src/lib/gn4/settings/gn4-settings.service.d.ts.map +1 -1
  68. package/libs/common/domain/src/lib/model/record/metadata.model.d.ts +20 -17
  69. package/libs/common/domain/src/lib/model/record/metadata.model.d.ts.map +1 -1
  70. package/libs/common/domain/src/lib/repository/records-repository.interface.d.ts +6 -0
  71. package/libs/common/domain/src/lib/repository/records-repository.interface.d.ts.map +1 -1
  72. package/libs/feature/editor/src/lib/components/multilingual-panel/multilingual-panel.component.d.ts +48 -3
  73. package/libs/feature/editor/src/lib/components/multilingual-panel/multilingual-panel.component.d.ts.map +1 -1
  74. package/libs/feature/record/src/lib/state/mdview.actions.d.ts +10 -0
  75. package/libs/feature/record/src/lib/state/mdview.actions.d.ts.map +1 -1
  76. package/libs/feature/record/src/lib/state/mdview.effects.d.ts +6 -0
  77. package/libs/feature/record/src/lib/state/mdview.effects.d.ts.map +1 -1
  78. package/libs/feature/record/src/lib/state/mdview.facade.d.ts +2 -0
  79. package/libs/feature/record/src/lib/state/mdview.facade.d.ts.map +1 -1
  80. package/libs/feature/record/src/lib/state/mdview.reducer.d.ts +2 -0
  81. package/libs/feature/record/src/lib/state/mdview.reducer.d.ts.map +1 -1
  82. package/libs/feature/record/src/lib/state/mdview.selectors.d.ts +2 -0
  83. package/libs/feature/record/src/lib/state/mdview.selectors.d.ts.map +1 -1
  84. package/libs/feature/search/src/lib/constants.d.ts.map +1 -1
  85. package/libs/feature/search/src/lib/fuzzy-search/fuzzy-search.component.d.ts +2 -1
  86. package/libs/feature/search/src/lib/fuzzy-search/fuzzy-search.component.d.ts.map +1 -1
  87. package/libs/feature/search/src/lib/results-table/results-table-container.component.d.ts +6 -5
  88. package/libs/feature/search/src/lib/results-table/results-table-container.component.d.ts.map +1 -1
  89. package/libs/ui/elements/src/lib/internal-link-card/internal-link-card.component.d.ts +3 -6
  90. package/libs/ui/elements/src/lib/internal-link-card/internal-link-card.component.d.ts.map +1 -1
  91. package/libs/ui/elements/src/lib/internal-link-card-contact/internal-link-card-contact.component.d.ts +14 -0
  92. package/libs/ui/elements/src/lib/internal-link-card-contact/internal-link-card-contact.component.d.ts.map +1 -0
  93. package/libs/ui/elements/src/lib/metadata-quality/metadata-quality.component.d.ts +3 -0
  94. package/libs/ui/elements/src/lib/metadata-quality/metadata-quality.component.d.ts.map +1 -1
  95. package/libs/ui/elements/src/lib/metadata-quality-item/metadata-quality-item.component.d.ts.map +1 -1
  96. package/libs/ui/elements/src/lib/record-feature-catalog/feature-catalog-list/feature-catalog-list.component.d.ts +20 -9
  97. package/libs/ui/elements/src/lib/record-feature-catalog/feature-catalog-list/feature-catalog-list.component.d.ts.map +1 -1
  98. package/libs/ui/elements/src/lib/ui-elements.module.d.ts +2 -1
  99. package/libs/ui/elements/src/lib/ui-elements.module.d.ts.map +1 -1
  100. package/libs/ui/layout/src/index.d.ts +1 -0
  101. package/libs/ui/layout/src/index.d.ts.map +1 -1
  102. package/libs/ui/layout/src/lib/cell-popin/cell-popin.component.d.ts +28 -0
  103. package/libs/ui/layout/src/lib/cell-popin/cell-popin.component.d.ts.map +1 -0
  104. package/libs/ui/layout/src/lib/interactive-table/interactive-table.component.d.ts +4 -2
  105. package/libs/ui/layout/src/lib/interactive-table/interactive-table.component.d.ts.map +1 -1
  106. package/libs/ui/layout/src/lib/truncated-text/truncated-text.component.d.ts +5 -10
  107. package/libs/ui/layout/src/lib/truncated-text/truncated-text.component.d.ts.map +1 -1
  108. package/libs/ui/search/src/index.d.ts +1 -0
  109. package/libs/ui/search/src/index.d.ts.map +1 -1
  110. package/libs/ui/search/src/lib/results-table/action-menu/action-menu.component.d.ts +4 -3
  111. package/libs/ui/search/src/lib/results-table/action-menu/action-menu.component.d.ts.map +1 -1
  112. package/libs/ui/search/src/lib/results-table/results-table.component.d.ts +10 -4
  113. package/libs/ui/search/src/lib/results-table/results-table.component.d.ts.map +1 -1
  114. package/libs/ui/search/src/lib/ui-search.module.d.ts +2 -1
  115. package/libs/ui/search/src/lib/ui-search.module.d.ts.map +1 -1
  116. package/libs/util/i18n/src/lib/i18n.constants.d.ts +1 -0
  117. package/libs/util/i18n/src/lib/i18n.constants.d.ts.map +1 -1
  118. package/libs/util/i18n/src/lib/language-codes.d.ts +23 -1
  119. package/libs/util/i18n/src/lib/language-codes.d.ts.map +1 -1
  120. package/package.json +1 -1
  121. package/src/libs/api/metadata-converter/src/lib/fixtures/geocat-ch.records.ts +1 -1
  122. package/src/libs/api/metadata-converter/src/lib/gn4/gn4.field.mapper.ts +61 -4
  123. package/src/libs/api/metadata-converter/src/lib/gn4/types/metadata.model.ts +0 -1
  124. package/src/libs/api/metadata-converter/src/lib/iso19115-3/read-parts.ts +13 -3
  125. package/src/libs/api/metadata-converter/src/lib/iso19115-3/write-parts.ts +5 -1
  126. package/src/libs/api/metadata-converter/src/lib/iso19139/read-parts.ts +6 -3
  127. package/src/libs/api/metadata-converter/src/lib/iso19139/write-parts.ts +4 -1
  128. package/src/libs/api/repository/src/lib/gn4/elasticsearch/constant.ts +1 -0
  129. package/src/libs/api/repository/src/lib/gn4/elasticsearch/elasticsearch.service.ts +2 -2
  130. package/src/libs/api/repository/src/lib/gn4/gn4-repository.ts +111 -15
  131. package/src/libs/api/repository/src/lib/gn4/platform/gn4-platform.service.ts +11 -0
  132. package/src/libs/api/repository/src/lib/gn4/settings/gn4-settings.service.ts +3 -0
  133. package/src/libs/common/domain/src/lib/model/record/metadata.model.ts +20 -14
  134. package/src/libs/common/domain/src/lib/repository/records-repository.interface.ts +6 -0
  135. package/src/libs/common/fixtures/src/lib/records.fixtures.ts +63 -0
  136. package/src/libs/feature/editor/src/lib/components/multilingual-panel/multilingual-panel.component.html +117 -11
  137. package/src/libs/feature/editor/src/lib/components/multilingual-panel/multilingual-panel.component.ts +316 -6
  138. package/src/libs/feature/record/src/lib/state/mdview.actions.ts +10 -0
  139. package/src/libs/feature/record/src/lib/state/mdview.effects.ts +22 -0
  140. package/src/libs/feature/record/src/lib/state/mdview.facade.ts +4 -0
  141. package/src/libs/feature/record/src/lib/state/mdview.reducer.ts +12 -0
  142. package/src/libs/feature/record/src/lib/state/mdview.selectors.ts +9 -0
  143. package/src/libs/feature/search/src/lib/constants.ts +2 -1
  144. package/src/libs/feature/search/src/lib/fuzzy-search/fuzzy-search.component.html +1 -1
  145. package/src/libs/feature/search/src/lib/fuzzy-search/fuzzy-search.component.ts +1 -0
  146. package/src/libs/feature/search/src/lib/results-table/results-table-container.component.html +1 -0
  147. package/src/libs/feature/search/src/lib/results-table/results-table-container.component.ts +13 -3
  148. package/src/libs/ui/elements/src/lib/internal-link-card/internal-link-card.component.html +7 -75
  149. package/src/libs/ui/elements/src/lib/internal-link-card/internal-link-card.component.ts +20 -51
  150. package/src/libs/ui/elements/src/lib/internal-link-card-contact/internal-link-card-contact.component.html +69 -0
  151. package/src/libs/ui/elements/src/lib/internal-link-card-contact/internal-link-card-contact.component.ts +61 -0
  152. package/src/libs/ui/elements/src/lib/metadata-quality/metadata-quality.component.ts +48 -9
  153. package/src/libs/ui/elements/src/lib/metadata-quality-item/metadata-quality-item.component.ts +4 -0
  154. package/src/libs/ui/elements/src/lib/record-feature-catalog/feature-catalog-list/feature-catalog-list.component.html +96 -37
  155. package/src/libs/ui/elements/src/lib/record-feature-catalog/feature-catalog-list/feature-catalog-list.component.ts +60 -29
  156. package/src/libs/ui/elements/src/lib/ui-elements.module.ts +2 -0
  157. package/src/libs/ui/inputs/src/lib/autocomplete/autocomplete.component.html +1 -1
  158. package/src/libs/ui/layout/src/index.ts +1 -0
  159. package/src/libs/ui/layout/src/lib/cell-popin/cell-popin.component.html +40 -0
  160. package/src/libs/ui/layout/src/lib/cell-popin/cell-popin.component.ts +141 -0
  161. package/src/libs/ui/layout/src/lib/interactive-table/interactive-table.component.html +3 -2
  162. package/src/libs/ui/layout/src/lib/interactive-table/interactive-table.component.ts +13 -6
  163. package/src/libs/ui/layout/src/lib/truncated-text/truncated-text.component.html +25 -42
  164. package/src/libs/ui/layout/src/lib/truncated-text/truncated-text.component.ts +10 -49
  165. package/src/libs/ui/layout/src/lib/ui-layout.module.ts +1 -1
  166. package/src/libs/ui/search/src/index.ts +1 -0
  167. package/src/libs/ui/search/src/lib/results-table/action-menu/action-menu.component.html +16 -6
  168. package/src/libs/ui/search/src/lib/results-table/action-menu/action-menu.component.ts +15 -11
  169. package/src/libs/ui/search/src/lib/results-table/results-table.component.css +4 -0
  170. package/src/libs/ui/search/src/lib/results-table/results-table.component.html +31 -27
  171. package/src/libs/ui/search/src/lib/results-table/results-table.component.ts +33 -15
  172. package/src/libs/ui/search/src/lib/ui-search.module.ts +3 -0
  173. package/src/libs/util/i18n/src/lib/i18n.constants.ts +42 -0
  174. package/src/libs/util/i18n/src/lib/language-codes.ts +23 -1
  175. package/tailwind.base.css +1 -1
  176. package/translations/de.json +41 -6
  177. package/translations/en.json +41 -6
  178. package/translations/es.json +40 -5
  179. package/translations/fr.json +41 -6
  180. package/translations/it.json +40 -5
  181. package/translations/nl.json +40 -5
  182. package/translations/pt.json +40 -5
  183. package/translations/sk.json +40 -5
@@ -60,8 +60,9 @@
60
60
  </h4>
61
61
  </div>
62
62
  <div
63
- *ngIf="size === 'L'"
64
- class="mr-6 text-xs text-gray-900 line-clamp-2 overflow-hidden"
63
+ *ngIf="displayAbstract()"
64
+ class="mr-6 text-xs text-gray-900 overflow-hidden"
65
+ [ngClass]="getAbstractClass()"
65
66
  data-cy="recordAbstract"
66
67
  >
67
68
  <gn-ui-markdown-parser
@@ -71,80 +72,11 @@
71
72
  </div>
72
73
  </div>
73
74
 
74
- <div
75
- data-cy="recordOrg"
75
+ <gn-ui-internal-link-card-contact
76
76
  *ngIf="size !== 'XS' && record.ownerOrganization?.name"
77
- class="flex items-center justify-evenly bg-gray-50 rounded-lg h-[53px] px-2"
78
- >
79
- <div class="flex items-center flex-1 min-w-0">
80
- <div
81
- class="w-[45px] h-[45px] rounded-lg overflow-hidden shrink-0 mr-3"
82
- >
83
- <gn-ui-thumbnail
84
- [thumbnailUrl]="
85
- record.ownerOrganization?.logoUrl?.toString() || ''
86
- "
87
- [fit]="'contain'"
88
- class="w-full h-full rounded-lg"
89
- ></gn-ui-thumbnail>
90
- </div>
91
- <div *ngIf="organization?.name" class="flex-1 w-0 overflow-hidden">
92
- <div
93
- class="text-xs text-black font-normal leading-tight truncate"
94
- translate
95
- >
96
- record.card.metadata.contact
97
- </div>
98
- <div
99
- data-cy="recordOrgName"
100
- class="text-xl text-primary-black font-medium truncate"
101
- >
102
- {{ organization.name }}
103
- </div>
104
- </div>
105
- </div>
106
- <div *ngIf="size === 'L'" class="ml-2 flex space-x-2">
107
- <div *ngIf="organization?.website" class="flex">
108
- <button
109
- [title]="organization.website"
110
- class="w-[40px] h-[32px] flex items-center justify-center rounded-lg border border-[#D4D3D7] px-[8px] py-[4px] hover:bg-primary-lightest"
111
- (click)="openExternalUrl($event, organization.website)"
112
- >
113
- <ng-icon name="iconoirInternet"></ng-icon>
114
- </button>
115
- </div>
116
- <div *ngIf="contacts[0]?.email" class="flex">
117
- <button
118
- [title]="contacts[0].email"
119
- class="w-[40px] h-[32px] flex items-center justify-center rounded-lg border border-[#D4D3D7] px-[8px] py-[4px] hover:bg-primary-lightest"
120
- data-cy="contact-email"
121
- (click)="openMailto($event, contacts[0].email)"
122
- >
123
- <ng-icon name="matEmailOutline"></ng-icon>
124
- </button>
125
- </div>
126
- <div *ngIf="contacts[0]?.phone" class="flex">
127
- <button
128
- [title]="'Copy to clipboard'"
129
- class="w-[40px] h-[32px] flex items-center justify-center rounded-lg border border-[#D4D3D7] px-[8px] py-[4px] hover:bg-primary-lightest relative group"
130
- data-cy="contact-phone"
131
- (click)="copyToClipboard($event, contacts[0].phone)"
132
- >
133
- <ng-icon name="matPhoneOutline"></ng-icon>
134
- </button>
135
- </div>
136
- <div *ngIf="contacts[0]?.address" class="flex">
137
- <button
138
- [title]="'Copy to clipboard'"
139
- class="w-[40px] h-[32px] flex items-center justify-center rounded-lg border border-[#D4D3D7] px-[8px] py-[4px] hover:bg-primary-lightest relative group"
140
- data-cy="contact-phone"
141
- (click)="copyToClipboard($event, contacts[0].address)"
142
- >
143
- <ng-icon name="matLocationOnOutline"></ng-icon>
144
- </button>
145
- </div>
146
- </div>
147
- </div>
77
+ [record]="record"
78
+ [size]="size"
79
+ ></gn-ui-internal-link-card-contact>
148
80
  </div>
149
81
  </div>
150
82
  </div>
@@ -7,35 +7,20 @@ import {
7
7
  EventEmitter,
8
8
  ElementRef,
9
9
  } from '@angular/core'
10
- import {
11
- CatalogRecord,
12
- Organization,
13
- } from '../../../../../../libs/common/domain/src/lib/model/record'
10
+ import { CatalogRecord } from '../../../../../../libs/common/domain/src/lib/model/record'
14
11
  import { NgClass, NgIf, NgTemplateOutlet } from '@angular/common'
15
12
  import { GeoDataBadgeComponent } from '../geo-data-badge/geo-data-badge.component'
16
13
  import { KindBadgeComponent } from '../kind-badge/kind-badge.component'
17
14
  import { MarkdownParserComponent } from '../markdown-parser/markdown-parser.component'
18
15
  import { MetadataQualityComponent } from '../metadata-quality/metadata-quality.component'
19
- import { ThumbnailComponent } from '../thumbnail/thumbnail.component'
20
- import {
21
- propagateToDocumentOnly,
22
- removeWhitespace,
23
- stripHtml,
24
- } from '../../../../../../libs/util/shared/src'
25
- import {
26
- NgIconComponent,
27
- provideIcons,
28
- provideNgIconsConfig,
29
- } from '@ng-icons/core'
30
- import {
31
- matLocationSearchingOutline,
32
- matEmailOutline,
33
- matPhoneOutline,
34
- matLocationOnOutline,
35
- } from '@ng-icons/material-icons/outline'
16
+ import { InternalLinkCardContactComponent } from '../internal-link-card-contact/internal-link-card-contact.component'
17
+ import { removeWhitespace, stripHtml } from '../../../../../../libs/util/shared/src'
18
+ import { provideIcons, provideNgIconsConfig } from '@ng-icons/core'
19
+ import { matLocationSearchingOutline } from '@ng-icons/material-icons/outline'
36
20
  import { iconoirInternet } from '@ng-icons/iconoir'
37
21
  import { TranslateModule } from '@ngx-translate/core'
38
22
  import { fromEvent, Subscription } from 'rxjs'
23
+ import { ThumbnailComponent } from '../thumbnail/thumbnail.component'
39
24
 
40
25
  type CardSize = 'L' | 'M' | 'S' | 'XS'
41
26
 
@@ -45,22 +30,19 @@ type CardSize = 'L' | 'M' | 'S' | 'XS'
45
30
  imports: [
46
31
  NgClass,
47
32
  NgIf,
48
- ThumbnailComponent,
49
33
  MetadataQualityComponent,
50
34
  NgTemplateOutlet,
51
- NgIconComponent,
52
35
  TranslateModule,
53
36
  GeoDataBadgeComponent,
54
37
  KindBadgeComponent,
55
38
  MarkdownParserComponent,
39
+ InternalLinkCardContactComponent,
40
+ ThumbnailComponent,
56
41
  ],
57
42
  providers: [
58
43
  provideIcons({
59
44
  iconoirInternet,
60
45
  matLocationSearchingOutline,
61
- matEmailOutline,
62
- matPhoneOutline,
63
- matLocationOnOutline,
64
46
  }),
65
47
  provideNgIconsConfig({
66
48
  size: '1.2em',
@@ -125,35 +107,22 @@ export class InternalLinkCardComponent implements OnInit {
125
107
  )
126
108
  }
127
109
 
128
- get organization(): Organization {
129
- return this.record.ownerOrganization
130
- }
131
-
132
- get contacts() {
133
- return (
134
- (this.record.kind === 'dataset'
135
- ? this.record.contactsForResource
136
- : this.record.contacts) || []
137
- )
138
- }
139
-
140
110
  getTitleClass() {
141
111
  return this.titleClassMap[this._size]
142
112
  }
143
-
144
- openExternalUrl(event: Event, url: URL): void {
145
- event.stopPropagation()
146
- window.open(url, '_blank')
147
- }
148
-
149
- openMailto(event: Event, email: string): void {
150
- event.stopPropagation()
151
- window.open(`mailto:${email}`, '_blank')
113
+ getAbstractClass(): string {
114
+ const marginClass = ['S', 'XS'].includes(this.size) ? 'ml-2' : ''
115
+ const clampClass =
116
+ this.size === 'L' && !this.record.ownerOrganization?.name
117
+ ? 'line-clamp-6'
118
+ : 'line-clamp-2'
119
+ return `${clampClass} ${marginClass}`.trim()
152
120
  }
153
-
154
- copyToClipboard(event: Event, text: string): void {
155
- event.stopPropagation()
156
- navigator.clipboard.writeText(text)
121
+ displayAbstract(): boolean {
122
+ return (
123
+ this.size === 'L' ||
124
+ (['M', 'S'].includes(this.size) && !this.record.ownerOrganization?.name)
125
+ )
157
126
  }
158
127
 
159
128
  get shouldShowThumbnail(): boolean {
@@ -0,0 +1,69 @@
1
+ <div
2
+ data-cy="recordOrg"
3
+ class="flex items-center justify-evenly bg-gray-50 rounded-lg h-[53px] px-2"
4
+ >
5
+ <div class="flex items-center flex-1 min-w-0">
6
+ <div class="w-[45px] h-[45px] rounded-lg overflow-hidden shrink-0 mr-3">
7
+ <gn-ui-thumbnail
8
+ [thumbnailUrl]="record.ownerOrganization?.logoUrl?.toString() || ''"
9
+ [fit]="'contain'"
10
+ class="w-full h-full rounded-lg"
11
+ ></gn-ui-thumbnail>
12
+ </div>
13
+ <div *ngIf="organization?.name" class="flex-1 w-0 overflow-hidden">
14
+ <div
15
+ class="text-xs text-black font-normal leading-tight truncate"
16
+ translate
17
+ >
18
+ record.card.metadata.contact
19
+ </div>
20
+ <div
21
+ data-cy="recordOrgName"
22
+ class="text-xl text-primary-black font-medium truncate"
23
+ >
24
+ {{ organization.name }}
25
+ </div>
26
+ </div>
27
+ </div>
28
+ <div *ngIf="size === 'L'" class="ml-2 flex space-x-2">
29
+ <div *ngIf="organization?.website" class="flex">
30
+ <button
31
+ [title]="organization.website"
32
+ class="w-[40px] h-[32px] flex items-center justify-center rounded-lg border border-[#D4D3D7] px-[8px] py-[4px] hover:bg-primary-lightest"
33
+ (click)="openExternalUrl($event, organization.website)"
34
+ >
35
+ <ng-icon name="iconoirInternet"></ng-icon>
36
+ </button>
37
+ </div>
38
+ <div *ngIf="contacts[0]?.email" class="flex">
39
+ <button
40
+ [title]="contacts[0].email"
41
+ class="w-[40px] h-[32px] flex items-center justify-center rounded-lg border border-[#D4D3D7] px-[8px] py-[4px] hover:bg-primary-lightest"
42
+ data-cy="contact-email"
43
+ (click)="openMailto($event, contacts[0].email)"
44
+ >
45
+ <ng-icon name="matEmailOutline"></ng-icon>
46
+ </button>
47
+ </div>
48
+ <div *ngIf="contacts[0]?.phone" class="flex">
49
+ <button
50
+ [title]="'Copy to clipboard'"
51
+ class="w-[40px] h-[32px] flex items-center justify-center rounded-lg border border-[#D4D3D7] px-[8px] py-[4px] hover:bg-primary-lightest relative group"
52
+ data-cy="contact-phone"
53
+ (click)="copyToClipboard($event, contacts[0].phone)"
54
+ >
55
+ <ng-icon name="matPhoneOutline"></ng-icon>
56
+ </button>
57
+ </div>
58
+ <div *ngIf="contacts[0]?.address" class="flex">
59
+ <button
60
+ [title]="'Copy to clipboard'"
61
+ class="w-[40px] h-[32px] flex items-center justify-center rounded-lg border border-[#D4D3D7] px-[8px] py-[4px] hover:bg-primary-lightest relative group"
62
+ data-cy="contact-phone"
63
+ (click)="copyToClipboard($event, contacts[0].address)"
64
+ >
65
+ <ng-icon name="matLocationOnOutline"></ng-icon>
66
+ </button>
67
+ </div>
68
+ </div>
69
+ </div>
@@ -0,0 +1,61 @@
1
+ import { Component, Input } from '@angular/core'
2
+ import { NgIf } from '@angular/common'
3
+ import {
4
+ CatalogRecord,
5
+ Organization,
6
+ } from '../../../../../../libs/common/domain/src/lib/model/record'
7
+ import { ThumbnailComponent } from '../thumbnail/thumbnail.component'
8
+ import { NgIconComponent, provideIcons } from '@ng-icons/core'
9
+ import {
10
+ matEmailOutline,
11
+ matPhoneOutline,
12
+ matLocationOnOutline,
13
+ } from '@ng-icons/material-icons/outline'
14
+ import { iconoirInternet } from '@ng-icons/iconoir'
15
+ import { TranslateModule } from '@ngx-translate/core'
16
+
17
+ @Component({
18
+ selector: 'gn-ui-internal-link-card-contact',
19
+ standalone: true,
20
+ imports: [NgIf, ThumbnailComponent, NgIconComponent, TranslateModule],
21
+ providers: [
22
+ provideIcons({
23
+ iconoirInternet,
24
+ matEmailOutline,
25
+ matPhoneOutline,
26
+ matLocationOnOutline,
27
+ }),
28
+ ],
29
+ templateUrl: './internal-link-card-contact.component.html',
30
+ })
31
+ export class InternalLinkCardContactComponent {
32
+ @Input() record: CatalogRecord
33
+ @Input() size: 'L' | 'M' | 'S' | 'XS' = 'M'
34
+
35
+ get organization(): Organization {
36
+ return this.record.ownerOrganization
37
+ }
38
+
39
+ get contacts() {
40
+ return (
41
+ (this.record.kind === 'dataset'
42
+ ? this.record.contactsForResource
43
+ : this.record.contacts) || []
44
+ )
45
+ }
46
+
47
+ openExternalUrl(event: Event, url: URL): void {
48
+ event.stopPropagation()
49
+ window.open(url, '_blank')
50
+ }
51
+
52
+ openMailto(event: Event, email: string): void {
53
+ event.stopPropagation()
54
+ window.open(`mailto:${email}`, '_blank')
55
+ }
56
+
57
+ copyToClipboard(event: Event, text: string): void {
58
+ event.stopPropagation()
59
+ navigator.clipboard.writeText(text)
60
+ }
61
+ }
@@ -20,6 +20,10 @@ import { NgIcon } from '@ng-icons/core'
20
20
  import { matInfoOutline } from '@ng-icons/material-icons/outline'
21
21
  import { provideIcons, provideNgIconsConfig } from '@ng-icons/core'
22
22
 
23
+ type QualityChecks = {
24
+ [key: string]: (metadata: Partial<CatalogRecord>) => boolean
25
+ }
26
+
23
27
  @Component({
24
28
  selector: 'gn-ui-metadata-quality',
25
29
  templateUrl: './metadata-quality.component.html',
@@ -69,18 +73,53 @@ export class MetadataQualityComponent implements OnChanges {
69
73
  this.items.push({ name, value })
70
74
  }
71
75
  }
76
+ hasGetCapabilities(url: string): boolean {
77
+ return url.toLowerCase().includes('capabilities')
78
+ }
79
+
80
+ private readonly COMMON_CHECKS: QualityChecks = {
81
+ title: (metadata) => !!metadata?.title,
82
+ description: (metadata) => !!metadata?.abstract,
83
+ keywords: (metadata) => (metadata?.keywords?.length ?? 0) > 0,
84
+ legalConstraints: (metadata) =>
85
+ (metadata?.legalConstraints?.length ?? 0) > 0,
86
+ contact: (metadata) => !!metadata?.contacts?.[0]?.email,
87
+ }
88
+
89
+ private readonly SPECIFIC_CHECKS: Record<string, QualityChecks> = {
90
+ dataset: {
91
+ updateFrequency: (metadata) => !!metadata?.updateFrequency,
92
+ topic: (metadata) => (metadata?.topics?.length ?? 0) > 0,
93
+ organisation: (metadata) => !!metadata?.contacts?.[0]?.organization?.name,
94
+ },
95
+ service: {
96
+ capabilities: (metadata) =>
97
+ (metadata?.onlineResources ?? []).some((resource) =>
98
+ this.hasGetCapabilities(resource?.url?.href ?? '')
99
+ ),
100
+ },
101
+ reuse: {
102
+ topic: (metadata) => (metadata?.topics?.length ?? 0) > 0,
103
+ organisation: (metadata) => !!metadata?.contacts?.[0]?.organization?.name,
104
+ source: (metadata) => !!metadata?.extras?.sourcesIdentifiers,
105
+ },
106
+ }
72
107
 
73
108
  initialize() {
74
- const contact = this.metadata?.contacts?.[0]
75
109
  this.items = []
76
- this.add('title', !!this.metadata?.title)
77
- this.add('description', !!this.metadata?.abstract)
78
- this.add('topic', this.metadata?.topics?.length > 0)
79
- this.add('keywords', this.metadata?.keywords?.length > 0)
80
- this.add('legalConstraints', this.metadata?.legalConstraints?.length > 0)
81
- this.add('organisation', !!contact?.organization)
82
- this.add('contact', !!contact?.email)
83
- this.add('updateFrequency', !!this.metadata?.updateFrequency)
110
+
111
+ Object.entries(this.COMMON_CHECKS).forEach(([name, check]) => {
112
+ this.add(name, check(this.metadata))
113
+ })
114
+
115
+ const datasetType = this.metadata?.kind
116
+ if (datasetType && this.SPECIFIC_CHECKS[datasetType]) {
117
+ Object.entries(this.SPECIFIC_CHECKS[datasetType]).forEach(
118
+ ([name, check]) => {
119
+ this.add(name, check(this.metadata))
120
+ }
121
+ )
122
+ }
84
123
  }
85
124
 
86
125
  ngOnChanges(changes: SimpleChanges): void {
@@ -20,6 +20,10 @@ marker('record.metadata.quality.updateFrequency.success')
20
20
  marker('record.metadata.quality.updateFrequency.failed')
21
21
  marker('record.metadata.quality.organisation.success')
22
22
  marker('record.metadata.quality.organisation.failed')
23
+ marker('record.metadata.quality.capabilities.success')
24
+ marker('record.metadata.quality.capabilities.failed')
25
+ marker('record.metadata.quality.source.success')
26
+ marker('record.metadata.quality.source.failed')
23
27
 
24
28
  export interface MetadataQualityItem {
25
29
  name: string
@@ -1,48 +1,107 @@
1
- <div class="flex flex-col gap-2 py-5 px-5 h-[562px] overflow-y-auto">
2
- <div
3
- *ngFor="let featureType of filteredFeatureCatalog?.featureTypes"
4
- class="rounded shadow bg-white"
5
- >
6
- <gn-ui-expandable-panel
7
- [collapsed]="filteredFeatureCatalog?.featureTypes?.length !== 1"
8
- iconColor="black"
1
+ <div
2
+ cdkScrollable
3
+ class="h-[562px] relative overflow-y-auto overflow-hidden"
4
+ data-cy="feat-catalog-scrollable"
5
+ >
6
+ <div #scrollContainer class="flex flex-col gap-2 py-5 px-5">
7
+ <div
8
+ *ngFor="let featureType of filteredFeatureCatalog?.featureTypes"
9
+ class="rounded shadow bg-white"
9
10
  >
10
- <ng-template #titleTemplate>
11
- <div class="px-2">
12
- <div class="text-lg font-bold">{{ featureType.name }}</div>
13
- <div class="text-sm" *ngIf="featureType.definition">
14
- {{ featureType.definition }}
15
- </div>
16
- </div>
17
- </ng-template>
18
- <div
19
- class="grid gap-0"
20
- [style.grid-template-columns]="gridTemplateColumns"
21
- data-cy="feature-type-content"
11
+ <gn-ui-expandable-panel
12
+ #expanel
13
+ [collapsed]="filteredFeatureCatalog?.featureTypes?.length !== 1"
14
+ iconColor="black"
22
15
  >
16
+ <ng-template #titleTemplate>
17
+ <div class="px-2">
18
+ <div class="text-lg font-bold">{{ featureType.name }}</div>
19
+ <div class="text-sm" *ngIf="featureType.definition">
20
+ {{ featureType.definition }}
21
+ </div>
22
+ </div>
23
+ </ng-template>
23
24
  <div
24
- class="py-1 px-2 text-sm font-bold text-left border-t"
25
- [class.border-l]="i > 0"
26
- [class.border-gray-300]="i > 0"
27
- *ngFor="let col of columns; let i = index"
28
- data-test="column-label"
25
+ *ngIf="expanel && !expanel.collapsed"
26
+ class="grid gap-0"
27
+ [style.grid-template-columns]="
28
+ getGridTemplateColumns(featureType.attributes)
29
+ "
30
+ data-cy="feature-type-content"
29
31
  >
30
- {{ col.label | translate }}
31
- </div>
32
- <ng-container *ngFor="let row of featureType.attributes">
33
32
  <div
34
- class="bg-white text-sm font-normal text-left border-t"
33
+ class="py-1 px-2 text-sm font-bold text-left border-t break-all"
35
34
  [class.border-l]="i > 0"
36
35
  [class.border-gray-300]="i > 0"
37
- *ngFor="let col of columns; let i = index"
36
+ [ngClass]="col.class || ''"
37
+ *ngFor="
38
+ let col of getColumnsDefinition(featureType.attributes);
39
+ trackBy: trackByColumn;
40
+ let i = index
41
+ "
42
+ data-test="column-label"
38
43
  >
39
- <gn-ui-truncated-text
40
- extraClass="py-3 px-2"
41
- [text]="row[col.key]"
42
- ></gn-ui-truncated-text>
44
+ {{ 'feature.catalog.attribute.' + col.key | translate }}
43
45
  </div>
44
- </ng-container>
45
- </div>
46
- </gn-ui-expandable-panel>
46
+ <ng-container *ngIf="expanel && !expanel.collapsed">
47
+ <ng-container *ngFor="let row of featureType.attributes">
48
+ <div
49
+ class="bg-white text-sm font-normal text-left border-t"
50
+ [class.border-l]="i > 0"
51
+ [class.border-gray-300]="i > 0"
52
+ *ngFor="
53
+ let col of getColumnsDefinition(featureType.attributes);
54
+ trackBy: trackByColumn;
55
+ let i = index
56
+ "
57
+ data-test="cell-content"
58
+ >
59
+ <ng-container
60
+ *ngIf="col.key === 'values'; then tplValues; else defaultTpl"
61
+ ></ng-container>
62
+
63
+ <ng-template #defaultTpl>
64
+ <gn-ui-truncated-text
65
+ [scrollContainer]="scrollContainer"
66
+ [cdkScrollContainer]="scrollable"
67
+ [text]="row[col.key]"
68
+ ></gn-ui-truncated-text>
69
+ </ng-template>
70
+
71
+ <ng-template #tplValues>
72
+ <gn-ui-cell-popin
73
+ #popinRef
74
+ [scrollContainer]="scrollContainer"
75
+ [cdkScrollContainer]="scrollable"
76
+ *ngIf="row.values?.length > 0"
77
+ >
78
+ <div cellContent data-cy="values-cell-popin-trigger">
79
+ <gn-ui-button
80
+ type="light"
81
+ extraClass="bg-transparent border-none"
82
+ (buttonClick)="popinRef.openOverlay()"
83
+ >
84
+ <ng-icon name="iconoirList" size="24"></ng-icon>
85
+ </gn-ui-button>
86
+ </div>
87
+ <div
88
+ popinContent
89
+ class="max-h-60 overflow-y-auto min-w-64 py-4 px-6"
90
+ style="scrollbar-width: thin"
91
+ >
92
+ <ul class="list-disc list-inside mr-4">
93
+ <li *ngFor="let v of row.values">
94
+ {{ v.label || v.code }}
95
+ </li>
96
+ </ul>
97
+ </div>
98
+ </gn-ui-cell-popin>
99
+ </ng-template>
100
+ </div>
101
+ </ng-container>
102
+ </ng-container>
103
+ </div>
104
+ </gn-ui-expandable-panel>
105
+ </div>
47
106
  </div>
48
107
  </div>