geonetwork-ui 2.6.0-dev.ceb4be4c1 → 2.6.0-dev.d1dbf336f

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 (69) hide show
  1. package/esm2022/libs/api/metadata-converter/src/lib/iso19115-3/read-parts.mjs +8 -4
  2. package/esm2022/libs/api/metadata-converter/src/lib/iso19115-3/write-parts.mjs +5 -2
  3. package/esm2022/libs/api/metadata-converter/src/lib/iso19139/read-parts.mjs +3 -3
  4. package/esm2022/libs/api/metadata-converter/src/lib/iso19139/write-parts.mjs +2 -2
  5. package/esm2022/libs/api/repository/src/lib/gn4/gn4-repository.mjs +14 -5
  6. package/esm2022/libs/common/domain/src/lib/repository/records-repository.interface.mjs +1 -1
  7. package/esm2022/libs/feature/editor/src/lib/components/multilingual-panel/multilingual-panel.component.mjs +266 -11
  8. package/esm2022/libs/ui/search/src/index.mjs +2 -1
  9. package/esm2022/libs/ui/search/src/lib/results-table/action-menu/action-menu.component.mjs +19 -12
  10. package/esm2022/libs/ui/search/src/lib/results-table/results-table.component.mjs +3 -3
  11. package/esm2022/libs/ui/search/src/lib/ui-search.module.mjs +10 -4
  12. package/esm2022/libs/util/i18n/src/lib/i18n.constants.mjs +42 -1
  13. package/esm2022/libs/util/i18n/src/lib/language-codes.mjs +24 -2
  14. package/esm2022/translations/de.json +28 -4
  15. package/esm2022/translations/en.json +28 -4
  16. package/esm2022/translations/es.json +27 -3
  17. package/esm2022/translations/fr.json +28 -4
  18. package/esm2022/translations/it.json +27 -3
  19. package/esm2022/translations/nl.json +27 -3
  20. package/esm2022/translations/pt.json +27 -3
  21. package/fesm2022/geonetwork-ui.mjs +604 -107
  22. package/fesm2022/geonetwork-ui.mjs.map +1 -1
  23. package/libs/api/metadata-converter/src/lib/iso19115-3/read-parts.d.ts +1 -0
  24. package/libs/api/metadata-converter/src/lib/iso19115-3/read-parts.d.ts.map +1 -1
  25. package/libs/api/metadata-converter/src/lib/iso19115-3/write-parts.d.ts.map +1 -1
  26. package/libs/api/metadata-converter/src/lib/iso19139/read-parts.d.ts.map +1 -1
  27. package/libs/api/metadata-converter/src/lib/iso19139/write-parts.d.ts.map +1 -1
  28. package/libs/api/repository/src/lib/gn4/gn4-repository.d.ts +4 -2
  29. package/libs/api/repository/src/lib/gn4/gn4-repository.d.ts.map +1 -1
  30. package/libs/common/domain/src/lib/repository/records-repository.interface.d.ts +1 -0
  31. package/libs/common/domain/src/lib/repository/records-repository.interface.d.ts.map +1 -1
  32. package/libs/feature/editor/src/lib/components/multilingual-panel/multilingual-panel.component.d.ts +48 -3
  33. package/libs/feature/editor/src/lib/components/multilingual-panel/multilingual-panel.component.d.ts.map +1 -1
  34. package/libs/ui/search/src/index.d.ts +1 -0
  35. package/libs/ui/search/src/index.d.ts.map +1 -1
  36. package/libs/ui/search/src/lib/results-table/action-menu/action-menu.component.d.ts +3 -3
  37. package/libs/ui/search/src/lib/results-table/action-menu/action-menu.component.d.ts.map +1 -1
  38. package/libs/ui/search/src/lib/ui-search.module.d.ts +2 -1
  39. package/libs/ui/search/src/lib/ui-search.module.d.ts.map +1 -1
  40. package/libs/util/i18n/src/lib/i18n.constants.d.ts +1 -0
  41. package/libs/util/i18n/src/lib/i18n.constants.d.ts.map +1 -1
  42. package/libs/util/i18n/src/lib/language-codes.d.ts +23 -1
  43. package/libs/util/i18n/src/lib/language-codes.d.ts.map +1 -1
  44. package/package.json +1 -1
  45. package/src/libs/api/metadata-converter/src/lib/fixtures/geocat-ch.records.ts +1 -1
  46. package/src/libs/api/metadata-converter/src/lib/iso19115-3/read-parts.ts +13 -3
  47. package/src/libs/api/metadata-converter/src/lib/iso19115-3/write-parts.ts +5 -1
  48. package/src/libs/api/metadata-converter/src/lib/iso19139/read-parts.ts +6 -3
  49. package/src/libs/api/metadata-converter/src/lib/iso19139/write-parts.ts +4 -1
  50. package/src/libs/api/repository/src/lib/gn4/gn4-repository.ts +16 -2
  51. package/src/libs/common/domain/src/lib/repository/records-repository.interface.ts +1 -0
  52. package/src/libs/feature/editor/src/lib/components/multilingual-panel/multilingual-panel.component.html +117 -11
  53. package/src/libs/feature/editor/src/lib/components/multilingual-panel/multilingual-panel.component.ts +316 -6
  54. package/src/libs/ui/search/src/index.ts +1 -0
  55. package/src/libs/ui/search/src/lib/results-table/action-menu/action-menu.component.html +11 -3
  56. package/src/libs/ui/search/src/lib/results-table/action-menu/action-menu.component.ts +14 -11
  57. package/src/libs/ui/search/src/lib/results-table/results-table.component.html +1 -1
  58. package/src/libs/ui/search/src/lib/ui-search.module.ts +3 -0
  59. package/src/libs/util/i18n/src/lib/i18n.constants.ts +42 -0
  60. package/src/libs/util/i18n/src/lib/language-codes.ts +23 -1
  61. package/tailwind.base.css +1 -1
  62. package/translations/de.json +28 -4
  63. package/translations/en.json +28 -4
  64. package/translations/es.json +27 -3
  65. package/translations/fr.json +28 -4
  66. package/translations/it.json +27 -3
  67. package/translations/nl.json +27 -3
  68. package/translations/pt.json +27 -3
  69. package/translations/sk.json +27 -3
@@ -1,14 +1,120 @@
1
1
  <div
2
- class="flex flex-col h-full w-[302px] bg-neutral-100 border-l border-gray-300 py-9 px-3 gap-8"
2
+ class="flex flex-col h-full w-[302px] bg-neutral-100 border-l border-gray-300 py-8 px-3 gap-6 overflow-auto"
3
3
  >
4
- <span class="text-3xl font-title text-black/80 px-2" translate
5
- >editor.record.form.multilingual.title</span
6
- >
7
- <gn-ui-check-toggle
8
- class="p-2"
9
- [label]="'editor.record.form.multilingual.enable' | translate"
10
- [color]="'primary'"
11
- [value]="translationsEnabled"
12
- (toggled)="translationsEnabled = $event"
13
- ></gn-ui-check-toggle>
4
+ <div class="flex flex-row px-2 justify-between">
5
+ <span class="text-3xl font-title text-black/80" translate
6
+ >editor.record.form.multilingual.title</span
7
+ >
8
+ <button
9
+ [title]="'editor.record.form.multilingual.open' | translate"
10
+ (click)="toggleLanguageSelection()"
11
+ *ngIf="isMultilingual"
12
+ data-test="activateSelection"
13
+ >
14
+ <ng-icon class="mt-1" name="iconoirSettings"></ng-icon>
15
+ </button>
16
+ </div>
17
+ <div class="flex flex-col gap-2" *ngIf="editTranslations || !isMultilingual">
18
+ <gn-ui-check-toggle
19
+ class="p-2"
20
+ [label]="'editor.record.form.multilingual.enable' | translate"
21
+ [color]="'primary'"
22
+ [(value)]="isMultilingual"
23
+ (toggled)="switchMultilingual($event)"
24
+ ></gn-ui-check-toggle>
25
+ <div *ngIf="isMultilingual" class="flex flex-col gap-2">
26
+ <div class="flex flex-row justify-between border-t border-gray-300 p-3">
27
+ <span class="mt-2 text-sm text-gray-600" translate
28
+ >editor.record.form.multilingual.activate</span
29
+ >
30
+ <gn-ui-button
31
+ extraClass="w-16 h-8 font-bold"
32
+ type="gray"
33
+ (buttonClick)="validateTranslations()"
34
+ data-test="validateSelection"
35
+ >{{ 'editor.record.form.multilingual.validate' | translate }}
36
+ </gn-ui-button>
37
+ </div>
38
+ <ng-container *ngIf="supportedLanguages$ | async as languages">
39
+ <div
40
+ class="flex flex-col gap-2 w-full px-2"
41
+ data-test="langAvailable"
42
+ *ngFor="let lang of languages"
43
+ >
44
+ <gn-ui-button
45
+ [extraClass]="getExtraClass(lang)"
46
+ type="gray"
47
+ (buttonClick)="toggleLanguage(lang)"
48
+ [disabled]="lang === _record.defaultLanguage"
49
+ [title]="getToggleTitle(lang)"
50
+ >
51
+ <span [class]="getIconClass(lang)"></span>
52
+ <span class="ml-2">{{ 'language.' + lang | translate }}</span>
53
+ </gn-ui-button>
54
+ </div>
55
+ </ng-container>
56
+ </div>
57
+ </div>
58
+ <div *ngIf="!editTranslations && isMultilingual" class="flex flex-col gap-2">
59
+ <gn-ui-button
60
+ *ngFor="let recordLang of sortLanguages(recordLanguages); let i = index"
61
+ extraClass="flex flex-row justify-between bg-white border border-white rounded mb-1 h-[34px] w-full"
62
+ [ngClass]="{
63
+ 'mt-8': isFirstUnsupported(i),
64
+ '': true,
65
+ }"
66
+ (buttonClick)="switchFormLang(recordLang)"
67
+ type="outline"
68
+ data-test="langSwitch"
69
+ >
70
+ <div class="flex flex-row gap-2 items-center">
71
+ <ng-icon
72
+ *ngIf="recordLang === formLanguage"
73
+ class="text-primary mt-1"
74
+ name="iconoirCheckCircle"
75
+ ></ng-icon>
76
+ <ng-icon
77
+ *ngIf="recordLang !== formLanguage"
78
+ class="text-gray-800 mt-1"
79
+ name="iconoirCircle"
80
+ ></ng-icon>
81
+ <span
82
+ *ngIf="recordLang.length === 2"
83
+ [class]="getIconClass(recordLang) + 'mt-1'"
84
+ ></span>
85
+ <span [ngClass]="recordLang === formLanguage ? 'text-black' : ''">{{
86
+ isLangSupported(recordLang)
87
+ ? ('language.' + recordLang | translate)
88
+ : recordLang.toUpperCase()
89
+ }}</span>
90
+ </div>
91
+ <div class="flex flex-row gap-2 items-center">
92
+ <span
93
+ *ngIf="recordLang === formLanguage"
94
+ class="text-xs text-base"
95
+ translate
96
+ >editor.record.form.multilingual.default</span
97
+ >
98
+ <button
99
+ (click)="
100
+ openActionMenu(recordLang, template); $event.stopPropagation()
101
+ "
102
+ cdkOverlayOrigin
103
+ #actionMenuButton
104
+ >
105
+ <ng-icon class="pb-5" name="matMoreHorizOutline"></ng-icon>
106
+ </button>
107
+ <ng-template #template>
108
+ <gn-ui-action-menu
109
+ [canDelete]="recordLang !== _record.defaultLanguage"
110
+ page="record"
111
+ (delete)="confirmDeleteAction(recordLang)"
112
+ (closeActionMenu)="closeActionMenu()"
113
+ (switch)="switchDefaultLang(recordLang)"
114
+ >
115
+ </gn-ui-action-menu>
116
+ </ng-template>
117
+ </div>
118
+ </gn-ui-button>
119
+ </div>
14
120
  </div>
@@ -1,15 +1,325 @@
1
- import { Component } from '@angular/core'
1
+ import {
2
+ ChangeDetectorRef,
3
+ Component,
4
+ ElementRef,
5
+ Input,
6
+ OnDestroy,
7
+ QueryList,
8
+ ViewChildren,
9
+ ViewContainerRef,
10
+ } from '@angular/core'
2
11
  import { CommonModule } from '@angular/common'
3
- import { CheckToggleComponent } from '../../../../../../../libs/ui/inputs/src'
4
- import { TranslateModule } from '@ngx-translate/core'
12
+ import { ButtonComponent, CheckToggleComponent } from '../../../../../../../libs/ui/inputs/src'
13
+ import { TranslateModule, TranslateService } from '@ngx-translate/core'
14
+ import { CatalogRecord } from '../../../../../../../libs/common/domain/src/lib/model/record'
15
+ import {
16
+ NgIconComponent,
17
+ provideIcons,
18
+ provideNgIconsConfig,
19
+ } from '@ng-icons/core'
20
+ import {
21
+ iconoirCheckCircle,
22
+ iconoirCircle,
23
+ iconoirSettings,
24
+ } from '@ng-icons/iconoir'
25
+ import { matMoreHorizOutline } from '@ng-icons/material-icons/outline'
26
+ import { EditorFacade } from '../../+state/editor.facade'
27
+ import { ConfirmationDialogComponent } from '../../../../../../../libs/ui/elements/src'
28
+ import { MatDialog } from '@angular/material/dialog'
29
+ import { RecordsRepositoryInterface } from '../../../../../../../libs/common/domain/src/lib/repository/records-repository.interface'
30
+ import { map, Subscription } from 'rxjs'
31
+ import { Overlay, OverlayRef } from '@angular/cdk/overlay'
32
+ import { TemplatePortal } from '@angular/cdk/portal'
33
+ import { ActionMenuComponent } from '../../../../../../../libs/ui/search/src'
34
+
35
+ const extraFlagMap: { [key: string]: string } = {
36
+ ar: 'arab',
37
+ en: 'gb',
38
+ ko: 'kr',
39
+ cs: 'cz',
40
+ zh: 'cn',
41
+ ca: 'es-ct',
42
+ rm: 'ch',
43
+ da: 'dk',
44
+ sv: 'se',
45
+ cy: 'gb-wls',
46
+ hy: 'am',
47
+ ka: 'ge',
48
+ uk: 'ua',
49
+ }
5
50
 
6
51
  @Component({
7
52
  selector: 'gn-ui-multilingual-panel',
8
53
  standalone: true,
9
- imports: [CommonModule, CheckToggleComponent, TranslateModule],
54
+ imports: [
55
+ CommonModule,
56
+ CheckToggleComponent,
57
+ TranslateModule,
58
+ ButtonComponent,
59
+ NgIconComponent,
60
+ ActionMenuComponent,
61
+ ],
62
+ providers: [
63
+ provideIcons({
64
+ iconoirSettings,
65
+ matMoreHorizOutline,
66
+ iconoirCheckCircle,
67
+ iconoirCircle,
68
+ }),
69
+ provideNgIconsConfig({
70
+ size: '1.25em',
71
+ }),
72
+ ],
10
73
  templateUrl: './multilingual-panel.component.html',
11
74
  styleUrl: './multilingual-panel.component.css',
12
75
  })
13
- export class MultilingualPanelComponent {
14
- translationsEnabled = false
76
+ export class MultilingualPanelComponent implements OnDestroy {
77
+ isMultilingual: boolean
78
+ _record: CatalogRecord
79
+ editTranslations: boolean
80
+ selectedLanguages = []
81
+ recordLanguages = []
82
+ formLanguage = ''
83
+ @Input() set record(value: CatalogRecord) {
84
+ this._record = value
85
+ this.isMultilingual = value.otherLanguages.length > 0
86
+ this.editTranslations = false
87
+ this.recordLanguages = [...value.otherLanguages, value.defaultLanguage]
88
+ this.selectedLanguages = this.recordLanguages
89
+ this.formLanguage = value.defaultLanguage
90
+ }
91
+ @ViewChildren('actionMenuButton', { read: ElementRef })
92
+ actionMenuButtons!: QueryList<ElementRef>
93
+ private overlayRef!: OverlayRef
94
+
95
+ isActionMenuOpen = false
96
+ subscription = new Subscription()
97
+
98
+ supportedLanguages$ = this.recordsRepository
99
+ .getApplicationLanguages()
100
+ .pipe(map((languages) => this.sortLanguages(languages)))
101
+
102
+ constructor(
103
+ public facade: EditorFacade,
104
+ public dialog: MatDialog,
105
+ private translateService: TranslateService,
106
+ private recordsRepository: RecordsRepositoryInterface,
107
+ private overlay: Overlay,
108
+ private viewContainerRef: ViewContainerRef,
109
+ private cdr: ChangeDetectorRef
110
+ ) {}
111
+
112
+ ngOnDestroy() {
113
+ this.subscription.unsubscribe()
114
+ }
115
+
116
+ sortLanguages(languages: string[]) {
117
+ return languages
118
+ .map((lang) => {
119
+ const label = this.translateService.instant('language.' + lang)
120
+ const isTranslated = label !== 'language.' + lang
121
+
122
+ return {
123
+ lang,
124
+ label,
125
+ isTranslated,
126
+ }
127
+ })
128
+ .sort((a, b) => {
129
+ if (a.isTranslated && !b.isTranslated) return -1
130
+ if (!a.isTranslated && b.isTranslated) return 1
131
+
132
+ return a.label.localeCompare(b.label)
133
+ })
134
+ .map((item) => item.lang)
135
+ }
136
+
137
+ toggleLanguageSelection() {
138
+ this.editTranslations = !this.editTranslations
139
+ }
140
+
141
+ getIconClass(lang: string) {
142
+ return extraFlagMap[lang]
143
+ ? `fi fi-${extraFlagMap[lang]} w-4 h-3`
144
+ : `fi fi-${lang} w-4 h-3`
145
+ }
146
+
147
+ switchMultilingual() {
148
+ if (this.isMultilingual && this.selectedLanguages.length > 1) {
149
+ this.confirmDeleteAction()
150
+ } else {
151
+ this.isMultilingual = true
152
+ this.editTranslations = true
153
+ }
154
+ }
155
+
156
+ getExtraClass(lang: string) {
157
+ const baseClass = 'h-[34px] w-full font-bold justify-start hover:bg-white'
158
+ if (this.selectedLanguages.includes(lang)) {
159
+ return `${baseClass} bg-white border border-black`
160
+ }
161
+ return baseClass
162
+ }
163
+
164
+ toggleLanguage(lang: string) {
165
+ if (this.selectedLanguages.includes(lang)) {
166
+ this.removeSelectedLanguage(lang)
167
+ } else {
168
+ this.selectedLanguages.push(lang)
169
+ }
170
+ }
171
+
172
+ removeSelectedLanguage(lang: string) {
173
+ this.selectedLanguages = this.selectedLanguages.filter(
174
+ (language) => language !== lang
175
+ )
176
+ }
177
+
178
+ validateTranslations() {
179
+ const equalLength =
180
+ this.selectedLanguages.length === this.recordLanguages.length
181
+ if (
182
+ this.selectedLanguages.length < this.recordLanguages.length ||
183
+ (equalLength && this.selectedLanguages !== this.recordLanguages)
184
+ ) {
185
+ this.confirmDeleteAction(this.selectedLanguages)
186
+ } else {
187
+ this.updateTranslations()
188
+ }
189
+ }
190
+
191
+ updateTranslations() {
192
+ this.facade.updateRecordField(
193
+ 'otherLanguages',
194
+ this.selectedLanguages.filter((lang) => lang !== this.formLanguage)
195
+ )
196
+ this.recordLanguages = this.selectedLanguages
197
+ this.editTranslations = false
198
+ }
199
+
200
+ switchFormLang(lang) {
201
+ // TO IMPLEMENT FURTHER
202
+ }
203
+
204
+ switchDefaultLang(lang: string) {
205
+ this.formLanguage = lang
206
+ this.facade.updateRecordField('defaultLanguage', lang)
207
+ this.facade.updateRecordField(
208
+ 'otherLanguages',
209
+ this.selectedLanguages.filter((lang) => lang !== this.formLanguage)
210
+ )
211
+ this.closeActionMenu()
212
+ }
213
+
214
+ confirmDeleteAction(lang?: string[] | string) {
215
+ const dialogRef = this.dialog.open(ConfirmationDialogComponent, {
216
+ data: {
217
+ title: this.translateService.instant(
218
+ 'editor.record.multilingual.confirmation.title'
219
+ ),
220
+ message: this.translateService.instant(
221
+ 'editor.record.multilingual.confirmation.message'
222
+ ),
223
+ confirmText: this.translateService.instant(
224
+ 'editor.record.multilingual.confirmation.confirmText'
225
+ ),
226
+ cancelText: this.translateService.instant(
227
+ 'editor.record.multilingual.confirmation.cancelText'
228
+ ),
229
+ },
230
+ restoreFocus: true,
231
+ })
232
+ this.subscription.add(
233
+ dialogRef.afterClosed().subscribe((confirmed) => {
234
+ if (confirmed) {
235
+ if (lang) {
236
+ if (!Array.isArray(lang)) {
237
+ this.removeSelectedLanguage(lang)
238
+ this.closeActionMenu()
239
+ }
240
+ this.updateTranslations()
241
+ } else {
242
+ this.facade.updateRecordField('otherLanguages', [])
243
+ this.isMultilingual = false
244
+ this.selectedLanguages = []
245
+ }
246
+ } else {
247
+ this.isMultilingual = true
248
+ this.selectedLanguages = this.recordLanguages
249
+ }
250
+ this.editTranslations = false
251
+ })
252
+ )
253
+ }
254
+
255
+ isFirstUnsupported(index: number): boolean {
256
+ const langs = this.sortLanguages(this.recordLanguages)
257
+ return (
258
+ langs[index].length === 3 &&
259
+ langs.slice(0, index).every((lang) => lang.length !== 3)
260
+ )
261
+ }
262
+
263
+ isLangSupported(lang: string) {
264
+ return lang.length === 2
265
+ }
266
+
267
+ getToggleTitle(lang: string) {
268
+ if (lang === this._record.defaultLanguage) {
269
+ return this.translateService.instant(
270
+ 'editor.record.form.multilingual.forbidden'
271
+ )
272
+ }
273
+ return ''
274
+ }
275
+
276
+ openActionMenu(item: string, template) {
277
+ this.isActionMenuOpen = true
278
+ const index = this.sortLanguages(this.selectedLanguages).indexOf(item)
279
+ const buttonElement = this.actionMenuButtons.toArray()[index]
280
+
281
+ const positionStrategy = this.overlay
282
+ .position()
283
+ .flexibleConnectedTo(buttonElement)
284
+ .withFlexibleDimensions(true)
285
+ .withPush(true)
286
+ .withPositions([
287
+ {
288
+ originX: 'end',
289
+ originY: 'bottom',
290
+ overlayX: 'end',
291
+ overlayY: 'top',
292
+ },
293
+ {
294
+ originX: 'end',
295
+ originY: 'top',
296
+ overlayX: 'end',
297
+ overlayY: 'bottom',
298
+ },
299
+ ])
300
+
301
+ this.overlayRef = this.overlay.create({
302
+ hasBackdrop: true,
303
+ backdropClass: 'cdk-overlay-transparent-backdrop',
304
+ positionStrategy: positionStrategy,
305
+ scrollStrategy: this.overlay.scrollStrategies.reposition(),
306
+ })
307
+
308
+ const portal = new TemplatePortal(template, this.viewContainerRef)
309
+
310
+ this.overlayRef.attach(portal)
311
+ this.subscription.add(
312
+ this.overlayRef.backdropClick().subscribe(() => {
313
+ this.closeActionMenu()
314
+ })
315
+ )
316
+ }
317
+
318
+ closeActionMenu() {
319
+ if (this.overlayRef) {
320
+ this.isActionMenuOpen = false
321
+ this.overlayRef.dispose()
322
+ this.cdr.markForCheck()
323
+ }
324
+ }
15
325
  }
@@ -19,3 +19,4 @@ export * from './lib/results-list-item/results-list-item.component'
19
19
  export * from './lib/results-hits-search-kind/results-hits-search-kind.component'
20
20
  export * from './lib/results-hits-number/results-hits-number.component'
21
21
  export * from './lib/results-table/results-table.component'
22
+ export * from './lib/results-table/action-menu/action-menu.component'
@@ -6,7 +6,7 @@
6
6
  >
7
7
  <ul class="flex flex-col gap-2 w-full">
8
8
  <gn-ui-button
9
- *ngIf="!isDraftPage"
9
+ *ngIf="page === 'main'"
10
10
  type="light"
11
11
  extraClass="flex flex-row items-center gap-2 w-full justify-start"
12
12
  (buttonClick)="duplicate.emit()"
@@ -18,14 +18,22 @@
18
18
  >record.action.duplicating</span
19
19
  ></gn-ui-button
20
20
  >
21
+ <gn-ui-button
22
+ *ngIf="page === 'record'"
23
+ type="light"
24
+ extraClass="flex flex-row items-center gap-2 w-full justify-start"
25
+ (buttonClick)="switch.emit()"
26
+ data-test="record-menu-switch-button"
27
+ ><span translate>record.action.switchLang</span></gn-ui-button
28
+ >
21
29
  <gn-ui-button
22
30
  type="light"
23
31
  extraClass="flex flex-row items-center gap-2 w-full justify-start"
24
32
  (buttonClick)="displayDeleteMenu()"
25
33
  [disabled]="!canDelete"
26
34
  data-test="record-menu-delete-button"
27
- ><span *ngIf="!isDraftPage" translate>record.action.delete</span>
28
- <span *ngIf="isDraftPage" translate
35
+ ><span *ngIf="page !== 'draft'" translate>record.action.delete</span>
36
+ <span *ngIf="page === 'draft'" translate
29
37
  >record.action.rollback</span
30
38
  ></gn-ui-button
31
39
  >
@@ -30,13 +30,14 @@ type ActionMenuPage = 'mainMenu' | 'deleteMenu' | 'rollbackMenu'
30
30
  ],
31
31
  })
32
32
  export class ActionMenuComponent {
33
- @Input() canDuplicate: boolean
34
- @Input() canDelete: boolean
35
- @Input() isDraftPage: boolean
33
+ @Input() canDuplicate = true
34
+ @Input() canDelete = true
35
+ @Input() page: 'draft' | 'main' | 'record'
36
36
  @Output() duplicate = new EventEmitter<void>()
37
37
  @Output() delete = new EventEmitter<void>()
38
38
  @Output() closeActionMenu = new EventEmitter<void>()
39
39
  @Output() rollback = new EventEmitter<void>()
40
+ @Output() switch = new EventEmitter<void>()
40
41
 
41
42
  @ViewChild(MatMenuTrigger) trigger: MatMenuTrigger
42
43
 
@@ -47,20 +48,22 @@ export class ActionMenuComponent {
47
48
  private cdr: ChangeDetectorRef
48
49
  ) {}
49
50
 
50
- openMenu() {
51
- this.trigger.openMenu()
52
- }
53
-
54
51
  displayMainMenu() {
55
52
  this.sectionDisplayed = 'mainMenu'
56
53
  this.cdr.markForCheck()
57
54
  }
58
55
 
59
56
  displayDeleteMenu() {
60
- if (this.isDraftPage) {
61
- this.sectionDisplayed = 'rollbackMenu'
62
- } else {
63
- this.sectionDisplayed = 'deleteMenu'
57
+ switch (this.page) {
58
+ case 'draft':
59
+ this.sectionDisplayed = 'rollbackMenu'
60
+ break
61
+ case 'record':
62
+ this.delete.emit()
63
+ break
64
+ case 'main':
65
+ default:
66
+ this.sectionDisplayed = 'deleteMenu'
64
67
  }
65
68
  this.cdr.markForCheck()
66
69
  }
@@ -175,7 +175,7 @@
175
175
  <gn-ui-action-menu
176
176
  [canDuplicate]="canDuplicate(item) && !isDuplicating"
177
177
  [canDelete]="canDelete(item)"
178
- [isDraftPage]="isDraftPage"
178
+ [page]="isDraftPage ? 'draft' : 'main'"
179
179
  (duplicate)="handleDuplicate(item)"
180
180
  (delete)="handleDelete(item)"
181
181
  (closeActionMenu)="closeActionMenu()"
@@ -39,6 +39,7 @@ import {
39
39
  matMapOutline,
40
40
  } from '@ng-icons/material-icons/outline'
41
41
  import { matFace } from '@ng-icons/material-icons/baseline'
42
+ import { ActionMenuComponent } from './results-table/action-menu/action-menu.component'
42
43
 
43
44
  @NgModule({
44
45
  declarations: [
@@ -79,6 +80,7 @@ import { matFace } from '@ng-icons/material-icons/baseline'
79
80
  }),
80
81
  KindBadgeComponent,
81
82
  MetadataQualityComponent,
83
+ ActionMenuComponent,
82
84
  ],
83
85
  exports: [
84
86
  RecordPreviewListComponent,
@@ -93,6 +95,7 @@ import { matFace } from '@ng-icons/material-icons/baseline'
93
95
  ResultsHitsSearchKindComponent,
94
96
  RecordPreviewFeedComponent,
95
97
  RecordPreviewRowComponent,
98
+ ActionMenuComponent,
96
99
  ],
97
100
  providers: [
98
101
  provideNgIconsConfig({
@@ -19,12 +19,54 @@ marker('language.pt')
19
19
  marker('language.ru')
20
20
  marker('language.zh')
21
21
  marker('language.sk')
22
+ marker('language.rm')
23
+ marker('language.ar')
24
+ marker('language.da')
25
+ marker('language.no')
26
+ marker('language.pl')
27
+ marker('language.sv')
28
+ marker('language.tr')
29
+ marker('language.hy')
30
+ marker('language.az')
31
+ marker('language.ka')
32
+ marker('language.uk')
33
+ marker('language.cy')
22
34
 
23
35
  export const DEFAULT_LANG = 'en'
24
36
 
25
37
  // Caution: changing this can break language selection from third parties!
26
38
  export const LANGUAGE_STORAGE_KEY = 'geonetwork-ui-language'
27
39
 
40
+ export const SUPPORTED_LANGUAGES = [
41
+ 'en',
42
+ 'nl',
43
+ 'fr',
44
+ 'de',
45
+ 'ko',
46
+ 'es',
47
+ 'cs',
48
+ 'ca',
49
+ 'fi',
50
+ 'is',
51
+ 'it',
52
+ 'pt',
53
+ 'ru',
54
+ 'zh',
55
+ 'sk',
56
+ 'rm',
57
+ 'ar',
58
+ 'da',
59
+ 'no',
60
+ 'pl',
61
+ 'sv',
62
+ 'tr',
63
+ 'hy',
64
+ 'az',
65
+ 'ka',
66
+ 'uk',
67
+ 'cy',
68
+ ]
69
+
28
70
  export function HttpLoaderFactory(http: HttpClient) {
29
71
  return new FileTranslateLoader(http, './assets/i18n/')
30
72
  }
@@ -15,6 +15,17 @@ export const LANG_3_TO_2_MAPPER = {
15
15
  chi: 'zh',
16
16
  slo: 'sk',
17
17
  roh: 'rm',
18
+ ara: 'ar',
19
+ dan: 'da',
20
+ nor: 'no',
21
+ pol: 'pl',
22
+ swe: 'sv',
23
+ tur: 'tr',
24
+ arm: 'hy',
25
+ aze: 'az',
26
+ geo: 'ka',
27
+ ukr: 'uk',
28
+ wel: 'cy',
18
29
  }
19
30
 
20
31
  export const LANGUAGE_NAMES = {
@@ -33,7 +44,18 @@ export const LANGUAGE_NAMES = {
33
44
  ru: 'Русский',
34
45
  zh: '中文',
35
46
  sk: 'Slovenčina',
36
- roh: 'Rumantsch',
47
+ rm: 'Rumantsch',
48
+ ar: 'العربية',
49
+ da: 'Dansk',
50
+ no: 'Norsk',
51
+ pl: 'Polski',
52
+ sv: 'Swedish',
53
+ tr: 'Türkçe',
54
+ hy: 'հայերեն',
55
+ az: 'Azərbaycan dili',
56
+ ka: 'ქართული',
57
+ uk: 'українська',
58
+ wel: 'Cymraeg',
37
59
  }
38
60
 
39
61
  export const LANG_2_TO_3_MAPPER = Object.entries(LANG_3_TO_2_MAPPER).reduce(
package/tailwind.base.css CHANGED
@@ -131,7 +131,7 @@
131
131
  .gn-ui-btn-light {
132
132
  @apply gn-ui-btn
133
133
  hover:bg-gray-50 focus:bg-gray-50 active:bg-gray-100
134
- border-white focus:ring-4 focus:ring-gray-300;
134
+ bg-white border-white focus:ring-4 focus:ring-gray-300;
135
135
  }
136
136
 
137
137
  .gn-ui-card-s {