geonetwork-ui 2.10.0-dev.d3efc146c → 2.10.0-dev.de2fa8e42
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/fesm2022/geonetwork-ui.mjs +469 -127
- package/fesm2022/geonetwork-ui.mjs.map +1 -1
- package/index.d.ts +97 -29
- package/index.d.ts.map +1 -1
- package/package.json +2 -2
- package/src/libs/api/repository/src/lib/gn4/gn4-repository.ts +14 -2
- package/src/libs/api/repository/src/lib/gn4/gn4.provider.ts +6 -1
- package/src/libs/feature/editor/src/lib/+state/editor.actions.ts +6 -0
- package/src/libs/feature/editor/src/lib/+state/editor.effects.ts +0 -1
- package/src/libs/feature/editor/src/lib/+state/editor.facade.ts +10 -1
- package/src/libs/feature/editor/src/lib/components/contact-details/contact-details-form.component.css +0 -0
- package/src/libs/feature/editor/src/lib/components/contact-details/contact-details-form.component.html +67 -0
- package/src/libs/feature/editor/src/lib/components/contact-details/contact-details-form.component.ts +39 -0
- package/src/libs/feature/editor/src/lib/components/metadata-quality-panel/metadata-quality-panel.component.html +18 -3
- package/src/libs/feature/editor/src/lib/components/metadata-quality-panel/metadata-quality-panel.component.ts +33 -40
- package/src/libs/feature/editor/src/lib/components/record-form/form-field/field-focus.directive.ts +38 -0
- package/src/libs/feature/editor/src/lib/components/record-form/form-field/form-field-constraints-shortcuts/form-field-constraints-shortcuts.component.ts +33 -34
- package/src/libs/feature/editor/src/lib/components/record-form/form-field/form-field-contacts/form-field-contacts.component.html +13 -13
- package/src/libs/feature/editor/src/lib/components/record-form/form-field/form-field-contacts/form-field-contacts.component.ts +18 -4
- package/src/libs/feature/editor/src/lib/components/record-form/form-field/form-field-contacts-for-resource/form-field-contacts-for-resource.component.html +8 -7
- package/src/libs/feature/editor/src/lib/components/record-form/form-field/form-field-contacts-for-resource/form-field-contacts-for-resource.component.ts +6 -6
- package/src/libs/feature/editor/src/lib/components/record-form/form-field/form-field-online-single-link-resource/form-field-online-single-link-resource.component.css +0 -0
- package/src/libs/feature/editor/src/lib/components/record-form/form-field/form-field-online-single-link-resource/form-field-online-single-link-resource.component.html +5 -0
- package/src/libs/feature/editor/src/lib/components/record-form/form-field/form-field-online-single-link-resource/form-field-online-single-link-resource.component.ts +89 -0
- package/src/libs/feature/editor/src/lib/components/record-form/form-field/form-field.component.css +37 -0
- package/src/libs/feature/editor/src/lib/components/record-form/form-field/form-field.component.html +1 -0
- package/src/libs/feature/editor/src/lib/components/record-form/form-field/form-field.component.ts +5 -0
- package/src/libs/feature/editor/src/lib/components/record-form/form-field/index.ts +1 -0
- package/src/libs/feature/editor/src/lib/components/record-form/record-form.component.ts +59 -3
- package/src/libs/feature/editor/src/lib/models/editor-config.model.ts +4 -0
- package/src/libs/feature/notify-reuse/src/index.ts +1 -0
- package/src/libs/feature/notify-reuse/src/lib/notify-reuse-form/notify-reuse-form.component.css +0 -0
- package/src/libs/feature/notify-reuse/src/lib/notify-reuse-form/notify-reuse-form.component.html +1 -0
- package/src/libs/feature/notify-reuse/src/lib/notify-reuse-form/notify-reuse-form.component.ts +21 -0
- package/src/libs/ui/elements/src/index.ts +2 -0
- package/src/libs/ui/elements/src/lib/contact-details/contact-details.component.html +96 -0
- package/src/libs/ui/elements/src/lib/contact-details/contact-details.component.ts +45 -0
- package/src/libs/ui/elements/src/lib/contact-pill/contact-pill.component.html +37 -0
- package/src/libs/ui/elements/src/lib/contact-pill/contact-pill.component.ts +70 -0
- package/src/libs/ui/elements/src/lib/internal-link-card/internal-link-card.component.html +1 -1
- package/src/libs/ui/elements/src/lib/internal-link-card/internal-link-card.component.ts +0 -1
- package/src/libs/ui/elements/src/lib/metadata-contact/metadata-contact.component.ts +2 -5
- package/src/libs/ui/elements/src/lib/metadata-info/metadata-info.component.css +0 -4
- package/src/libs/ui/elements/src/lib/metadata-info/metadata-info.component.html +28 -67
- package/src/libs/ui/elements/src/lib/metadata-info/metadata-info.component.ts +30 -10
- package/src/libs/ui/inputs/src/lib/button/button.component.ts +4 -0
- package/src/libs/ui/inputs/src/lib/url-input/url-input.component.html +2 -2
- package/src/libs/ui/inputs/src/lib/url-input/url-input.component.ts +2 -1
- package/src/libs/ui/map/src/lib/components/map-container/map-container.component.ts +2 -1
- package/src/libs/ui/map/src/lib/components/spatial-extent/spatial-extent.component.ts +11 -10
- package/src/libs/ui/search/src/lib/record-preview-row/record-preview-row.component.html +0 -1
- package/src/libs/util/app-config/src/lib/app-config.ts +36 -0
- package/src/libs/util/app-config/src/lib/model.ts +4 -0
- package/src/libs/util/app-config/src/lib/parse-utils.ts +23 -1
- package/src/libs/util/shared/src/lib/record/quality-score.util.ts +33 -18
- package/src/libs/util/shared/src/lib/utils/index.ts +1 -0
- package/src/libs/util/shared/src/lib/utils/user-display.ts +32 -0
- package/tailwind.base.css +11 -2
- package/translations/de.json +10 -1
- package/translations/en.json +10 -1
- package/translations/es.json +10 -1
- package/translations/fr.json +10 -1
- package/translations/it.json +10 -1
- package/translations/nl.json +10 -1
- package/translations/pt.json +10 -1
- package/translations/sk.json +10 -1
|
@@ -0,0 +1,67 @@
|
|
|
1
|
+
<div class="flex flex-col gap-7">
|
|
2
|
+
<div class="grid grid-cols-2 gap-4">
|
|
3
|
+
<div class="min-w-0">
|
|
4
|
+
<h3 class="text-[16px] font-bold text-main mb-[12px]" translate>
|
|
5
|
+
editor.record.form.field.contactDetails.lastName
|
|
6
|
+
</h3>
|
|
7
|
+
<gn-ui-text-input
|
|
8
|
+
class="block w-full"
|
|
9
|
+
extraClass="w-full"
|
|
10
|
+
[(value)]="contact.lastName"
|
|
11
|
+
(valueChange)="emitContactChange()"
|
|
12
|
+
[placeholder]="
|
|
13
|
+
'editor.record.form.field.contactDetails.lastName.placeholder'
|
|
14
|
+
| translate
|
|
15
|
+
"
|
|
16
|
+
data-test="contactDetailsLastName"
|
|
17
|
+
></gn-ui-text-input>
|
|
18
|
+
</div>
|
|
19
|
+
<div class="min-w-0">
|
|
20
|
+
<h3 class="text-[16px] font-bold text-main mb-[12px]" translate>
|
|
21
|
+
editor.record.form.field.contactDetails.firstName
|
|
22
|
+
</h3>
|
|
23
|
+
<gn-ui-text-input
|
|
24
|
+
class="block w-full"
|
|
25
|
+
extraClass="w-full"
|
|
26
|
+
[(value)]="contact.firstName"
|
|
27
|
+
(valueChange)="emitContactChange()"
|
|
28
|
+
[placeholder]="
|
|
29
|
+
'editor.record.form.field.contactDetails.firstName.placeholder'
|
|
30
|
+
| translate
|
|
31
|
+
"
|
|
32
|
+
data-test="contactDetailsFirstName"
|
|
33
|
+
></gn-ui-text-input>
|
|
34
|
+
</div>
|
|
35
|
+
</div>
|
|
36
|
+
<div>
|
|
37
|
+
<h3 class="text-[16px] font-bold text-main mb-[12px]" translate>
|
|
38
|
+
editor.record.form.field.contactDetails.email
|
|
39
|
+
</h3>
|
|
40
|
+
<gn-ui-text-input
|
|
41
|
+
class="block w-full"
|
|
42
|
+
extraClass="w-full"
|
|
43
|
+
[value]="contact.organization?.email ?? ''"
|
|
44
|
+
(valueChange)="handleOrganizationChange({ email: $event })"
|
|
45
|
+
[placeholder]="
|
|
46
|
+
'editor.record.form.field.contactDetails.email.placeholder' | translate
|
|
47
|
+
"
|
|
48
|
+
data-test="contactDetailsEmail"
|
|
49
|
+
></gn-ui-text-input>
|
|
50
|
+
</div>
|
|
51
|
+
<div>
|
|
52
|
+
<h3 class="text-[16px] font-bold text-main mb-[12px]" translate>
|
|
53
|
+
editor.record.form.field.contactDetails.organization
|
|
54
|
+
</h3>
|
|
55
|
+
<gn-ui-text-input
|
|
56
|
+
class="block w-full"
|
|
57
|
+
extraClass="w-full"
|
|
58
|
+
[value]="contact.organization?.name ?? ''"
|
|
59
|
+
(valueChange)="handleOrganizationChange({ name: $event })"
|
|
60
|
+
[placeholder]="
|
|
61
|
+
'editor.record.form.field.contactDetails.organization.placeholder'
|
|
62
|
+
| translate
|
|
63
|
+
"
|
|
64
|
+
data-test="contactDetailsOrganization"
|
|
65
|
+
></gn-ui-text-input>
|
|
66
|
+
</div>
|
|
67
|
+
</div>
|
package/src/libs/feature/editor/src/lib/components/contact-details/contact-details-form.component.ts
ADDED
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
import {
|
|
2
|
+
ChangeDetectionStrategy,
|
|
3
|
+
Component,
|
|
4
|
+
EventEmitter,
|
|
5
|
+
Input,
|
|
6
|
+
Output,
|
|
7
|
+
} from '@angular/core'
|
|
8
|
+
import {
|
|
9
|
+
Individual,
|
|
10
|
+
Organization,
|
|
11
|
+
} from '../../../../../../../libs/common/domain/src/lib/model/record'
|
|
12
|
+
import { TextInputComponent } from '../../../../../../../libs/ui/inputs/src'
|
|
13
|
+
import { TranslateDirective, TranslatePipe } from '@ngx-translate/core'
|
|
14
|
+
|
|
15
|
+
@Component({
|
|
16
|
+
selector: 'gn-ui-contact-details-form',
|
|
17
|
+
templateUrl: './contact-details-form.component.html',
|
|
18
|
+
styleUrls: ['./contact-details-form.component.css'],
|
|
19
|
+
changeDetection: ChangeDetectionStrategy.OnPush,
|
|
20
|
+
standalone: true,
|
|
21
|
+
imports: [TextInputComponent, TranslateDirective, TranslatePipe],
|
|
22
|
+
})
|
|
23
|
+
export class ContactDetailsFormComponent {
|
|
24
|
+
@Input() contact: Individual
|
|
25
|
+
@Output() contactChange = new EventEmitter<Individual>()
|
|
26
|
+
|
|
27
|
+
emitContactChange() {
|
|
28
|
+
this.contactChange.emit(this.contact)
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
handleOrganizationChange(change: Partial<Organization>) {
|
|
32
|
+
this.contact.organization = {
|
|
33
|
+
...(this.contact.organization ?? ({} as Organization)),
|
|
34
|
+
...change,
|
|
35
|
+
} as Organization
|
|
36
|
+
|
|
37
|
+
this.emitContactChange()
|
|
38
|
+
}
|
|
39
|
+
}
|
|
@@ -7,11 +7,26 @@
|
|
|
7
7
|
>editor.record.form.metadataQuality.title</span
|
|
8
8
|
>
|
|
9
9
|
</div>
|
|
10
|
-
@for (
|
|
10
|
+
@for (
|
|
11
|
+
properties of (propertiesByPage$ | async) ?? [];
|
|
12
|
+
track properties;
|
|
13
|
+
let isLast = $last
|
|
14
|
+
) {
|
|
11
15
|
<div class="flex flex-col gap-2">
|
|
12
16
|
@for (property of properties; track property) {
|
|
13
17
|
<gn-ui-button
|
|
14
|
-
|
|
18
|
+
style="
|
|
19
|
+
--gn-ui-button-justify: space-between;
|
|
20
|
+
--gn-ui-button-height: 34px;
|
|
21
|
+
--gn-ui-button-width: 100%;
|
|
22
|
+
--gn-ui-button-border-width: 0;
|
|
23
|
+
--gn-ui-button-color: black;
|
|
24
|
+
--gn-ui-button-background: transparent;
|
|
25
|
+
--gn-ui-button-bg-hover: #f3f4f6;
|
|
26
|
+
--gn-ui-button-disabled-opacity: 1;
|
|
27
|
+
"
|
|
28
|
+
[disabled]="property.value"
|
|
29
|
+
(buttonClick)="onCriterionClick(property)"
|
|
15
30
|
type="outline"
|
|
16
31
|
attr.data-cy="md-quality-btn-{{ property.label }}"
|
|
17
32
|
>
|
|
@@ -28,7 +43,7 @@
|
|
|
28
43
|
</div>
|
|
29
44
|
</gn-ui-button>
|
|
30
45
|
}
|
|
31
|
-
@if (
|
|
46
|
+
@if (!isLast) {
|
|
32
47
|
<hr class="border-gray-300 w-11/12 mx-auto" />
|
|
33
48
|
}
|
|
34
49
|
</div>
|
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
import { Component,
|
|
2
|
-
import {
|
|
1
|
+
import { Component, inject } from '@angular/core'
|
|
2
|
+
import { CatalogRecordKeys } from '../../../../../../../libs/common/domain/src/lib/model/record'
|
|
3
3
|
import { ButtonComponent } from '../../../../../../../libs/ui/inputs/src'
|
|
4
4
|
import {
|
|
5
5
|
getAllKeysValidator,
|
|
@@ -13,8 +13,10 @@ import {
|
|
|
13
13
|
} from '@ng-icons/core'
|
|
14
14
|
import { iconoirBadgeCheck, iconoirSystemShut } from '@ng-icons/iconoir'
|
|
15
15
|
import { TranslateDirective, TranslatePipe } from '@ngx-translate/core'
|
|
16
|
-
import { EditorConfig } from '../../models'
|
|
17
16
|
import { marker } from '@biesbjerg/ngx-translate-extract-marker'
|
|
17
|
+
import { EditorFacade } from '../../+state/editor.facade'
|
|
18
|
+
import { combineLatest, map } from 'rxjs'
|
|
19
|
+
import { AsyncPipe } from '@angular/common'
|
|
18
20
|
|
|
19
21
|
//forced translations that are not available in fields.config.ts
|
|
20
22
|
marker('editor.record.form.field.keywords')
|
|
@@ -29,6 +31,7 @@ marker('editor.record.form.field.organisation')
|
|
|
29
31
|
TranslatePipe,
|
|
30
32
|
ButtonComponent,
|
|
31
33
|
NgIconComponent,
|
|
34
|
+
AsyncPipe,
|
|
32
35
|
],
|
|
33
36
|
providers: [
|
|
34
37
|
provideIcons({
|
|
@@ -42,47 +45,37 @@ marker('editor.record.form.field.organisation')
|
|
|
42
45
|
templateUrl: './metadata-quality-panel.component.html',
|
|
43
46
|
styleUrl: './metadata-quality-panel.component.css',
|
|
44
47
|
})
|
|
45
|
-
export class MetadataQualityPanelComponent
|
|
48
|
+
export class MetadataQualityPanelComponent {
|
|
49
|
+
facade = inject(EditorFacade)
|
|
46
50
|
propsToValidate: ValidatorMapperKeys[] = getAllKeysValidator()
|
|
47
|
-
propertiesByPage: { label: string; value: boolean }[][] = []
|
|
48
|
-
@Input() editorConfig: EditorConfig
|
|
49
|
-
@Input() record: CatalogRecord
|
|
50
51
|
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
this.record,
|
|
71
|
-
fields as ValidatorMapperKeys[]
|
|
72
|
-
).map(({ name, validator }) => ({
|
|
73
|
-
label: `editor.record.form.field.${name}`, // use same translations as in fields.config.ts
|
|
74
|
-
value: validator(),
|
|
75
|
-
}))
|
|
52
|
+
propertiesByPage$ = combineLatest([
|
|
53
|
+
this.facade.editorConfig$,
|
|
54
|
+
this.facade.record$,
|
|
55
|
+
]).pipe(
|
|
56
|
+
map(([editorConfig, record]) => {
|
|
57
|
+
if (!editorConfig || !record) return []
|
|
58
|
+
const validators = getQualityValidators(record, this.propsToValidate)
|
|
59
|
+
return editorConfig.pages
|
|
60
|
+
.map((page) =>
|
|
61
|
+
page.sections
|
|
62
|
+
.flatMap((section) => section.fields)
|
|
63
|
+
.flatMap(({ model }) =>
|
|
64
|
+
validators.filter((v) => (v.alias ?? v.name) === model)
|
|
65
|
+
)
|
|
66
|
+
.map(({ name, validator, alias }) => ({
|
|
67
|
+
label: `editor.record.form.field.${name}`,
|
|
68
|
+
value: validator(),
|
|
69
|
+
model: (alias ?? name) as CatalogRecordKeys,
|
|
70
|
+
}))
|
|
76
71
|
)
|
|
77
72
|
.filter((arr) => arr.length > 0)
|
|
78
|
-
}
|
|
79
|
-
|
|
73
|
+
})
|
|
74
|
+
)
|
|
80
75
|
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
? `${baseClasses} bg-neutral-100 hover:bg-neutral-100`
|
|
86
|
-
: `${baseClasses} bg-transparent hover:bg-transparent`
|
|
76
|
+
onCriterionClick(property: { value: boolean; model: CatalogRecordKeys }) {
|
|
77
|
+
if (!property.value) {
|
|
78
|
+
this.facade.setFocusedField(property.model)
|
|
79
|
+
}
|
|
87
80
|
}
|
|
88
81
|
}
|
package/src/libs/feature/editor/src/lib/components/record-form/form-field/field-focus.directive.ts
ADDED
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
import { Directive, ElementRef, inject, Input } from '@angular/core'
|
|
2
|
+
|
|
3
|
+
@Directive({
|
|
4
|
+
selector: '[gnUiFieldFocus]',
|
|
5
|
+
standalone: true,
|
|
6
|
+
exportAs: 'fieldFocus',
|
|
7
|
+
})
|
|
8
|
+
export class FieldFocusDirective {
|
|
9
|
+
@Input() gnUiFieldFocusGlowClass = 'gn-ui-field-focus-glow'
|
|
10
|
+
|
|
11
|
+
private el = inject(ElementRef)
|
|
12
|
+
|
|
13
|
+
public focusField() {
|
|
14
|
+
setTimeout(() => {
|
|
15
|
+
const host = this.el.nativeElement as HTMLElement
|
|
16
|
+
const glowClass = this.gnUiFieldFocusGlowClass
|
|
17
|
+
|
|
18
|
+
host.classList.remove(glowClass)
|
|
19
|
+
void host.offsetWidth
|
|
20
|
+
host.classList.add(glowClass)
|
|
21
|
+
host.addEventListener(
|
|
22
|
+
'animationend',
|
|
23
|
+
() => host.classList.remove(glowClass),
|
|
24
|
+
{ once: true }
|
|
25
|
+
)
|
|
26
|
+
|
|
27
|
+
host.scrollIntoView({ behavior: 'smooth', block: 'start' })
|
|
28
|
+
const target =
|
|
29
|
+
host.querySelector<HTMLElement>(
|
|
30
|
+
'input, textarea, select, [contenteditable="true"]'
|
|
31
|
+
) ??
|
|
32
|
+
host.querySelector<HTMLElement>(
|
|
33
|
+
'button:not([disabled]), [tabindex]:not([tabindex="-1"])'
|
|
34
|
+
)
|
|
35
|
+
target?.focus({ preventScroll: true })
|
|
36
|
+
})
|
|
37
|
+
}
|
|
38
|
+
}
|
|
@@ -2,7 +2,7 @@ import {
|
|
|
2
2
|
ChangeDetectionStrategy,
|
|
3
3
|
Component,
|
|
4
4
|
OnDestroy,
|
|
5
|
-
|
|
5
|
+
afterNextRender,
|
|
6
6
|
inject,
|
|
7
7
|
} from '@angular/core'
|
|
8
8
|
import { CommonModule } from '@angular/common'
|
|
@@ -67,9 +67,7 @@ export type ConstraintChoice =
|
|
|
67
67
|
styleUrls: ['./form-field-constraints-shortcuts.component.css'],
|
|
68
68
|
changeDetection: ChangeDetectionStrategy.OnPush,
|
|
69
69
|
})
|
|
70
|
-
export class FormFieldConstraintsShortcutsComponent
|
|
71
|
-
implements OnInit, OnDestroy
|
|
72
|
-
{
|
|
70
|
+
export class FormFieldConstraintsShortcutsComponent implements OnDestroy {
|
|
73
71
|
private editorFacade = inject(EditorFacade)
|
|
74
72
|
|
|
75
73
|
legalConstraints$ = this.editorFacade.record$.pipe(
|
|
@@ -112,37 +110,38 @@ export class FormFieldConstraintsShortcutsComponent
|
|
|
112
110
|
|
|
113
111
|
onDestroy$ = new Subject<void>()
|
|
114
112
|
|
|
115
|
-
|
|
116
|
-
//
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
.
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
113
|
+
constructor() {
|
|
114
|
+
// Deferred to afterNextRender to avoid dispatching store actions
|
|
115
|
+
// synchronously during Angular's change detection cycle (NG0100)
|
|
116
|
+
afterNextRender(() => {
|
|
117
|
+
this.anyToggleActivated$
|
|
118
|
+
.pipe(takeUntil(this.onDestroy$), distinctUntilChanged())
|
|
119
|
+
.subscribe((anyToggleActivated) => {
|
|
120
|
+
if (anyToggleActivated) {
|
|
121
|
+
this.hideAllConstraintSections()
|
|
122
|
+
}
|
|
123
|
+
})
|
|
124
124
|
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
}
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
hideEmptyConstraints(this.otherConstraints$, 'otherConstraints')
|
|
125
|
+
const hideEmptyConstraints = (
|
|
126
|
+
constraints$: Observable<Constraint[]>,
|
|
127
|
+
model: ConstraintChoice
|
|
128
|
+
) => {
|
|
129
|
+
const isConstraintNotEmpty$ = constraints$.pipe(
|
|
130
|
+
takeUntil(this.onDestroy$),
|
|
131
|
+
map((c) => c.length > 0),
|
|
132
|
+
distinctUntilChanged()
|
|
133
|
+
)
|
|
134
|
+
combineLatest([
|
|
135
|
+
isConstraintNotEmpty$,
|
|
136
|
+
this.anyToggleActivated$,
|
|
137
|
+
]).subscribe(([isNotEmpty, anyToggleActivated]) => {
|
|
138
|
+
const visible = isNotEmpty && !anyToggleActivated
|
|
139
|
+
this.editorFacade.setFieldVisibility({ model }, visible)
|
|
140
|
+
})
|
|
141
|
+
}
|
|
142
|
+
hideEmptyConstraints(this.securityConstraints$, 'securityConstraints')
|
|
143
|
+
hideEmptyConstraints(this.otherConstraints$, 'otherConstraints')
|
|
144
|
+
})
|
|
146
145
|
}
|
|
147
146
|
|
|
148
147
|
ngOnDestroy() {
|
|
@@ -10,21 +10,21 @@
|
|
|
10
10
|
</gn-ui-autocomplete>
|
|
11
11
|
|
|
12
12
|
@if (contacts.length > 0) {
|
|
13
|
-
|
|
14
|
-
|
|
13
|
+
<gn-ui-sortable-list
|
|
14
|
+
[items]="contacts"
|
|
15
|
+
(itemsOrderChange)="handleContactsChanged($event)"
|
|
16
|
+
[elementTemplate]="contactTemplate"
|
|
17
|
+
></gn-ui-sortable-list>
|
|
18
|
+
<ng-template #contactTemplate let-contact let-index="index">
|
|
19
|
+
@if (modelSpecifier === 'contact:editableDetails') {
|
|
20
|
+
<gn-ui-contact-details-form
|
|
21
|
+
[contact]="contact"
|
|
22
|
+
(contactChange)="handleContactChanged($event, index)"
|
|
23
|
+
></gn-ui-contact-details-form>
|
|
24
|
+
} @else {
|
|
15
25
|
<gn-ui-contact-card [contact]="contact"></gn-ui-contact-card>
|
|
16
26
|
}
|
|
17
|
-
|
|
18
|
-
@if (contacts.length > 1) {
|
|
19
|
-
<gn-ui-sortable-list
|
|
20
|
-
[items]="contacts"
|
|
21
|
-
(itemsOrderChange)="handleContactsChanged($event)"
|
|
22
|
-
[elementTemplate]="contactTemplate"
|
|
23
|
-
></gn-ui-sortable-list>
|
|
24
|
-
<ng-template #contactTemplate let-contact>
|
|
25
|
-
<gn-ui-contact-card [contact]="contact"></gn-ui-contact-card>
|
|
26
|
-
</ng-template>
|
|
27
|
-
}
|
|
27
|
+
</ng-template>
|
|
28
28
|
} @else {
|
|
29
29
|
<div
|
|
30
30
|
class="p-4 text-sm border border-primary bg-primary-lightest rounded-lg"
|
|
@@ -28,9 +28,15 @@ import { UserModel } from '../../../../../../../../../libs/common/domain/src/lib
|
|
|
28
28
|
import { PlatformServiceInterface } from '../../../../../../../../../libs/common/domain/src/lib/platform.service.interface'
|
|
29
29
|
import { OrganizationsServiceInterface } from '../../../../../../../../../libs/common/domain/src/lib/organizations.service.interface'
|
|
30
30
|
import { ContactCardComponent } from '../../../contact-card/contact-card.component'
|
|
31
|
-
import {
|
|
31
|
+
import { ContactDetailsFormComponent } from '../../../contact-details/contact-details-form.component'
|
|
32
|
+
import {
|
|
33
|
+
createFuzzyFilter,
|
|
34
|
+
getIndividualDisplayName,
|
|
35
|
+
toIndividual,
|
|
36
|
+
} from '../../../../../../../../../libs/util/shared/src'
|
|
32
37
|
import { map } from 'rxjs/operators'
|
|
33
38
|
import { SortableListComponent } from '../../../../../../../../../libs/ui/layout/src'
|
|
39
|
+
import { FieldModelSpecifier } from '../../../../models/editor-config.model'
|
|
34
40
|
|
|
35
41
|
@Component({
|
|
36
42
|
selector: 'gn-ui-form-field-contacts',
|
|
@@ -43,6 +49,7 @@ import { SortableListComponent } from '../../../../../../../../../libs/ui/layout
|
|
|
43
49
|
TranslateDirective,
|
|
44
50
|
TranslatePipe,
|
|
45
51
|
ContactCardComponent,
|
|
52
|
+
ContactDetailsFormComponent,
|
|
46
53
|
SortableListComponent,
|
|
47
54
|
],
|
|
48
55
|
})
|
|
@@ -52,6 +59,7 @@ export class FormFieldContactsComponent implements OnDestroy, OnChanges {
|
|
|
52
59
|
private changeDetectorRef = inject(ChangeDetectorRef)
|
|
53
60
|
|
|
54
61
|
@Input() value: Individual[]
|
|
62
|
+
@Input() modelSpecifier: FieldModelSpecifier
|
|
55
63
|
@Output() valueChange: EventEmitter<Individual[]> = new EventEmitter()
|
|
56
64
|
|
|
57
65
|
contacts: Individual[] = []
|
|
@@ -113,13 +121,19 @@ export class FormFieldContactsComponent implements OnDestroy, OnChanges {
|
|
|
113
121
|
this.valueChange.emit(contacts)
|
|
114
122
|
}
|
|
115
123
|
|
|
124
|
+
handleContactChanged(updatedContact: Individual, index: number) {
|
|
125
|
+
const contacts = this.contacts.map((contact, i) =>
|
|
126
|
+
i === index ? updatedContact : contact
|
|
127
|
+
)
|
|
128
|
+
|
|
129
|
+
this.handleContactsChanged(contacts)
|
|
130
|
+
}
|
|
131
|
+
|
|
116
132
|
/**
|
|
117
133
|
* gn-ui-autocomplete
|
|
118
134
|
*/
|
|
119
135
|
displayWithFn: (user: UserModel) => string = (user) =>
|
|
120
|
-
|
|
121
|
-
user.organisation ? `(${user.organisation})` : ''
|
|
122
|
-
}`
|
|
136
|
+
getIndividualDisplayName(toIndividual(user))
|
|
123
137
|
|
|
124
138
|
/**
|
|
125
139
|
* gn-ui-autocomplete
|
|
@@ -8,6 +8,14 @@
|
|
|
8
8
|
</gn-ui-button>
|
|
9
9
|
}
|
|
10
10
|
</div>
|
|
11
|
+
@if (value.length === 0) {
|
|
12
|
+
<div
|
|
13
|
+
class="p-4 border border-primary bg-primary-lightest rounded-lg"
|
|
14
|
+
translate
|
|
15
|
+
>
|
|
16
|
+
editor.record.form.field.contactsForResource.noContact
|
|
17
|
+
</div>
|
|
18
|
+
}
|
|
11
19
|
@if (roleSectionsToDisplay && roleSectionsToDisplay.length > 0) {
|
|
12
20
|
<div class="mt-8" data-test="displayedRoles">
|
|
13
21
|
@for (
|
|
@@ -53,12 +61,5 @@
|
|
|
53
61
|
</div>
|
|
54
62
|
}
|
|
55
63
|
</div>
|
|
56
|
-
} @else {
|
|
57
|
-
<div
|
|
58
|
-
class="p-4 border border-primary bg-primary-lightest rounded-lg"
|
|
59
|
-
translate
|
|
60
|
-
>
|
|
61
|
-
editor.record.form.field.contactsForResource.noContact
|
|
62
|
-
</div>
|
|
63
64
|
}
|
|
64
65
|
</div>
|
|
@@ -23,7 +23,11 @@ import {
|
|
|
23
23
|
AutocompleteComponent,
|
|
24
24
|
ButtonComponent,
|
|
25
25
|
} from '../../../../../../../../../libs/ui/inputs/src'
|
|
26
|
-
import {
|
|
26
|
+
import {
|
|
27
|
+
createFuzzyFilter,
|
|
28
|
+
getIndividualDisplayName,
|
|
29
|
+
toIndividual,
|
|
30
|
+
} from '../../../../../../../../../libs/util/shared/src'
|
|
27
31
|
import { TranslateDirective, TranslatePipe } from '@ngx-translate/core'
|
|
28
32
|
import {
|
|
29
33
|
debounceTime,
|
|
@@ -169,11 +173,7 @@ export class FormFieldContactsForResourceComponent
|
|
|
169
173
|
* gn-ui-autocomplete
|
|
170
174
|
*/
|
|
171
175
|
displayWithFn: (user: UserModel) => string = (user) =>
|
|
172
|
-
user
|
|
173
|
-
? `${user.name} ${user.surname} ${
|
|
174
|
-
user.organisation ? `(${user.organisation})` : ''
|
|
175
|
-
}`
|
|
176
|
-
: ``
|
|
176
|
+
getIndividualDisplayName(toIndividual(user))
|
|
177
177
|
|
|
178
178
|
/**
|
|
179
179
|
* gn-ui-autocomplete
|
|
File without changes
|
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
import {
|
|
2
|
+
ChangeDetectionStrategy,
|
|
3
|
+
Component,
|
|
4
|
+
EventEmitter,
|
|
5
|
+
Input,
|
|
6
|
+
Output,
|
|
7
|
+
inject,
|
|
8
|
+
} from '@angular/core'
|
|
9
|
+
import { marker } from '@biesbjerg/ngx-translate-extract-marker'
|
|
10
|
+
import {
|
|
11
|
+
OnlineLinkResource,
|
|
12
|
+
OnlineResource,
|
|
13
|
+
} from '../../../../../../../../../libs/common/domain/src/lib/model/record'
|
|
14
|
+
import { NotificationsService } from '../../../../../../../../../libs/feature/notifications/src'
|
|
15
|
+
import { UrlInputComponent } from '../../../../../../../../../libs/ui/inputs/src'
|
|
16
|
+
import { TranslateService } from '@ngx-translate/core'
|
|
17
|
+
|
|
18
|
+
marker('editor.record.form.field.onlineLinkageResource.defaultName')
|
|
19
|
+
|
|
20
|
+
@Component({
|
|
21
|
+
selector: 'gn-ui-form-field-online-single-link-resource',
|
|
22
|
+
templateUrl: './form-field-online-single-link-resource.component.html',
|
|
23
|
+
styleUrls: ['./form-field-online-single-link-resource.component.css'],
|
|
24
|
+
changeDetection: ChangeDetectionStrategy.OnPush,
|
|
25
|
+
standalone: true,
|
|
26
|
+
imports: [UrlInputComponent],
|
|
27
|
+
})
|
|
28
|
+
export class FormFieldOnlineSingleLinkResourceComponent {
|
|
29
|
+
private notificationsService = inject(NotificationsService)
|
|
30
|
+
private translateService = inject(TranslateService)
|
|
31
|
+
|
|
32
|
+
@Input() set value(onlineResources: Array<OnlineResource>) {
|
|
33
|
+
this.allResources = onlineResources ?? []
|
|
34
|
+
const firstResource = this.allResources[0]
|
|
35
|
+
this.displayUrl = firstResource?.url?.toString() ?? ''
|
|
36
|
+
}
|
|
37
|
+
@Output() valueChange: EventEmitter<Array<OnlineResource>> =
|
|
38
|
+
new EventEmitter()
|
|
39
|
+
|
|
40
|
+
private allResources: OnlineResource[] = []
|
|
41
|
+
displayUrl = ''
|
|
42
|
+
|
|
43
|
+
handleUrlChange(url: string | null) {
|
|
44
|
+
if (!url) return
|
|
45
|
+
|
|
46
|
+
try {
|
|
47
|
+
const parsedUrl = new URL(url)
|
|
48
|
+
|
|
49
|
+
if (this.allResources.length === 0) {
|
|
50
|
+
const defaultName = this.translateService.instant(
|
|
51
|
+
'editor.record.form.field.onlineLinkageResource.defaultName'
|
|
52
|
+
)
|
|
53
|
+
const newResource: OnlineLinkResource = {
|
|
54
|
+
type: 'link',
|
|
55
|
+
url: parsedUrl,
|
|
56
|
+
name: defaultName,
|
|
57
|
+
}
|
|
58
|
+
this.valueChange.emit([newResource])
|
|
59
|
+
} else {
|
|
60
|
+
const updatedFirst: OnlineResource = {
|
|
61
|
+
...this.allResources[0],
|
|
62
|
+
url: parsedUrl,
|
|
63
|
+
}
|
|
64
|
+
this.valueChange.emit([updatedFirst, ...this.allResources.slice(1)])
|
|
65
|
+
}
|
|
66
|
+
} catch (error) {
|
|
67
|
+
this.handleError(error as Error)
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
private handleError(error: Error) {
|
|
72
|
+
this.notificationsService.showNotification(
|
|
73
|
+
{
|
|
74
|
+
type: 'error',
|
|
75
|
+
title: this.translateService.instant(
|
|
76
|
+
'editor.record.onlineResourceError.title'
|
|
77
|
+
),
|
|
78
|
+
text: `${this.translateService.instant(
|
|
79
|
+
'editor.record.onlineResourceError.body'
|
|
80
|
+
)} ${error.message}`,
|
|
81
|
+
closeMessage: this.translateService.instant(
|
|
82
|
+
'editor.record.onlineResourceError.closeMessage'
|
|
83
|
+
),
|
|
84
|
+
},
|
|
85
|
+
undefined,
|
|
86
|
+
error
|
|
87
|
+
)
|
|
88
|
+
}
|
|
89
|
+
}
|
package/src/libs/feature/editor/src/lib/components/record-form/form-field/form-field.component.css
CHANGED
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
:host {
|
|
2
|
+
scroll-margin-top: 90px;
|
|
3
|
+
}
|
|
4
|
+
|
|
5
|
+
:host.gn-ui-field-focus-glow {
|
|
6
|
+
position: relative;
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
:host.gn-ui-field-focus-glow::after {
|
|
10
|
+
content: '';
|
|
11
|
+
position: absolute;
|
|
12
|
+
inset: -8px;
|
|
13
|
+
border-radius: 8px;
|
|
14
|
+
background-color: color-mix(in srgb, var(--color-primary) 12%, transparent);
|
|
15
|
+
pointer-events: none;
|
|
16
|
+
animation: gn-ui-field-focus-glow 3s ease-out forwards;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
:host.gn-ui-field-focus-glow::before {
|
|
20
|
+
content: '';
|
|
21
|
+
position: absolute;
|
|
22
|
+
inset: -16px;
|
|
23
|
+
border: 2px dashed var(--color-primary);
|
|
24
|
+
border-radius: 10px;
|
|
25
|
+
pointer-events: none;
|
|
26
|
+
animation: gn-ui-field-focus-glow 3s ease-out forwards;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
@keyframes gn-ui-field-focus-glow {
|
|
30
|
+
0%,
|
|
31
|
+
55% {
|
|
32
|
+
opacity: 1;
|
|
33
|
+
}
|
|
34
|
+
100% {
|
|
35
|
+
opacity: 0;
|
|
36
|
+
}
|
|
37
|
+
}
|