geonetwork-ui 2.4.0-dev.f6f2f023 → 2.4.0-dev.fe59f179

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 (211) hide show
  1. package/esm2022/libs/api/repository/src/lib/gn4/gn4-repository.mjs +10 -1
  2. package/esm2022/libs/api/repository/src/lib/gn4/organizations/organizations-from-groups.service.mjs +2 -1
  3. package/esm2022/libs/api/repository/src/lib/gn4/organizations/organizations-from-metadata.service.mjs +3 -1
  4. package/esm2022/libs/api/repository/src/lib/gn4/platform/gn4-platform.mapper.mjs +2 -2
  5. package/esm2022/libs/api/repository/src/lib/gn4/platform/gn4-platform.service.mjs +9 -5
  6. package/esm2022/libs/common/domain/src/lib/model/record/organization.model.mjs +1 -1
  7. package/esm2022/libs/common/domain/src/lib/platform.service.interface.mjs +1 -1
  8. package/esm2022/libs/common/domain/src/lib/repository/records-repository.interface.mjs +1 -1
  9. package/esm2022/libs/data-access/gn4/src/openapi/model/user.api.model.mjs +1 -1
  10. package/esm2022/libs/feature/catalog/src/index.mjs +2 -1
  11. package/esm2022/libs/feature/catalog/src/lib/feature-catalog.module.mjs +1 -1
  12. package/esm2022/libs/feature/catalog/src/lib/organisations/organisations.component.mjs +6 -6
  13. package/esm2022/libs/feature/catalog/src/lib/organization-url.token.mjs +4 -0
  14. package/esm2022/libs/feature/catalog/src/lib/records/records.service.mjs +4 -6
  15. package/esm2022/libs/feature/dataviz/src/lib/chart-view/chart-view.component.mjs +4 -4
  16. package/esm2022/libs/feature/dataviz/src/lib/feature-dataviz.module.mjs +7 -4
  17. package/esm2022/libs/feature/dataviz/src/lib/table-view/table-view.component.mjs +1 -1
  18. package/esm2022/libs/feature/editor/src/lib/components/record-form/form-field/form-field-keywords/form-field-keywords.component.mjs +16 -44
  19. package/esm2022/libs/feature/editor/src/lib/components/record-form/form-field/form-field.component.mjs +2 -2
  20. package/esm2022/libs/feature/record/src/lib/feature-record.module.mjs +7 -4
  21. package/esm2022/libs/feature/record/src/lib/map-view/map-view.component.mjs +3 -3
  22. package/esm2022/libs/feature/record/src/lib/state/mdview.facade.mjs +8 -12
  23. package/esm2022/libs/feature/router/src/lib/default/constants.mjs +2 -1
  24. package/esm2022/libs/feature/router/src/lib/default/router.config.mjs +1 -1
  25. package/esm2022/libs/feature/router/src/lib/default/router.service.mjs +9 -2
  26. package/esm2022/libs/feature/router/src/lib/default/state/router.facade.mjs +8 -1
  27. package/esm2022/libs/feature/search/src/index.mjs +2 -2
  28. package/esm2022/libs/feature/search/src/lib/feature-search.module.mjs +7 -4
  29. package/esm2022/libs/feature/search/src/lib/fuzzy-search/fuzzy-search.component.mjs +3 -3
  30. package/esm2022/libs/feature/search/src/lib/results-table/results-table-container.component.mjs +49 -0
  31. package/esm2022/libs/feature/search/src/lib/state/search.facade.mjs +3 -2
  32. package/esm2022/libs/feature/search/src/lib/state/selectors.mjs +4 -1
  33. package/esm2022/libs/ui/catalog/src/lib/organisation-preview/organisation-preview.component.mjs +5 -5
  34. package/esm2022/libs/ui/catalog/src/lib/ui-catalog.module.mjs +5 -2
  35. package/esm2022/libs/ui/dataviz/src/lib/figure/figure.component.mjs +5 -8
  36. package/esm2022/libs/ui/elements/src/index.mjs +1 -2
  37. package/esm2022/libs/ui/elements/src/lib/error/error.component.mjs +5 -3
  38. package/esm2022/libs/ui/elements/src/lib/metadata-info/metadata-info.component.mjs +2 -3
  39. package/esm2022/libs/ui/elements/src/lib/related-record-card/related-record-card.component.mjs +24 -5
  40. package/esm2022/libs/ui/elements/src/lib/ui-elements.module.mjs +1 -6
  41. package/esm2022/libs/ui/inputs/src/lib/autocomplete/autocomplete.component.mjs +59 -26
  42. package/esm2022/libs/ui/inputs/src/lib/ui-inputs.module.mjs +2 -7
  43. package/esm2022/libs/ui/layout/src/index.mjs +2 -1
  44. package/esm2022/libs/ui/layout/src/lib/max-lines/max-lines.component.mjs +71 -0
  45. package/esm2022/libs/ui/layout/src/lib/ui-layout.module.mjs +1 -1
  46. package/esm2022/libs/ui/search/src/index.mjs +2 -1
  47. package/esm2022/libs/ui/search/src/lib/results-table/results-table.component.mjs +120 -0
  48. package/esm2022/libs/ui/widgets/src/lib/popup-alert/popup-alert.component.mjs +5 -3
  49. package/esm2022/libs/ui/widgets/src/lib/ui-widgets.module.mjs +1 -6
  50. package/esm2022/translations/de.json +27 -4
  51. package/esm2022/translations/en.json +30 -4
  52. package/esm2022/translations/es.json +27 -4
  53. package/esm2022/translations/fr.json +31 -5
  54. package/esm2022/translations/it.json +27 -4
  55. package/esm2022/translations/nl.json +27 -4
  56. package/esm2022/translations/pt.json +27 -4
  57. package/fesm2022/geonetwork-ui.mjs +774 -525
  58. package/fesm2022/geonetwork-ui.mjs.map +1 -1
  59. package/libs/api/repository/src/lib/gn4/gn4-repository.d.ts +1 -0
  60. package/libs/api/repository/src/lib/gn4/gn4-repository.d.ts.map +1 -1
  61. package/libs/api/repository/src/lib/gn4/organizations/organizations-from-groups.service.d.ts.map +1 -1
  62. package/libs/api/repository/src/lib/gn4/organizations/organizations-from-metadata.service.d.ts.map +1 -1
  63. package/libs/api/repository/src/lib/gn4/platform/gn4-platform.service.d.ts +2 -1
  64. package/libs/api/repository/src/lib/gn4/platform/gn4-platform.service.d.ts.map +1 -1
  65. package/libs/common/domain/src/lib/model/record/organization.model.d.ts +1 -0
  66. package/libs/common/domain/src/lib/model/record/organization.model.d.ts.map +1 -1
  67. package/libs/common/domain/src/lib/platform.service.interface.d.ts +2 -1
  68. package/libs/common/domain/src/lib/platform.service.interface.d.ts.map +1 -1
  69. package/libs/common/domain/src/lib/repository/records-repository.interface.d.ts +2 -0
  70. package/libs/common/domain/src/lib/repository/records-repository.interface.d.ts.map +1 -1
  71. package/libs/data-access/gn4/src/openapi/model/user.api.model.d.ts +1 -1
  72. package/libs/feature/catalog/src/index.d.ts +1 -0
  73. package/libs/feature/catalog/src/index.d.ts.map +1 -1
  74. package/libs/feature/catalog/src/lib/feature-catalog.module.d.ts.map +1 -1
  75. package/libs/feature/catalog/src/lib/organisations/organisations.component.d.ts.map +1 -1
  76. package/libs/feature/catalog/src/lib/organization-url.token.d.ts +3 -0
  77. package/libs/feature/catalog/src/lib/organization-url.token.d.ts.map +1 -0
  78. package/libs/feature/catalog/src/lib/records/records.service.d.ts.map +1 -1
  79. package/libs/feature/dataviz/src/lib/feature-dataviz.module.d.ts +2 -1
  80. package/libs/feature/dataviz/src/lib/feature-dataviz.module.d.ts.map +1 -1
  81. package/libs/feature/editor/src/lib/components/record-form/form-field/form-field-keywords/form-field-keywords.component.d.ts +14 -24
  82. package/libs/feature/editor/src/lib/components/record-form/form-field/form-field-keywords/form-field-keywords.component.d.ts.map +1 -1
  83. package/libs/feature/record/src/lib/feature-record.module.d.ts +2 -1
  84. package/libs/feature/record/src/lib/feature-record.module.d.ts.map +1 -1
  85. package/libs/feature/record/src/lib/state/mdview.facade.d.ts.map +1 -1
  86. package/libs/feature/router/src/lib/default/constants.d.ts +1 -0
  87. package/libs/feature/router/src/lib/default/constants.d.ts.map +1 -1
  88. package/libs/feature/router/src/lib/default/router.config.d.ts +1 -0
  89. package/libs/feature/router/src/lib/default/router.config.d.ts.map +1 -1
  90. package/libs/feature/router/src/lib/default/router.service.d.ts +1 -0
  91. package/libs/feature/router/src/lib/default/router.service.d.ts.map +1 -1
  92. package/libs/feature/router/src/lib/default/state/router.facade.d.ts +1 -0
  93. package/libs/feature/router/src/lib/default/state/router.facade.d.ts.map +1 -1
  94. package/libs/feature/search/src/index.d.ts +1 -1
  95. package/libs/feature/search/src/index.d.ts.map +1 -1
  96. package/libs/feature/search/src/lib/feature-search.module.d.ts +2 -1
  97. package/libs/feature/search/src/lib/feature-search.module.d.ts.map +1 -1
  98. package/libs/feature/search/src/lib/fuzzy-search/fuzzy-search.component.d.ts.map +1 -1
  99. package/libs/feature/search/src/lib/results-table/results-table-container.component.d.ts +25 -0
  100. package/libs/feature/search/src/lib/results-table/results-table-container.component.d.ts.map +1 -0
  101. package/libs/feature/search/src/lib/state/search.facade.d.ts +1 -0
  102. package/libs/feature/search/src/lib/state/search.facade.d.ts.map +1 -1
  103. package/libs/feature/search/src/lib/state/selectors.d.ts +1 -0
  104. package/libs/feature/search/src/lib/state/selectors.d.ts.map +1 -1
  105. package/libs/ui/catalog/src/lib/organisation-preview/organisation-preview.component.d.ts +2 -2
  106. package/libs/ui/catalog/src/lib/ui-catalog.module.d.ts +2 -1
  107. package/libs/ui/catalog/src/lib/ui-catalog.module.d.ts.map +1 -1
  108. package/libs/ui/dataviz/src/lib/figure/figure.component.d.ts +1 -2
  109. package/libs/ui/dataviz/src/lib/figure/figure.component.d.ts.map +1 -1
  110. package/libs/ui/elements/src/index.d.ts +0 -1
  111. package/libs/ui/elements/src/index.d.ts.map +1 -1
  112. package/libs/ui/elements/src/lib/error/error.component.d.ts +3 -1
  113. package/libs/ui/elements/src/lib/error/error.component.d.ts.map +1 -1
  114. package/libs/ui/elements/src/lib/related-record-card/related-record-card.component.d.ts +5 -1
  115. package/libs/ui/elements/src/lib/related-record-card/related-record-card.component.d.ts.map +1 -1
  116. package/libs/ui/elements/src/lib/ui-elements.module.d.ts +18 -19
  117. package/libs/ui/elements/src/lib/ui-elements.module.d.ts.map +1 -1
  118. package/libs/ui/inputs/src/lib/autocomplete/autocomplete.component.d.ts +7 -4
  119. package/libs/ui/inputs/src/lib/autocomplete/autocomplete.component.d.ts.map +1 -1
  120. package/libs/ui/inputs/src/lib/ui-inputs.module.d.ts +35 -36
  121. package/libs/ui/inputs/src/lib/ui-inputs.module.d.ts.map +1 -1
  122. package/libs/ui/layout/src/index.d.ts +1 -0
  123. package/libs/ui/layout/src/index.d.ts.map +1 -1
  124. package/libs/ui/{elements → layout}/src/lib/max-lines/max-lines.component.d.ts +2 -2
  125. package/libs/ui/layout/src/lib/max-lines/max-lines.component.d.ts.map +1 -0
  126. package/libs/ui/layout/src/lib/ui-layout.module.d.ts.map +1 -1
  127. package/libs/ui/search/src/index.d.ts +1 -0
  128. package/libs/ui/search/src/index.d.ts.map +1 -1
  129. package/libs/ui/search/src/lib/results-table/results-table.component.d.ts +30 -0
  130. package/libs/ui/search/src/lib/results-table/results-table.component.d.ts.map +1 -0
  131. package/libs/ui/widgets/src/lib/popup-alert/popup-alert.component.d.ts +1 -1
  132. package/libs/ui/widgets/src/lib/popup-alert/popup-alert.component.d.ts.map +1 -1
  133. package/libs/ui/widgets/src/lib/ui-widgets.module.d.ts +10 -11
  134. package/libs/ui/widgets/src/lib/ui-widgets.module.d.ts.map +1 -1
  135. package/package.json +1 -1
  136. package/src/libs/api/repository/src/lib/gn4/gn4-repository.ts +14 -0
  137. package/src/libs/api/repository/src/lib/gn4/organizations/organizations-from-groups.service.ts +1 -0
  138. package/src/libs/api/repository/src/lib/gn4/organizations/organizations-from-metadata.service.ts +3 -1
  139. package/src/libs/api/repository/src/lib/gn4/platform/gn4-platform.mapper.ts +1 -1
  140. package/src/libs/api/repository/src/lib/gn4/platform/gn4-platform.service.ts +26 -15
  141. package/src/libs/common/domain/src/lib/model/record/organization.model.ts +1 -0
  142. package/src/libs/common/domain/src/lib/platform.service.interface.ts +5 -1
  143. package/src/libs/common/domain/src/lib/repository/records-repository.interface.ts +3 -0
  144. package/src/libs/common/fixtures/src/lib/organisations.fixture.ts +28 -0
  145. package/src/libs/common/fixtures/src/lib/records.fixtures.ts +8 -0
  146. package/src/libs/data-access/datafeeder/src/openapi/model/datasetMetadata.api.model.ts +4 -0
  147. package/src/libs/data-access/datafeeder/src/openapi/model/datasetUploadStatus.api.model.ts +2 -0
  148. package/src/libs/data-access/gn4/src/openapi/model/user.api.model.ts +1 -1
  149. package/src/libs/feature/catalog/src/index.ts +1 -0
  150. package/src/libs/feature/catalog/src/lib/feature-catalog.module.ts +0 -1
  151. package/src/libs/feature/catalog/src/lib/organisations/organisations.component.html +1 -1
  152. package/src/libs/feature/catalog/src/lib/organisations/organisations.component.ts +2 -2
  153. package/src/libs/feature/catalog/src/lib/organization-url.token.ts +6 -0
  154. package/src/libs/feature/catalog/src/lib/records/records.service.ts +6 -8
  155. package/src/libs/feature/dataviz/src/lib/feature-dataviz.module.ts +2 -1
  156. package/src/libs/feature/editor/src/lib/components/record-form/form-field/form-field-keywords/form-field-keywords.component.html +2 -3
  157. package/src/libs/feature/editor/src/lib/components/record-form/form-field/form-field-keywords/form-field-keywords.component.ts +23 -63
  158. package/src/libs/feature/record/src/lib/feature-record.module.ts +3 -2
  159. package/src/libs/feature/record/src/lib/state/mdview.facade.ts +42 -31
  160. package/src/libs/feature/router/src/lib/default/constants.ts +1 -0
  161. package/src/libs/feature/router/src/lib/default/router.config.ts +1 -0
  162. package/src/libs/feature/router/src/lib/default/router.service.ts +13 -1
  163. package/src/libs/feature/router/src/lib/default/state/router.facade.ts +9 -1
  164. package/src/libs/feature/search/src/index.ts +1 -1
  165. package/src/libs/feature/search/src/lib/feature-search.module.ts +2 -1
  166. package/src/libs/feature/search/src/lib/fuzzy-search/fuzzy-search.component.ts +2 -2
  167. package/src/libs/feature/search/src/lib/results-table/results-table-container.component.html +9 -0
  168. package/src/libs/feature/search/src/lib/results-table/results-table-container.component.ts +49 -0
  169. package/src/libs/feature/search/src/lib/state/search.facade.ts +5 -0
  170. package/src/libs/feature/search/src/lib/state/selectors.ts +7 -0
  171. package/src/libs/ui/catalog/src/lib/organisation-preview/organisation-preview.component.html +8 -6
  172. package/src/libs/ui/catalog/src/lib/organisation-preview/organisation-preview.component.ts +2 -2
  173. package/src/libs/ui/catalog/src/lib/ui-catalog.module.ts +2 -0
  174. package/src/libs/ui/dataviz/src/lib/figure/figure.component.html +11 -3
  175. package/src/libs/ui/dataviz/src/lib/figure/figure.component.ts +3 -7
  176. package/src/libs/ui/elements/src/index.ts +0 -1
  177. package/src/libs/ui/elements/src/lib/error/error.component.html +30 -6
  178. package/src/libs/ui/elements/src/lib/error/error.component.ts +2 -0
  179. package/src/libs/ui/elements/src/lib/related-record-card/related-record-card.component.html +1 -1
  180. package/src/libs/ui/elements/src/lib/related-record-card/related-record-card.component.ts +22 -1
  181. package/src/libs/ui/elements/src/lib/ui-elements.module.ts +0 -3
  182. package/src/libs/ui/inputs/src/lib/autocomplete/autocomplete.component.css +0 -1
  183. package/src/libs/ui/inputs/src/lib/autocomplete/autocomplete.component.html +23 -19
  184. package/src/libs/ui/inputs/src/lib/autocomplete/autocomplete.component.ts +79 -27
  185. package/src/libs/ui/inputs/src/lib/ui-inputs.module.ts +0 -3
  186. package/src/libs/ui/layout/src/index.ts +1 -0
  187. package/src/libs/ui/{elements → layout}/src/lib/max-lines/max-lines.component.ts +9 -5
  188. package/src/libs/ui/layout/src/lib/ui-layout.module.ts +0 -1
  189. package/src/libs/ui/search/src/index.ts +1 -0
  190. package/src/libs/ui/search/src/lib/results-table/results-table.component.css +0 -0
  191. package/src/libs/{feature → ui}/search/src/lib/results-table/results-table.component.html +8 -8
  192. package/src/libs/ui/search/src/lib/results-table/results-table.component.ts +138 -0
  193. package/src/libs/ui/widgets/src/lib/popup-alert/popup-alert.component.ts +4 -0
  194. package/src/libs/ui/widgets/src/lib/ui-widgets.module.ts +0 -3
  195. package/translations/de.json +27 -4
  196. package/translations/en.json +30 -4
  197. package/translations/es.json +27 -4
  198. package/translations/fr.json +31 -5
  199. package/translations/it.json +27 -4
  200. package/translations/nl.json +27 -4
  201. package/translations/pt.json +27 -4
  202. package/translations/sk.json +28 -5
  203. package/esm2022/libs/feature/search/src/lib/results-table/results-table.component.mjs +0 -136
  204. package/esm2022/libs/ui/elements/src/lib/max-lines/max-lines.component.mjs +0 -69
  205. package/libs/feature/search/src/lib/results-table/results-table.component.d.ts +0 -36
  206. package/libs/feature/search/src/lib/results-table/results-table.component.d.ts.map +0 -1
  207. package/libs/ui/elements/src/lib/max-lines/max-lines.component.d.ts.map +0 -1
  208. package/src/libs/feature/search/src/lib/results-table/results-table.component.ts +0 -171
  209. /package/src/libs/feature/search/src/lib/results-table/{results-table.component.css → results-table-container.component.css} +0 -0
  210. /package/src/libs/ui/{elements → layout}/src/lib/max-lines/max-lines.component.css +0 -0
  211. /package/src/libs/ui/{elements → layout}/src/lib/max-lines/max-lines.component.html +0 -0
@@ -0,0 +1,49 @@
1
+ import { Component, EventEmitter, Output } from '@angular/core'
2
+ import { CatalogRecord } from '../../../../../../libs/common/domain/src/lib/model/record'
3
+ import { SearchFacade } from '../state/search.facade'
4
+ import { SelectionService } from '../../../../../../libs/api/repository/src'
5
+ import { SearchService } from '../utils/service/search.service'
6
+ import { RecordsRepositoryInterface } from '../../../../../../libs/common/domain/src/lib/repository/records-repository.interface'
7
+ import { ResultsTableComponent } from '../../../../../../libs/ui/search/src'
8
+ import { CommonModule } from '@angular/common'
9
+
10
+ @Component({
11
+ selector: 'gn-ui-results-table-container',
12
+ templateUrl: './results-table-container.component.html',
13
+ styleUrls: ['./results-table-container.component.css'],
14
+ standalone: true,
15
+ imports: [CommonModule, ResultsTableComponent],
16
+ })
17
+ export class ResultsTableContainerComponent {
18
+ @Output() recordClick = new EventEmitter<CatalogRecord>()
19
+
20
+ records$ = this.searchFacade.results$
21
+ selectedRecords$ = this.selectionService.selectedRecordsIdentifiers$
22
+ sortBy$ = this.searchFacade.sortBy$
23
+
24
+ hasDraft = (record: CatalogRecord): boolean =>
25
+ this.recordsRepository.recordHasDraft(record.uniqueIdentifier)
26
+
27
+ constructor(
28
+ private searchFacade: SearchFacade,
29
+ private searchService: SearchService,
30
+ private selectionService: SelectionService,
31
+ private recordsRepository: RecordsRepositoryInterface
32
+ ) {}
33
+
34
+ handleRecordClick(item: unknown) {
35
+ this.recordClick.emit(item as CatalogRecord)
36
+ }
37
+
38
+ handleSortByChange(col: string, order: 'asc' | 'desc') {
39
+ this.searchService.setSortBy([order, col])
40
+ }
41
+
42
+ handleRecordsSelectedChange(records: CatalogRecord[], selected: boolean) {
43
+ if (!selected) {
44
+ this.selectionService.deselectRecords(records)
45
+ } else {
46
+ this.selectionService.selectRecords(records)
47
+ }
48
+ }
49
+ }
@@ -37,6 +37,7 @@ import {
37
37
  getSearchResultsLoading,
38
38
  getSearchSortBy,
39
39
  getSpatialFilterEnabled,
40
+ isBeginningOfResults,
40
41
  isEndOfResults,
41
42
  totalPages,
42
43
  } from './selectors'
@@ -58,6 +59,7 @@ export class SearchFacade {
58
59
  layout$: Observable<string>
59
60
  sortBy$: Observable<SortByField>
60
61
  isLoading$: Observable<boolean>
62
+ isBeginningOfResults$: Observable<boolean>
61
63
  isEndOfResults$: Observable<boolean>
62
64
  totalPages$: Observable<number>
63
65
  currentPage$: Observable<number>
@@ -98,6 +100,9 @@ export class SearchFacade {
98
100
  this.isLoading$ = this.store.pipe(select(getSearchResultsLoading, searchId))
99
101
  this.searchFilters$ = this.store.pipe(select(getSearchFilters, searchId))
100
102
  this.resultsHits$ = this.store.pipe(select(getSearchResultsHits, searchId))
103
+ this.isBeginningOfResults$ = this.store.pipe(
104
+ select(isBeginningOfResults, searchId)
105
+ )
101
106
  this.isEndOfResults$ = this.store.pipe(select(isEndOfResults, searchId))
102
107
  this.totalPages$ = this.store.pipe(select(totalPages, searchId))
103
108
  this.currentPage$ = this.store.pipe(select(currentPage, searchId))
@@ -50,6 +50,13 @@ export const getSearchResultsHits = createSelector(
50
50
  (state: SearchStateSearch) => state.results.count
51
51
  )
52
52
 
53
+ export const isBeginningOfResults = createSelector(
54
+ getSearchStateSearch,
55
+ (state: SearchStateSearch) => {
56
+ return state.params.currentPage === 0
57
+ }
58
+ )
59
+
53
60
  export const isEndOfResults = createSelector(
54
61
  getSearchStateSearch,
55
62
  (state: SearchStateSearch) => {
@@ -1,14 +1,14 @@
1
1
  <a href (click)="clickOrganisation($event)" [attr.href]="organisationUrl">
2
2
  <div
3
3
  class="group cursor-pointer rounded-lg h-full flex flex-col"
4
- [title]="organisation.name"
4
+ [title]="organization.name"
5
5
  >
6
6
  <div
7
7
  class="shrink-0 bg-gray-100 rounded-lg overflow-hidden w-full border border-gray-300 h-36"
8
8
  >
9
9
  <gn-ui-thumbnail
10
10
  class="relative h-full w-full"
11
- [thumbnailUrl]="organisation.logoUrl"
11
+ [thumbnailUrl]="organization.logoUrl"
12
12
  [fit]="'contain'"
13
13
  >
14
14
  </gn-ui-thumbnail>
@@ -18,22 +18,24 @@
18
18
  class="shrink-0 mb-3 mt-5 font-title text-21 text-title group-hover:text-primary line-clamp-2 sm:mt-2 transition-colors"
19
19
  data-cy="organizationName"
20
20
  >
21
- {{ organisation.name }}</span
21
+ {{ organization.name }}</span
22
22
  >
23
23
  <p
24
24
  class="abstract mt-4 mb-5 sm:mb-2 sm:mt-0 grow shrink-1 overflow-hidden"
25
25
  data-cy="organizationDesc"
26
26
  >
27
- {{ organisation.description }}
27
+ {{ organization.description }}
28
28
  </p>
29
29
  <div class="shrink-0 text-primary opacity-50 flex leading-6">
30
30
  <mat-icon class="material-symbols-outlined text-primary opacity-50 mr-1"
31
31
  >folder_open
32
32
  </mat-icon>
33
33
  <span class="mx-1" data-cy="organizationRecordsCount">{{
34
- organisation.recordCount
34
+ organization.recordCount
35
35
  }}</span>
36
- <span translate>record.metadata.publications</span>
36
+ <span translate [translateParams]="{ count: organization.recordCount }"
37
+ >record.metadata.publications</span
38
+ >
37
39
  </div>
38
40
  </div>
39
41
  </div>
@@ -14,12 +14,12 @@ import { Organization } from '../../../../../../libs/common/domain/src/lib/model
14
14
  changeDetection: ChangeDetectionStrategy.OnPush,
15
15
  })
16
16
  export class OrganisationPreviewComponent {
17
- @Input() organisation: Organization
17
+ @Input() organization: Organization
18
18
  @Input() organisationUrl: string
19
19
  @Output() clickedOrganisation = new EventEmitter<Organization>()
20
20
 
21
21
  clickOrganisation(event: Event) {
22
22
  event.preventDefault()
23
- this.clickedOrganisation.emit(this.organisation)
23
+ this.clickedOrganisation.emit(this.organization)
24
24
  }
25
25
  }
@@ -9,6 +9,7 @@ import { OrganisationsFilterComponent } from './organisations-filter/organisatio
9
9
  import { UiInputsModule } from '../../../../../libs/ui/inputs/src'
10
10
  import { LanguageSwitcherComponent } from './language-switcher/language-switcher.component'
11
11
  import { OrganisationsResultComponent } from './organisations-result/organisations-result.component'
12
+ import { RouterLink } from '@angular/router'
12
13
 
13
14
  @NgModule({
14
15
  declarations: [
@@ -24,6 +25,7 @@ import { OrganisationsResultComponent } from './organisations-result/organisatio
24
25
  UiElementsModule,
25
26
  UiInputsModule,
26
27
  MatIconModule,
28
+ RouterLink,
27
29
  ],
28
30
  exports: [
29
31
  CatalogTitleComponent,
@@ -1,7 +1,13 @@
1
1
  <div
2
- translate
3
2
  class="flex flex-row justify-start items-center overflow-hidden"
4
- [title]="hoverTitle"
3
+ data-test="figureTitle"
4
+ [title]="
5
+ figure.toString() +
6
+ ' ' +
7
+ unit +
8
+ ' ' +
9
+ (title | translate: { count: figure })
10
+ "
5
11
  >
6
12
  <mat-icon
7
13
  class="material-symbols-outlined {{ bgClass }} {{
@@ -13,7 +19,9 @@
13
19
  </mat-icon>
14
20
  <div class="shrink overflow-hidden">
15
21
  <div class="figure-block text-[1.5em] text-black">
16
- <span class="figure font-medium mr-[0.3em]">{{ figure }}</span>
22
+ <span class="figure font-medium mr-[0.3em]" data-test="figure">{{
23
+ figure
24
+ }}</span>
17
25
  <span class="unit text-[0.665em]">{{ unit }}</span>
18
26
  </div>
19
27
  <div translate class="title truncate" [translateParams]="{ count: figure }">
@@ -1,4 +1,4 @@
1
- import { Component, ChangeDetectionStrategy, Input } from '@angular/core'
1
+ import { ChangeDetectionStrategy, Component, Input } from '@angular/core'
2
2
 
3
3
  @Component({
4
4
  selector: 'gn-ui-figure',
@@ -10,17 +10,13 @@ export class FigureComponent {
10
10
  @Input() icon!: string
11
11
  @Input() title!: string
12
12
  @Input() figure!: string | number
13
- @Input() unit?: string
13
+ @Input() unit = ''
14
14
  @Input() color: 'primary' | 'secondary' = 'primary'
15
15
 
16
- get hoverTitle() {
17
- return `${this.figure.toString()} ${this.unit || ''}
18
- ${this.title}`
19
- }
20
-
21
16
  get textClass() {
22
17
  return this.color === 'primary' ? 'text-primary' : 'text-secondary'
23
18
  }
19
+
24
20
  get bgClass() {
25
21
  return this.color === 'primary' ? 'bg-primary-white' : 'bg-secondary-white'
26
22
  }
@@ -8,7 +8,6 @@ export * from './lib/image-overlay-preview/image-overlay-preview.component'
8
8
  export * from './lib/link-card/link-card.component'
9
9
  export * from './lib/markdown-editor/markdown-editor.component'
10
10
  export * from './lib/markdown-parser/markdown-parser.component'
11
- export * from './lib/max-lines/max-lines.component'
12
11
  export * from './lib/metadata-catalog/metadata-catalog.component'
13
12
  export * from './lib/metadata-contact/metadata-contact.component'
14
13
  export * from './lib/metadata-info/metadata-info.component'
@@ -8,11 +8,11 @@
8
8
  <div class="relative opacity-40">
9
9
  <mat-icon class="material-symbols-outlined face">face</mat-icon>
10
10
  <mat-icon class="material-symbols-outlined question-mark1"
11
- >question_mark</mat-icon
12
- >
11
+ >question_mark
12
+ </mat-icon>
13
13
  <mat-icon class="material-symbols-outlined question-mark2"
14
- >question_mark</mat-icon
15
- >
14
+ >question_mark
15
+ </mat-icon>
16
16
  </div>
17
17
  <div translate>search.error.couldNotReachApi</div>
18
18
  </div>
@@ -32,6 +32,15 @@
32
32
  <div translate>search.error.receivedError</div>
33
33
  <div *ngIf="error">{{ error }}</div>
34
34
  </div>
35
+ <div *ngIf="type === types.ORGANIZATION_HAS_NO_DATASET">
36
+ <div class="relative opacity-40">
37
+ <mat-icon class="material-symbols-outlined computer">computer</mat-icon>
38
+ <mat-icon class="material-symbols-outlined computer-question-mark"
39
+ >question_mark
40
+ </mat-icon>
41
+ </div>
42
+ <div translate>search.error.organizationHasNoDataset</div>
43
+ </div>
35
44
  <div
36
45
  *ngIf="type === types.RECORD_NOT_FOUND"
37
46
  data-test="record-not-found-error"
@@ -39,12 +48,27 @@
39
48
  <div class="relative opacity-40">
40
49
  <mat-icon class="material-symbols-outlined computer">computer</mat-icon>
41
50
  <mat-icon class="material-symbols-outlined computer-question-mark"
42
- >question_mark</mat-icon
43
- >
51
+ >question_mark
52
+ </mat-icon>
44
53
  </div>
45
54
  <div translate [translateParams]="{ id: recordId }">
46
55
  search.error.recordNotFound
47
56
  </div>
48
57
  <div *ngIf="error">{{ error }}</div>
49
58
  </div>
59
+ <div
60
+ *ngIf="type === types.ORGANIZATION_NOT_FOUND"
61
+ data-test="org-not-found-error"
62
+ >
63
+ <div class="relative opacity-40">
64
+ <mat-icon class="material-symbols-outlined computer">computer</mat-icon>
65
+ <mat-icon class="material-symbols-outlined computer-question-mark"
66
+ >question_mark
67
+ </mat-icon>
68
+ </div>
69
+ <div translate [translateParams]="{ id: recordId }">
70
+ search.error.organizationNotFound
71
+ </div>
72
+ <div *ngIf="error">{{ error }}</div>
73
+ </div>
50
74
  </div>
@@ -5,6 +5,8 @@ export enum ErrorType {
5
5
  RECEIVED_ERROR,
6
6
  RECORD_NOT_FOUND,
7
7
  DATASET_HAS_NO_LINK,
8
+ ORGANIZATION_HAS_NO_DATASET,
9
+ ORGANIZATION_NOT_FOUND,
8
10
  }
9
11
 
10
12
  @Component({
@@ -1,5 +1,5 @@
1
1
  <a
2
- class="w-72 h-96 overflow-hidden rounded-lg bg-white cursor-pointer block hover:-translate-y-2 duration-[180ms]"
2
+ [class]="classList"
3
3
  [routerLink]="['/dataset', record.uniqueIdentifier]"
4
4
  target="_blank"
5
5
  >
@@ -1,4 +1,4 @@
1
- import { Component, ChangeDetectionStrategy, Input } from '@angular/core'
1
+ import { ChangeDetectionStrategy, Component, Input } from '@angular/core'
2
2
  import { CatalogRecord } from '../../../../../../libs/common/domain/src/lib/model/record'
3
3
 
4
4
  @Component({
@@ -8,5 +8,26 @@ import { CatalogRecord } from '../../../../../../libs/common/domain/src/lib/mode
8
8
  changeDetection: ChangeDetectionStrategy.OnPush,
9
9
  })
10
10
  export class RelatedRecordCardComponent {
11
+ private readonly baseClasses: string
12
+
11
13
  @Input() record: CatalogRecord
14
+ @Input() extraClass = ''
15
+
16
+ constructor() {
17
+ this.baseClasses = [
18
+ 'w-72',
19
+ 'h-96',
20
+ 'overflow-hidden',
21
+ 'rounded-lg',
22
+ 'bg-white',
23
+ 'cursor-pointer',
24
+ 'block',
25
+ 'hover:-translate-y-2 ',
26
+ 'duration-[180ms]',
27
+ ].join(' ')
28
+ }
29
+
30
+ get classList() {
31
+ return `${this.baseClasses} ${this.extraClass}`
32
+ }
12
33
  }
@@ -26,7 +26,6 @@ import { AvatarComponent } from './avatar/avatar.component'
26
26
  import { UserPreviewComponent } from './user-preview/user-preview.component'
27
27
  import { GnUiLinkifyDirective } from './metadata-info/linkify.directive'
28
28
  import { PaginationButtonsComponent } from './pagination-buttons/pagination-buttons.component'
29
- import { MaxLinesComponent } from './max-lines/max-lines.component'
30
29
  import { RecordApiFormComponent } from './record-api-form/record-api-form.component'
31
30
  import { MarkdownParserComponent } from './markdown-parser/markdown-parser.component'
32
31
  import { ImageOverlayPreviewComponent } from './image-overlay-preview/image-overlay-preview.component'
@@ -68,7 +67,6 @@ import { TimeSincePipe } from './user-feedback-item/time-since.pipe'
68
67
  UserPreviewComponent,
69
68
  GnUiLinkifyDirective,
70
69
  PaginationButtonsComponent,
71
- MaxLinesComponent,
72
70
  RecordApiFormComponent,
73
71
  UserFeedbackItemComponent,
74
72
  ImageOverlayPreviewComponent,
@@ -90,7 +88,6 @@ import { TimeSincePipe } from './user-feedback-item/time-since.pipe'
90
88
  AvatarComponent,
91
89
  UserPreviewComponent,
92
90
  PaginationButtonsComponent,
93
- MaxLinesComponent,
94
91
  RecordApiFormComponent,
95
92
  MarkdownParserComponent,
96
93
  UserFeedbackItemComponent,
@@ -3,7 +3,6 @@
3
3
  }
4
4
  .clear-btn {
5
5
  width: var(--input-height);
6
- right: var(--input-height);
7
6
  height: 100%;
8
7
  }
9
8
  .search-btn {
@@ -8,23 +8,27 @@
8
8
  [matAutocomplete]="auto"
9
9
  (keyup.enter)="handleEnter(searchInput.value)"
10
10
  />
11
- <button
12
- type="button"
13
- class="text-primary-lightest hover:text-primary hover:bg-gray-50 absolute transition-all duration-100 clear-btn inset-y-0"
14
- *ngIf="searchInput.value"
15
- aria-label="Clear"
16
- (click)="clear()"
17
- >
18
- <mat-icon class="material-symbols-outlined">close</mat-icon>
19
- </button>
20
- <button
21
- type="button"
22
- class="text-primary bg-white hover:text-primary-darkest hover:bg-gray-100 border-gray-300 hover:border-gray-500 absolute transition-all duration-100 search-btn rounded-r inset-y-0 right-0"
23
- aria-label="Trigger search"
24
- (click)="handleClickSearch()"
25
- >
26
- <mat-icon class="material-symbols-outlined">search</mat-icon>
27
- </button>
11
+ <div class="flex flex-row absolute inset-y-0 right-0">
12
+ <button
13
+ type="button"
14
+ class="text-primary-lightest hover:text-primary hover:bg-gray-50 transition-all duration-100 clear-btn"
15
+ *ngIf="searchInput.value"
16
+ aria-label="Clear"
17
+ (click)="clear()"
18
+ >
19
+ <mat-icon class="material-symbols-outlined">close</mat-icon>
20
+ </button>
21
+ <button
22
+ type="button"
23
+ class="text-primary bg-white hover:text-primary-darkest hover:bg-gray-100 border-gray-300 hover:border-gray-500 transition-all duration-100 search-btn rounded-r"
24
+ aria-label="Trigger search"
25
+ *ngIf="allowSubmit"
26
+ data-test="autocomplete-submit-btn"
27
+ (click)="handleClickSearch()"
28
+ >
29
+ <mat-icon class="material-symbols-outlined">search</mat-icon>
30
+ </button>
31
+ </div>
28
32
  <gn-ui-popup-alert
29
33
  *ngIf="error"
30
34
  class="absolute mt-2 w-full top-[100%] left-0"
@@ -39,13 +43,13 @@
39
43
  <mat-autocomplete
40
44
  #auto="matAutocomplete"
41
45
  (optionSelected)="handleSelection($event)"
42
- [displayWith]="displayWithFn"
46
+ [displayWith]="displayWithFnInternal"
43
47
  >
44
48
  <mat-option
45
49
  *ngFor="let suggestion of suggestions$ | async"
46
50
  [value]="suggestion"
47
51
  class="p-2 suggestion"
48
52
  >
49
- {{ displayWithFn(suggestion) }}
53
+ {{ displayWithFnInternal(suggestion) }}
50
54
  </mat-option>
51
55
  </mat-autocomplete>
@@ -13,25 +13,29 @@ import {
13
13
  SimpleChanges,
14
14
  ViewChild,
15
15
  } from '@angular/core'
16
- import { UntypedFormControl } from '@angular/forms'
16
+ import { ReactiveFormsModule, UntypedFormControl } from '@angular/forms'
17
17
  import {
18
18
  MatAutocomplete,
19
+ MatAutocompleteModule,
19
20
  MatAutocompleteSelectedEvent,
20
21
  MatAutocompleteTrigger,
21
22
  } from '@angular/material/autocomplete'
22
- import { merge, Observable, of, ReplaySubject, Subscription } from 'rxjs'
23
+ import { first, merge, Observable, of, ReplaySubject, Subscription } from 'rxjs'
23
24
  import {
24
25
  catchError,
25
26
  debounceTime,
26
27
  distinctUntilChanged,
27
28
  filter,
28
29
  finalize,
29
- first,
30
30
  map,
31
31
  switchMap,
32
32
  take,
33
33
  tap,
34
34
  } from 'rxjs/operators'
35
+ import { MatIconModule } from '@angular/material/icon'
36
+ import { PopupAlertComponent } from '../../../../../../libs/ui/widgets/src'
37
+ import { CommonModule } from '@angular/common'
38
+ import { TranslateModule } from '@ngx-translate/core'
35
39
 
36
40
  export type AutocompleteItem = unknown
37
41
 
@@ -40,6 +44,15 @@ export type AutocompleteItem = unknown
40
44
  templateUrl: './autocomplete.component.html',
41
45
  styleUrls: ['./autocomplete.component.css'],
42
46
  changeDetection: ChangeDetectionStrategy.OnPush,
47
+ standalone: true,
48
+ imports: [
49
+ MatIconModule,
50
+ PopupAlertComponent,
51
+ MatAutocompleteModule,
52
+ CommonModule,
53
+ TranslateModule,
54
+ ReactiveFormsModule,
55
+ ],
43
56
  })
44
57
  export class AutocompleteComponent
45
58
  implements OnInit, AfterViewInit, OnDestroy, OnChanges
@@ -49,6 +62,8 @@ export class AutocompleteComponent
49
62
  @Input() value?: AutocompleteItem
50
63
  @Input() clearOnSelection = false
51
64
  @Input() autoFocus = false
65
+ @Input() minCharacterCount? = 3
66
+ @Input() allowSubmit = true
52
67
  @Output() itemSelected = new EventEmitter<AutocompleteItem>()
53
68
  @Output() inputSubmitted = new EventEmitter<string>()
54
69
  @Output() inputCleared = new EventEmitter<void>()
@@ -57,22 +72,28 @@ export class AutocompleteComponent
57
72
  @ViewChild('searchInput') inputRef: ElementRef<HTMLInputElement>
58
73
 
59
74
  searching: boolean
60
- suggestions$: Observable<AutocompleteItem[]>
61
75
  control = new UntypedFormControl()
62
- subscription = new Subscription()
63
76
  cancelEnter = true
64
77
  selectionSubject = new ReplaySubject<MatAutocompleteSelectedEvent>(1)
65
78
  lastInputValue$ = new ReplaySubject<string>(1)
66
79
  error: string | null = null
80
+ suggestions$: Observable<AutocompleteItem[]>
81
+ subscription = new Subscription()
67
82
 
68
- @Input() displayWithFn: (AutocompleteItem) => string = (item) => item
83
+ @Input() displayWithFn: (item: AutocompleteItem) => string = (item) =>
84
+ item.toString()
85
+
86
+ displayWithFnInternal = (item?: AutocompleteItem) => {
87
+ if (item === null || item === undefined) return null
88
+ return this.displayWithFn(item)
89
+ }
69
90
 
70
91
  constructor(private cdRef: ChangeDetectorRef) {}
71
92
  ngOnChanges(changes: SimpleChanges): void {
72
93
  const { value } = changes
73
94
  if (value) {
74
- const previousTextValue = this.displayWithFn(value.previousValue)
75
- const currentTextValue = this.displayWithFn(value.currentValue)
95
+ const previousTextValue = this.displayWithFnInternal(value.previousValue)
96
+ const currentTextValue = this.displayWithFnInternal(value.currentValue)
76
97
  if (previousTextValue !== currentTextValue) {
77
98
  this.updateInputValue(value.currentValue)
78
99
  }
@@ -80,20 +101,33 @@ export class AutocompleteComponent
80
101
  }
81
102
 
82
103
  ngOnInit(): void {
83
- this.suggestions$ = merge(
104
+ const newValue$ = merge(
105
+ of(''),
106
+ this.inputCleared.pipe(map(() => '')),
84
107
  this.control.valueChanges.pipe(
85
108
  filter((value) => typeof value === 'string'),
86
- filter((value: string) => value.length > 2),
87
- debounceTime(400),
88
109
  distinctUntilChanged(),
89
- tap(() => (this.searching = true))
90
- ),
91
- this.control.valueChanges.pipe(
92
- filter((value) => typeof value === 'object' && value.title),
93
- map((item) => item.title)
110
+ debounceTime(400)
94
111
  )
112
+ )
113
+
114
+ const externalValueChange$ = this.control.valueChanges.pipe(
115
+ filter((value) => typeof value === 'object' && value.title),
116
+ map((item) => item.title)
117
+ )
118
+
119
+ // this observable emits arrays of suggestions loaded using the given action
120
+ const suggestionsFromAction = merge(
121
+ newValue$.pipe(
122
+ filter((value: string) => value.length >= this.minCharacterCount)
123
+ ),
124
+ externalValueChange$
95
125
  ).pipe(
96
- switchMap((value) => (value ? this.action(value) : of([]))),
126
+ tap(() => {
127
+ this.searching = true
128
+ this.error = null
129
+ }),
130
+ switchMap((value) => this.action(value)),
97
131
  catchError((error: Error) => {
98
132
  this.error = error.message
99
133
  return of([])
@@ -101,11 +135,32 @@ export class AutocompleteComponent
101
135
  finalize(() => (this.searching = false))
102
136
  )
103
137
 
104
- this.subscription = this.control.valueChanges.subscribe((any) => {
105
- if (any !== '') {
106
- this.cancelEnter = false
107
- }
108
- })
138
+ this.suggestions$ = merge(
139
+ suggestionsFromAction,
140
+ // if a new value is under the min char count, clear suggestions
141
+ newValue$.pipe(
142
+ filter((value: string) => value.length < this.minCharacterCount),
143
+ map(() => [])
144
+ )
145
+ )
146
+
147
+ // close the panel whenever suggestions are cleared
148
+ this.subscription.add(
149
+ this.suggestions$
150
+ .pipe(filter((suggestions) => suggestions.length === 0))
151
+ .subscribe(() => {
152
+ this.triggerRef?.closePanel()
153
+ })
154
+ )
155
+
156
+ this.subscription.add(
157
+ this.control.valueChanges.subscribe((any) => {
158
+ if (any !== '') {
159
+ this.cancelEnter = false
160
+ }
161
+ })
162
+ )
163
+
109
164
  this.control.valueChanges
110
165
  .pipe(filter((value) => typeof value === 'string'))
111
166
  .subscribe(this.lastInputValue$)
@@ -120,7 +175,7 @@ export class AutocompleteComponent
120
175
  }
121
176
 
122
177
  ngOnDestroy(): void {
123
- this.subscription.unsubscribe()
178
+ this.subscription?.unsubscribe()
124
179
  }
125
180
 
126
181
  updateInputValue(value: AutocompleteItem) {
@@ -139,19 +194,16 @@ export class AutocompleteComponent
139
194
  .pipe(take(1))
140
195
  .subscribe((selection) => selection && selection.option.deselect())
141
196
  this.inputRef.nativeElement.focus()
142
- this.triggerRef.closePanel()
143
197
  }
144
198
 
145
199
  handleEnter(any: string) {
146
- if (!this.cancelEnter) {
200
+ if (!this.cancelEnter && this.allowSubmit) {
147
201
  this.inputSubmitted.emit(any)
148
- this.triggerRef.closePanel()
149
202
  }
150
203
  }
151
204
 
152
205
  handleClickSearch() {
153
206
  this.inputSubmitted.emit(this.inputRef.nativeElement.value)
154
- this.triggerRef.closePanel()
155
207
  }
156
208
 
157
209
  handleSelection(event: MatAutocompleteSelectedEvent) {
@@ -4,7 +4,6 @@ import { UtilSharedModule } from '../../../../../libs/util/shared/src'
4
4
  import { TranslateModule } from '@ngx-translate/core'
5
5
  import { TagInputModule } from 'ngx-chips'
6
6
  import { NgxDropzoneModule } from 'ngx-dropzone'
7
- import { AutocompleteComponent } from './autocomplete/autocomplete.component'
8
7
  import { ButtonComponent } from './button/button.component'
9
8
  import { BadgeComponent } from './badge/badge.component'
10
9
  import { ChipsInputComponent } from './chips-input/chips-input.component'
@@ -37,7 +36,6 @@ import { ImageInputComponent } from './image-input/image-input.component'
37
36
 
38
37
  @NgModule({
39
38
  declarations: [
40
- AutocompleteComponent,
41
39
  TextInputComponent,
42
40
  DragAndDropFileInputComponent,
43
41
  ChipsInputComponent,
@@ -78,7 +76,6 @@ import { ImageInputComponent } from './image-input/image-input.component'
78
76
  ],
79
77
  exports: [
80
78
  DropdownSelectorComponent,
81
- AutocompleteComponent,
82
79
  ButtonComponent,
83
80
  TextInputComponent,
84
81
  DragAndDropFileInputComponent,