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.
- package/esm2022/libs/api/metadata-converter/src/lib/iso19115-3/read-parts.mjs +8 -4
- package/esm2022/libs/api/metadata-converter/src/lib/iso19115-3/write-parts.mjs +5 -2
- package/esm2022/libs/api/metadata-converter/src/lib/iso19139/read-parts.mjs +3 -3
- package/esm2022/libs/api/metadata-converter/src/lib/iso19139/write-parts.mjs +2 -2
- package/esm2022/libs/api/repository/src/lib/gn4/gn4-repository.mjs +14 -5
- package/esm2022/libs/common/domain/src/lib/repository/records-repository.interface.mjs +1 -1
- package/esm2022/libs/feature/editor/src/lib/components/multilingual-panel/multilingual-panel.component.mjs +266 -11
- package/esm2022/libs/ui/search/src/index.mjs +2 -1
- package/esm2022/libs/ui/search/src/lib/results-table/action-menu/action-menu.component.mjs +19 -12
- package/esm2022/libs/ui/search/src/lib/results-table/results-table.component.mjs +3 -3
- package/esm2022/libs/ui/search/src/lib/ui-search.module.mjs +10 -4
- package/esm2022/libs/util/i18n/src/lib/i18n.constants.mjs +42 -1
- package/esm2022/libs/util/i18n/src/lib/language-codes.mjs +24 -2
- package/esm2022/translations/de.json +28 -4
- package/esm2022/translations/en.json +28 -4
- package/esm2022/translations/es.json +27 -3
- package/esm2022/translations/fr.json +28 -4
- package/esm2022/translations/it.json +27 -3
- package/esm2022/translations/nl.json +27 -3
- package/esm2022/translations/pt.json +27 -3
- package/fesm2022/geonetwork-ui.mjs +604 -107
- package/fesm2022/geonetwork-ui.mjs.map +1 -1
- package/libs/api/metadata-converter/src/lib/iso19115-3/read-parts.d.ts +1 -0
- package/libs/api/metadata-converter/src/lib/iso19115-3/read-parts.d.ts.map +1 -1
- package/libs/api/metadata-converter/src/lib/iso19115-3/write-parts.d.ts.map +1 -1
- package/libs/api/metadata-converter/src/lib/iso19139/read-parts.d.ts.map +1 -1
- package/libs/api/metadata-converter/src/lib/iso19139/write-parts.d.ts.map +1 -1
- package/libs/api/repository/src/lib/gn4/gn4-repository.d.ts +4 -2
- package/libs/api/repository/src/lib/gn4/gn4-repository.d.ts.map +1 -1
- package/libs/common/domain/src/lib/repository/records-repository.interface.d.ts +1 -0
- package/libs/common/domain/src/lib/repository/records-repository.interface.d.ts.map +1 -1
- package/libs/feature/editor/src/lib/components/multilingual-panel/multilingual-panel.component.d.ts +48 -3
- package/libs/feature/editor/src/lib/components/multilingual-panel/multilingual-panel.component.d.ts.map +1 -1
- package/libs/ui/search/src/index.d.ts +1 -0
- package/libs/ui/search/src/index.d.ts.map +1 -1
- package/libs/ui/search/src/lib/results-table/action-menu/action-menu.component.d.ts +3 -3
- package/libs/ui/search/src/lib/results-table/action-menu/action-menu.component.d.ts.map +1 -1
- package/libs/ui/search/src/lib/ui-search.module.d.ts +2 -1
- package/libs/ui/search/src/lib/ui-search.module.d.ts.map +1 -1
- package/libs/util/i18n/src/lib/i18n.constants.d.ts +1 -0
- package/libs/util/i18n/src/lib/i18n.constants.d.ts.map +1 -1
- package/libs/util/i18n/src/lib/language-codes.d.ts +23 -1
- package/libs/util/i18n/src/lib/language-codes.d.ts.map +1 -1
- package/package.json +1 -1
- package/src/libs/api/metadata-converter/src/lib/fixtures/geocat-ch.records.ts +1 -1
- package/src/libs/api/metadata-converter/src/lib/iso19115-3/read-parts.ts +13 -3
- package/src/libs/api/metadata-converter/src/lib/iso19115-3/write-parts.ts +5 -1
- package/src/libs/api/metadata-converter/src/lib/iso19139/read-parts.ts +6 -3
- package/src/libs/api/metadata-converter/src/lib/iso19139/write-parts.ts +4 -1
- package/src/libs/api/repository/src/lib/gn4/gn4-repository.ts +16 -2
- package/src/libs/common/domain/src/lib/repository/records-repository.interface.ts +1 -0
- package/src/libs/feature/editor/src/lib/components/multilingual-panel/multilingual-panel.component.html +117 -11
- package/src/libs/feature/editor/src/lib/components/multilingual-panel/multilingual-panel.component.ts +316 -6
- package/src/libs/ui/search/src/index.ts +1 -0
- package/src/libs/ui/search/src/lib/results-table/action-menu/action-menu.component.html +11 -3
- package/src/libs/ui/search/src/lib/results-table/action-menu/action-menu.component.ts +14 -11
- package/src/libs/ui/search/src/lib/results-table/results-table.component.html +1 -1
- package/src/libs/ui/search/src/lib/ui-search.module.ts +3 -0
- package/src/libs/util/i18n/src/lib/i18n.constants.ts +42 -0
- package/src/libs/util/i18n/src/lib/language-codes.ts +23 -1
- package/tailwind.base.css +1 -1
- package/translations/de.json +28 -4
- package/translations/en.json +28 -4
- package/translations/es.json +27 -3
- package/translations/fr.json +28 -4
- package/translations/it.json +27 -3
- package/translations/nl.json +27 -3
- package/translations/pt.json +27 -3
- 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-
|
|
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
|
-
<
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
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 {
|
|
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: [
|
|
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
|
-
|
|
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="
|
|
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="
|
|
28
|
-
<span *ngIf="
|
|
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
|
|
34
|
-
@Input() canDelete
|
|
35
|
-
@Input()
|
|
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
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
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
|
-
[
|
|
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
|
-
|
|
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