@sebgroup/green-angular 5.4.0 → 5.4.1
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/src/v-angular/alert/alert.component.mjs +3 -3
- package/esm2022/src/v-angular/dropdown/dropdown-list/dropdown-list.component.mjs +49 -32
- package/esm2022/src/v-angular/dropdown/dropdown.component.mjs +3 -3
- package/esm2022/src/v-angular/dropdown/typeahead/typeahead-input/typeahead-input.component.mjs +3 -4
- package/esm2022/src/v-angular/input/input.component.mjs +2 -2
- package/esm2022/src/v-angular/modal/dialog/dialog.component.mjs +31 -9
- package/esm2022/src/v-angular/modal/slide-out/slide-out.component.mjs +3 -3
- package/esm2022/src/v-angular/table/table.component.mjs +2 -2
- package/esm2022/src/v-angular/textarea/textarea.component.mjs +2 -2
- package/esm2022/v-angular/alert/alert.component.mjs +3 -3
- package/esm2022/v-angular/core/core.globals.mjs +20 -0
- package/esm2022/v-angular/core/core.utils.mjs +62 -0
- package/esm2022/v-angular/core/index.mjs +3 -0
- package/esm2022/v-angular/dropdown/dropdown-list/dropdown-list.component.mjs +49 -32
- package/esm2022/v-angular/dropdown/dropdown.component.mjs +3 -3
- package/esm2022/v-angular/dropdown/typeahead/typeahead-input/typeahead-input.component.mjs +3 -4
- package/esm2022/v-angular/index.mjs +2 -1
- package/esm2022/v-angular/input/input.component.mjs +2 -2
- package/esm2022/v-angular/modal/dialog/dialog.component.mjs +31 -9
- package/esm2022/v-angular/modal/slide-out/slide-out.component.mjs +3 -3
- package/esm2022/v-angular/table/table.component.mjs +2 -2
- package/esm2022/v-angular/textarea/textarea.component.mjs +2 -2
- package/fesm2022/sebgroup-green-angular-src-v-angular-alert.mjs +2 -2
- package/fesm2022/sebgroup-green-angular-src-v-angular-alert.mjs.map +1 -1
- package/fesm2022/sebgroup-green-angular-src-v-angular-dropdown.mjs +52 -36
- package/fesm2022/sebgroup-green-angular-src-v-angular-dropdown.mjs.map +1 -1
- package/fesm2022/sebgroup-green-angular-src-v-angular-input.mjs +2 -2
- package/fesm2022/sebgroup-green-angular-src-v-angular-input.mjs.map +1 -1
- package/fesm2022/sebgroup-green-angular-src-v-angular-modal.mjs +32 -11
- package/fesm2022/sebgroup-green-angular-src-v-angular-modal.mjs.map +1 -1
- package/fesm2022/sebgroup-green-angular-src-v-angular-table.mjs +2 -2
- package/fesm2022/sebgroup-green-angular-src-v-angular-table.mjs.map +1 -1
- package/fesm2022/sebgroup-green-angular-src-v-angular-textarea.mjs +2 -2
- package/fesm2022/sebgroup-green-angular-src-v-angular-textarea.mjs.map +1 -1
- package/fesm2022/sebgroup-green-angular-v-angular.mjs +169 -53
- package/fesm2022/sebgroup-green-angular-v-angular.mjs.map +1 -1
- package/package.json +2 -2
- package/src/v-angular/dropdown/dropdown-list/dropdown-list.component.d.ts +8 -0
- package/src/v-angular/modal/dialog/dialog.component.d.ts +10 -3
- package/v-angular/core/core.globals.d.ts +13 -0
- package/v-angular/core/core.utils.d.ts +23 -0
- package/v-angular/core/index.d.ts +2 -0
- package/v-angular/dropdown/dropdown-list/dropdown-list.component.d.ts +8 -0
- package/v-angular/index.d.ts +1 -0
- package/v-angular/modal/dialog/dialog.component.d.ts +10 -3
|
@@ -1,6 +1,6 @@
|
|
|
1
|
-
import { Component, EventEmitter, HostBinding,
|
|
1
|
+
import { Component, EventEmitter, HostBinding, Inject, Input, Optional, Output, ViewChildren, } from '@angular/core';
|
|
2
2
|
import { TRANSLOCO_SCOPE } from '@jsverse/transloco';
|
|
3
|
-
import { Subject } from 'rxjs';
|
|
3
|
+
import { filter, fromEvent, Subject, takeUntil } from 'rxjs';
|
|
4
4
|
import scrollIntoView from 'scroll-into-view-if-needed';
|
|
5
5
|
import { DropdownUtils, } from '@sebgroup/green-angular/src/v-angular/core';
|
|
6
6
|
import * as i0 from "@angular/core";
|
|
@@ -36,6 +36,7 @@ export class NggvDropdownListComponent {
|
|
|
36
36
|
this.dropdownUtils = new DropdownUtils();
|
|
37
37
|
this._expanded = false;
|
|
38
38
|
this.closed$ = new Subject();
|
|
39
|
+
this._flattenedOptions = [];
|
|
39
40
|
if (this.translocoScope)
|
|
40
41
|
this.scope = this.translocoScope.toString();
|
|
41
42
|
}
|
|
@@ -75,20 +76,28 @@ export class NggvDropdownListComponent {
|
|
|
75
76
|
setExpanded(expanded = true) {
|
|
76
77
|
// update expanded state
|
|
77
78
|
this._expanded = expanded;
|
|
78
|
-
if (expanded)
|
|
79
|
+
if (expanded) {
|
|
79
80
|
this.refreshSelectedOption();
|
|
81
|
+
this.subscribeToKeyUpEvents();
|
|
82
|
+
this.subscribeToKeyDownEvents();
|
|
83
|
+
}
|
|
80
84
|
else {
|
|
81
85
|
this.closed$.next(true);
|
|
82
86
|
this.onClickSubscription?.unsubscribe();
|
|
87
|
+
this.onKeyDownSubscription?.unsubscribe();
|
|
88
|
+
this.onKeyUpSubscription?.unsubscribe();
|
|
89
|
+
// to trigger gc removal
|
|
90
|
+
this.onKeyDownSubscription = undefined;
|
|
91
|
+
this.onKeyUpSubscription = undefined;
|
|
83
92
|
}
|
|
84
93
|
}
|
|
85
94
|
/**
|
|
86
95
|
* @internal
|
|
87
96
|
*/
|
|
88
97
|
refreshSelectedOption() {
|
|
89
|
-
|
|
98
|
+
this._flattenedOptions = this.dropdownUtils.flattenOptions(this.options, !this.optionContentTpl);
|
|
90
99
|
this.activeIndex = this.getActiveIndex();
|
|
91
|
-
this.state =
|
|
100
|
+
this.state = this._flattenedOptions[this.activeIndex];
|
|
92
101
|
this.scrollToResult(this.state);
|
|
93
102
|
}
|
|
94
103
|
/**
|
|
@@ -100,15 +109,11 @@ export class NggvDropdownListComponent {
|
|
|
100
109
|
*/
|
|
101
110
|
getActiveIndex() {
|
|
102
111
|
if (!!this.selectedValue && this.selectedValue?.key != null) {
|
|
103
|
-
const selectedIndex = this.
|
|
104
|
-
.flattenOptions(this.options, !this.optionContentTpl)
|
|
105
|
-
.findIndex((option) => option.key != null && option.key === this.selectedValue?.key);
|
|
112
|
+
const selectedIndex = this._flattenedOptions.findIndex((option) => option.key != null && option.key === this.selectedValue?.key);
|
|
106
113
|
if (selectedIndex > -1)
|
|
107
114
|
return selectedIndex;
|
|
108
115
|
}
|
|
109
|
-
return this.
|
|
110
|
-
.flattenOptions(this.options, !this.optionContentTpl)
|
|
111
|
-
.findIndex((option) => option.key != null);
|
|
116
|
+
return this._flattenedOptions.findIndex((option) => option.key != null);
|
|
112
117
|
}
|
|
113
118
|
/**
|
|
114
119
|
* @internal
|
|
@@ -132,12 +137,20 @@ export class NggvDropdownListComponent {
|
|
|
132
137
|
*/
|
|
133
138
|
onKeyDown(event) {
|
|
134
139
|
switch (event.key) {
|
|
135
|
-
case '
|
|
140
|
+
case ' ': // Space - placed here to ensure the dropdown-list closes after selecting using "Space"
|
|
136
141
|
case 'ArrowUp': // Disable scrolling up
|
|
137
142
|
case 'ArrowDown': // Disable scrolling down
|
|
138
143
|
event.preventDefault();
|
|
139
144
|
event.stopPropagation();
|
|
140
145
|
return false;
|
|
146
|
+
case 'Enter': // Disable form submission and select the currently active value
|
|
147
|
+
event.preventDefault();
|
|
148
|
+
event.stopPropagation();
|
|
149
|
+
if (this.expanded) {
|
|
150
|
+
const option = this._flattenedOptions[this.activeIndex];
|
|
151
|
+
this.updateState(option, event);
|
|
152
|
+
}
|
|
153
|
+
return false;
|
|
141
154
|
}
|
|
142
155
|
return true;
|
|
143
156
|
}
|
|
@@ -149,7 +162,6 @@ export class NggvDropdownListComponent {
|
|
|
149
162
|
onKeyUp(event) {
|
|
150
163
|
if (!this.expanded)
|
|
151
164
|
return;
|
|
152
|
-
const options = this.dropdownUtils.flattenOptions(this.options, !this.optionContentTpl);
|
|
153
165
|
let option;
|
|
154
166
|
switch (event.key) {
|
|
155
167
|
case 'Tab':
|
|
@@ -157,14 +169,13 @@ export class NggvDropdownListComponent {
|
|
|
157
169
|
this.setExpanded(false);
|
|
158
170
|
this.closed.emit();
|
|
159
171
|
break;
|
|
160
|
-
case '
|
|
161
|
-
|
|
162
|
-
option = options[this.activeIndex];
|
|
172
|
+
case ' ': // Space - select the currently chosen value
|
|
173
|
+
option = this._flattenedOptions[this.activeIndex];
|
|
163
174
|
this.updateState(option, event);
|
|
164
175
|
break;
|
|
165
176
|
case 'Home': // Move to the first option
|
|
166
177
|
this.activeIndex = 0;
|
|
167
|
-
option =
|
|
178
|
+
option = this._flattenedOptions[this.activeIndex];
|
|
168
179
|
this.state = option;
|
|
169
180
|
this.scrollToResult(option);
|
|
170
181
|
break;
|
|
@@ -172,28 +183,40 @@ export class NggvDropdownListComponent {
|
|
|
172
183
|
if (this.activeIndex > 0)
|
|
173
184
|
this.activeIndex--;
|
|
174
185
|
else if (this.activeIndex === 0)
|
|
175
|
-
this.activeIndex =
|
|
176
|
-
option =
|
|
186
|
+
this.activeIndex = this._flattenedOptions.length - 1;
|
|
187
|
+
option = this._flattenedOptions[this.activeIndex];
|
|
177
188
|
this.state = option;
|
|
178
189
|
this.scrollToResult(option);
|
|
179
190
|
break;
|
|
180
191
|
case 'ArrowDown': // Move down one step to the next option
|
|
181
|
-
if (
|
|
192
|
+
if (this._flattenedOptions.length > this.activeIndex + 1)
|
|
182
193
|
this.activeIndex++;
|
|
183
|
-
else if (this.activeIndex ===
|
|
194
|
+
else if (this.activeIndex === this._flattenedOptions.length - 1)
|
|
184
195
|
this.activeIndex = 0;
|
|
185
|
-
option =
|
|
196
|
+
option = this._flattenedOptions[this.activeIndex];
|
|
186
197
|
this.state = option;
|
|
187
198
|
this.scrollToResult(option);
|
|
188
199
|
break;
|
|
189
200
|
case 'End': // Move to the last options
|
|
190
|
-
this.activeIndex =
|
|
191
|
-
option =
|
|
201
|
+
this.activeIndex = this._flattenedOptions.length - 1;
|
|
202
|
+
option = this._flattenedOptions[this.activeIndex];
|
|
192
203
|
this.state = option;
|
|
193
204
|
this.scrollToResult(option);
|
|
194
205
|
break;
|
|
195
206
|
}
|
|
196
207
|
}
|
|
208
|
+
/** @internal */
|
|
209
|
+
subscribeToKeyUpEvents() {
|
|
210
|
+
this.onKeyUpSubscription = fromEvent(document, 'keyup')
|
|
211
|
+
.pipe(filter(() => this.expanded), takeUntil(this.closed$))
|
|
212
|
+
.subscribe((event) => this.onKeyUp(event));
|
|
213
|
+
}
|
|
214
|
+
/** @internal */
|
|
215
|
+
subscribeToKeyDownEvents() {
|
|
216
|
+
this.onKeyDownSubscription = fromEvent(document, 'keydown')
|
|
217
|
+
.pipe(filter(() => this.expanded), takeUntil(this.closed$))
|
|
218
|
+
.subscribe((event) => this.onKeyDown(event));
|
|
219
|
+
}
|
|
197
220
|
/**
|
|
198
221
|
* Scrolls focused result into view with a specified offset.
|
|
199
222
|
* @param key the result index which to scroll to.
|
|
@@ -218,7 +241,7 @@ export class NggvDropdownListComponent {
|
|
|
218
241
|
}
|
|
219
242
|
}
|
|
220
243
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: NggvDropdownListComponent, deps: [{ token: TRANSLOCO_SCOPE, optional: true }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
221
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: NggvDropdownListComponent, selector: "nggv-dropdown-list", inputs: { expanded: "expanded", state: "state", scrollOffset: "scrollOffset", optionContentTpl: "optionContentTpl", groupLabelTpl: "groupLabelTpl", id: "id", thook: "thook", options: "options", textToHighlight: "textToHighlight", onlyEmitDistinctChanges: "onlyEmitDistinctChanges" }, outputs: { selectedValueChanged: "selectedValueChanged", closed: "closed" }, host: {
|
|
244
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: NggvDropdownListComponent, selector: "nggv-dropdown-list", inputs: { expanded: "expanded", state: "state", scrollOffset: "scrollOffset", optionContentTpl: "optionContentTpl", groupLabelTpl: "groupLabelTpl", id: "id", thook: "thook", options: "options", textToHighlight: "textToHighlight", onlyEmitDistinctChanges: "onlyEmitDistinctChanges" }, outputs: { selectedValueChanged: "selectedValueChanged", closed: "closed" }, host: { properties: { "attr.id": "this.id", "attr.data-thook": "this.thook" } }, viewQueries: [{ propertyName: "optionRefs", predicate: ["optionRefs"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<ng-container *transloco=\"let t; read: scope\">\n <ul\n class=\"gds-dropdown__options card options gds-reset\"\n [class.gds-dropdown__options-expanded]=\"expanded\"\n role=\"listbox\"\n tabindex=\"-1\"\n [attr.data-thook]=\"thook + '-options'\"\n [attr.aria-labelledby]=\"id + '-label'\"\n [attr.aria-activedescendant]=\"\n state ? id + '-option-' + state?.key : undefined\n \"\n >\n <ng-container *ngFor=\"let item of options\">\n <!-- OPTION -->\n <ng-container *ngIf=\"!isGroup(item)\">\n <ng-template\n *ngTemplateOutlet=\"listItemTemplate; context: { $implicit: item }\"\n ></ng-template>\n </ng-container>\n\n <!-- OPTION GROUP -->\n <li\n class=\"gds-dropdown__options__label group\"\n [attr.data-thook]=\"thook + '-option-group'\"\n *ngIf=\"isGroup(item)\"\n >\n <!-- group label template (default or custom) -->\n <ng-template\n *ngTemplateOutlet=\"listGroupTemplate; context: { $implicit: item }\"\n ></ng-template>\n\n <ul [attr.aria-disabled]=\"item.disabled\" class=\"gds-reset\">\n <ng-container *ngFor=\"let option of castGroup(item).options\">\n <ng-template\n *ngTemplateOutlet=\"\n listItemTemplate;\n context: { $implicit: option }\n \"\n ></ng-template>\n </ng-container>\n </ul>\n </li>\n </ng-container>\n </ul>\n\n <!-- TEMPLATE -->\n <ng-template #listItemTemplate let-option>\n <li\n #optionRefs\n *ngIf=\"!optionContentTpl\"\n tabindex=\"-1\"\n [id]=\"id + '-option-' + option.key\"\n class=\"gds-dropdown__options__label option\"\n role=\"option\"\n #liElem\n [attr.data-thook]=\"thook + '-option-' + option.key\"\n [attr.aria-disabled]=\"option.disabled\"\n [attr.aria-selected]=\"\n option.key === selectedValue?.key && !!selectedValue?.key\n \"\n [attr.aria-focus]=\"option.key === state?.key && !option.disabled\"\n [nggvTooltip]=\"isOverflow(liElem) ? t(option.label) : undefined\"\n (click)=\"updateState(option, $event)\"\n >\n <ng-template\n *ngTemplateOutlet=\"\n basicOptionContentTpl;\n context: { $implicit: option }\n \"\n >\n </ng-template>\n </li>\n <!-- Checking overflow on custom templates do not work skip adding nggvToolTip if custom template is provided -->\n <li\n #optionRefs\n *ngIf=\"!!optionContentTpl\"\n tabindex=\"-1\"\n [id]=\"id + '-option-' + option.key\"\n class=\"gds-dropdown__options__label option\"\n role=\"option\"\n #liElem\n [attr.data-thook]=\"thook + '-option-' + option.key\"\n [attr.aria-disabled]=\"option.disabled\"\n [attr.aria-selected]=\"\n option.key === selectedValue?.key && !!selectedValue?.key\n \"\n [attr.aria-focus]=\"option.key === state?.key && !option.disabled\"\n (click)=\"updateState(option, $event)\"\n >\n <ng-template\n *ngTemplateOutlet=\"optionContentTpl; context: { $implicit: option }\"\n >\n </ng-template>\n </li>\n </ng-template>\n\n <ng-template #listGroupTemplate let-item>\n <ng-container *ngIf=\"groupLabelTpl\">\n <ng-template\n *ngTemplateOutlet=\"groupLabelTpl; context: { $implicit: item }\"\n ></ng-template>\n </ng-container>\n <ng-container *ngIf=\"!groupLabelTpl\">\n <ng-template\n *ngTemplateOutlet=\"basicGroupLabelTpl; context: { $implicit: item }\"\n ></ng-template>\n </ng-container>\n </ng-template>\n\n <ng-template #basicOptionContentTpl let-option>\n <nggv-typeahead-highlight\n *ngIf=\"!!textToHighlight\"\n [textToHighlight]=\"textToHighlight\"\n [textContent]=\"t(option.label)\"\n >\n </nggv-typeahead-highlight>\n <ng-container *ngIf=\"!textToHighlight\">\n {{ t(option.label) }}\n </ng-container>\n </ng-template>\n\n <ng-template #basicGroupLabelTpl let-item>\n <div class=\"nggv-group-label\">{{ t(item.label) }}</div>\n </ng-template>\n</ng-container>\n", styles: [":host{--gds-ref-pallet-base300: hsl(0, 0%, 87%);--gds-ref-pallet-base400: hsl(0, 0%, 81%);--gds-ref-pallet-base500: hsl(0, 0%, 68%);--sg-z-index-dropdown: ;--sg-z-index-popover: 1060;--sg-z-index-dropdown-backdrop: 990;--sg-z-index-dropdown: 2000;--sg-popover-background: #fff;--sg-popover-box-shadow: 0 .125rem .375rem rgba(0, 0, 0, .15);--text-primary-color: #333;position:absolute;bottom:0;transform:translateY(calc(100% + .5rem));z-index:var(--sg-z-index-dropdown)}:host .hidden{visibility:hidden;display:none}:host ul[role=menu] [role=menuitem]{padding:.75rem;cursor:pointer}:host ul[role=menu] [role=menuitem]:hover,:host ul[role=menu] [role=menuitem]:focus-visible{background-color:var(--gds-ref-pallet-base200)}:host ul[role=menu] [role=menuitem]:active{background-color:var(--gds-ref-pallet-base300)}:host ul[role=menu] [role=menuitem]:focus{outline-color:#000;outline-offset:-.25rem}:host ul[role=listbox]{--z-index: var(--sg-z-index-popover);background-color:var(--sg-popover-background);flex-direction:column;justify-content:flex-end;inset:auto;z-index:var(--z-index);box-shadow:var(--sg-popover-box-shadow);color:var(--text-primary-color);padding:0;max-height:500px;overflow-y:auto;border:solid var(--sg-border-width) var(--sg-border-color);--border-color: var(--text-primary-color);--sg-border-color: var(--text-primary-color);border-radius:var(--sg-border-radius)}:host ul[role=listbox] [role=option]{padding:.75rem 1rem;line-height:1.25;cursor:pointer}:host ul[role=listbox] [role=option]:hover,:host ul[role=listbox] [role=option]:focus-visible{background-color:var(--grey-400)}:host ul[role=listbox] [role=option]:active{background-color:var(--grey-400)}:host ul[role=listbox] [role=option]:focus{outline-color:#000;outline-offset:-.25rem}:host ul[role=listbox] [role=option].active.sg-highlighted,:host ul[role=listbox] [role=option][aria-selected=true]{background:var(--grey-1000);color:#fff}:host ul[role=listbox] .group ul{list-style-type:none;padding:0}:host ul[role=listbox] .group:hover{color:inherit;background-color:inherit}:host .sg-fieldset-container{overflow-y:auto}:host .sg-fieldset-container fieldset[role=listbox][aria-multiselectable=true] [role=option]{display:flex;width:100%}:host .sg-fieldset-container fieldset[role=listbox][aria-multiselectable=true] [role=option].active.sg-highlighted input[type=checkbox]~i{border-color:#007ac7!important;box-shadow:0 0 .25em .0625em #41b0ee;outline-color:transparent;outline-style:solid}:host .gds-dropdown__options{padding-left:0;margin-bottom:0;margin-top:0;display:flex;flex-direction:column;list-style:none;display:none}:host .gds-dropdown__options>li{padding-bottom:.5rem;padding-top:.5rem;border:0;display:block;position:relative}:host .gds-dropdown__options>li:before{font-weight:500;display:inline-block;left:0;position:absolute;text-align:center}:host .gds-dropdown__options-expanded{display:block}:host .gds-dropdown__options__label:hover{background-color:var(--gds-ref-pallet-base400)}:host .gds-dropdown__options__label:focus-visible,:host .gds-dropdown__options__label[aria-focus=true]{background-color:var(--gds-ref-pallet-base300)}:host .gds-dropdown__options__label:active{background-color:var(--gds-ref-pallet-base500)}:host .gds-dropdown__options__label:focus-visible{outline-color:#000;outline-offset:-.25rem}:host .gds-dropdown__options__label[aria-hidden=true]{display:none}:host .gds-dropdown__options__label[highlighted]{color:#fff}:host li+.group{padding-top:0}:host .nggv-group-label{cursor:pointer;background-color:#e7e7e7;padding-inline:1rem;padding-block:.5rem}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i2.NggvTypeaheadHighlightComponent, selector: "nggv-typeahead-highlight", inputs: ["textContent", "textToHighlight"] }, { kind: "directive", type: i3.NggvTooltipDirective, selector: "[nggvTooltip]", inputs: ["nggvTooltip", "thook", "placement", "shown", "offset", "resizeThrottle", "tooltipId", "maxWidth"], outputs: ["nggvShow", "nggvHide"] }, { kind: "directive", type: i4.TranslocoDirective, selector: "[transloco]", inputs: ["transloco", "translocoParams", "translocoScope", "translocoRead", "translocoPrefix", "translocoLang", "translocoLoadingTpl"] }] }); }
|
|
222
245
|
}
|
|
223
246
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: NggvDropdownListComponent, decorators: [{
|
|
224
247
|
type: Component,
|
|
@@ -261,11 +284,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
261
284
|
type: Output
|
|
262
285
|
}], closed: [{
|
|
263
286
|
type: Output
|
|
264
|
-
}], onKeyDown: [{
|
|
265
|
-
type: HostListener,
|
|
266
|
-
args: ['document:keydown', ['$event']]
|
|
267
|
-
}], onKeyUp: [{
|
|
268
|
-
type: HostListener,
|
|
269
|
-
args: ['document:keyup', ['$event']]
|
|
270
287
|
}] } });
|
|
271
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dropdown-list.component.js","sourceRoot":"","sources":["../../../../../../../../libs/angular/src/v-angular/dropdown/dropdown-list/dropdown-list.component.ts","../../../../../../../../libs/angular/src/v-angular/dropdown/dropdown-list/dropdown-list.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ElementRef,\n  EventEmitter,\n  HostBinding,\n  HostListener,\n  Inject,\n  Input,\n  OnChanges,\n  OnInit,\n  Optional,\n  Output,\n  QueryList,\n  SimpleChanges,\n  TemplateRef,\n  ViewChildren,\n} from '@angular/core'\nimport { TRANSLOCO_SCOPE, TranslocoScope } from '@jsverse/transloco'\nimport { Subject, Subscription } from 'rxjs'\nimport scrollIntoView from 'scroll-into-view-if-needed'\n\nimport {\n  DropdownUtils,\n  Option,\n  OptionBase,\n} from '@sebgroup/green-angular/src/v-angular/core'\n\n@Component({\n  selector: 'nggv-dropdown-list',\n  templateUrl: './dropdown-list.component.html',\n  styleUrls: ['./dropdown-list.component.scss'],\n})\nexport class NggvDropdownListComponent implements OnInit, OnChanges {\n  @Input() set expanded(state: boolean) {\n    this.setExpanded(state)\n  }\n  get expanded(): boolean {\n    return this._expanded\n  }\n  @Input() state: any\n\n  /** The additional amount to show when option is scrolled into view. */\n  @Input() scrollOffset = 24\n\n  @Input() optionContentTpl: TemplateRef<OptionBase<any>> | undefined\n  @Input() groupLabelTpl: TemplateRef<OptionBase<any>> | undefined\n\n  /** @internal List of references to the option elements. */\n  @ViewChildren('optionRefs') optionRefs:\n    | QueryList<ElementRef<HTMLLIElement>>\n    | undefined\n\n  /** Id of the host element and is accessible by the children, automatically generated if not provided. */\n  @HostBinding('attr.id') @Input() id = (window as any).nggv?.nextId()\n\n  /** Special property used for selecting DOM elements during automated UI testing. */\n  @HostBinding('attr.data-thook') @Input() thook: string | null | undefined =\n    'dropdown'\n\n  @Input() options!: any[]\n\n  @Input() textToHighlight?: string\n\n  /**\n   * Used to control if \"selectedValueChanged\" only should emit distinct changes, or each time a value is selected\n   * When true, value is not emitted if there's no distinct change\n   * When false, value is emitted every time an option is selected\n   * */\n  @Input() onlyEmitDistinctChanges = true\n\n  @Output() selectedValueChanged = new EventEmitter<any>()\n\n  @Output() closed = new EventEmitter<void>()\n\n  /** The current active option based on numeric index. */\n  public activeIndex = -1\n\n  scope: string | undefined\n\n  private dropdownUtils: DropdownUtils<string | null, string, any> =\n    new DropdownUtils<string | null, string, any>()\n  private _expanded = false\n  private closed$ = new Subject<boolean>()\n  public selectedValue?: Option<string, any>\n\n  /** Subscribe if dropdown expanded to listen to click outside to close dropdown. */\n  protected onClickSubscription: Subscription | undefined\n\n  constructor(\n    @Optional()\n    @Inject(TRANSLOCO_SCOPE)\n    protected translocoScope: TranslocoScope,\n  ) {\n    if (this.translocoScope) this.scope = this.translocoScope.toString()\n  }\n\n  ngOnInit(): void {\n    if (this.state)\n      this.activeIndex = this.options.findIndex(\n        (option) => option.key === this.state.key,\n      )\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (!!changes.state && !changes.state.firstChange)\n      this.selectedValue = changes.state.currentValue\n    if (!!changes.options?.currentValue?.length && this.expanded)\n      this.refreshSelectedOption()\n  }\n\n  /**\n   * Returns true if argument is an {@link OptionGroup}.\n   * @param option the object to check.\n   */\n  isGroup(option: any): boolean {\n    return 'options' in option\n  }\n\n  /** @internal */\n  updateState(option: any, event: Event) {\n    if (option.disabled) return\n\n    if (\n      !this.onlyEmitDistinctChanges ||\n      !this.dropdownUtils.deepEqual(this.selectedValue, option)\n    ) {\n      this.selectedValue = option\n      this.state = option\n      this.selectedValueChanged.emit(option)\n    }\n\n    this.setExpanded(false)\n    event.stopPropagation()\n  }\n\n  /**\n   * @internal\n   */\n  setExpanded(expanded = true) {\n    // update expanded state\n    this._expanded = expanded\n\n    if (expanded) this.refreshSelectedOption()\n    else {\n      this.closed$.next(true)\n      this.onClickSubscription?.unsubscribe()\n    }\n  }\n\n  /**\n   * @internal\n   */\n  refreshSelectedOption() {\n    const options = this.dropdownUtils.flattenOptions(\n      this.options,\n      !this.optionContentTpl,\n    )\n    this.activeIndex = this.getActiveIndex()\n    this.state = options[this.activeIndex]\n    this.scrollToResult(this.state)\n  }\n\n  /**\n   * @internal\n   * @returns The active index (number) if option is found, -1 otherwise.\n   * - If a selectedValue exists that's not nullish and that options is found, return that index\n   * - Else, return first non nullish index\n   * - If none of the above criterias are met, -1 are returned\n   */\n  getActiveIndex(): number {\n    if (!!this.selectedValue && this.selectedValue?.key != null) {\n      const selectedIndex = this.dropdownUtils\n        .flattenOptions(this.options, !this.optionContentTpl)\n        .findIndex(\n          (option) =>\n            option.key != null && option.key === this.selectedValue?.key,\n        )\n      if (selectedIndex > -1) return selectedIndex\n    }\n    return this.dropdownUtils\n      .flattenOptions(this.options, !this.optionContentTpl)\n      .findIndex((option) => option.key != null)\n  }\n\n  /**\n   * @internal\n   * evaluates wether the HTML element overflows\n   * @param elem The HTMLElement to evaluate\n   * */\n  isOverflow(elem: HTMLElement) {\n    return elem.offsetWidth < elem.scrollWidth\n  }\n\n  /**\n   * Typecast anything to an {@link OptionGroup}.\n   * @param group the object to typecast.\n   */\n  castGroup(group: any): any {\n    return group\n  }\n\n  /**\n   * @internal\n   * Disables default events.\n   * @param event fired containing which key was pressed.\n   */\n  @HostListener('document:keydown', ['$event'])\n  onKeyDown(event: KeyboardEvent) {\n    switch (event.key) {\n      case 'Enter': //  Disable form submission\n      case 'ArrowUp': // Disable scrolling up\n      case 'ArrowDown': // Disable scrolling down\n        event.preventDefault()\n        event.stopPropagation()\n        return false\n    }\n    return true\n  }\n\n  /**\n   * @internal\n   * Enter toggles the dropdown, home, end, and, arrows change the index.\n   * @param event fired containing which key was released.\n   */\n  @HostListener('document:keyup', ['$event'])\n  onKeyUp(event: KeyboardEvent) {\n    if (!this.expanded) return\n    const options = this.dropdownUtils.flattenOptions(\n      this.options,\n      !this.optionContentTpl,\n    )\n    let option\n\n    switch (event.key) {\n      case 'Tab':\n      case 'Escape':\n        this.setExpanded(false)\n        this.closed.emit()\n        break\n      case 'Space': // Select the currently chosen value\n      case 'Enter': // Select the currently chosen value\n        option = options[this.activeIndex]\n        this.updateState(option, event)\n        break\n\n      case 'Home': // Move to the first option\n        this.activeIndex = 0\n\n        option = options[this.activeIndex]\n        this.state = option\n        this.scrollToResult(option)\n        break\n\n      case 'ArrowUp': // Move up one step to the previous option\n        if (this.activeIndex > 0) this.activeIndex--\n        else if (this.activeIndex === 0) this.activeIndex = options.length - 1\n\n        option = options[this.activeIndex]\n        this.state = option\n        this.scrollToResult(option)\n        break\n\n      case 'ArrowDown': // Move down one step to the next option\n        if (options.length > this.activeIndex + 1) this.activeIndex++\n        else if (this.activeIndex === options.length - 1) this.activeIndex = 0\n\n        option = options[this.activeIndex]\n        this.state = option\n        this.scrollToResult(option)\n        break\n\n      case 'End': // Move to the last options\n        this.activeIndex = options.length - 1\n\n        option = options[this.activeIndex]\n        this.state = option\n        this.scrollToResult(option)\n        break\n    }\n  }\n\n  /**\n   * Scrolls focused result into view with a specified offset.\n   * @param key the result index which to scroll to.\n   */\n  scrollToResult(option: any) {\n    if (!this.optionRefs || !option) return\n    const optionRef = this.optionRefs.find(\n      (li) => li.nativeElement.id === this.id + '-option-' + option.key,\n    )\n    const offset = this.scrollOffset\n    if (optionRef) {\n      let delta = window.scrollY || document.documentElement.scrollTop\n\n      // The list seems not to be visible at the time of scrolling, but this setTimeout \"hack\" makes it work...\n      setTimeout(() => {\n        scrollIntoView(optionRef.nativeElement, {\n          scrollMode: 'if-needed',\n          block: 'nearest',\n        })\n\n        delta -= window.scrollY || document.documentElement.scrollTop\n        if (delta) window.scrollBy(0, delta > 0 ? -offset : offset)\n      }, 0)\n    }\n  }\n}\n","<ng-container *transloco=\"let t; read: scope\">\n  <ul\n    class=\"gds-dropdown__options card options gds-reset\"\n    [class.gds-dropdown__options-expanded]=\"expanded\"\n    role=\"listbox\"\n    tabindex=\"-1\"\n    [attr.data-thook]=\"thook + '-options'\"\n    [attr.aria-labelledby]=\"id + '-label'\"\n    [attr.aria-activedescendant]=\"\n      state ? id + '-option-' + state?.key : undefined\n    \"\n  >\n    <ng-container *ngFor=\"let item of options\">\n      <!-- OPTION -->\n      <ng-container *ngIf=\"!isGroup(item)\">\n        <ng-template\n          *ngTemplateOutlet=\"listItemTemplate; context: { $implicit: item }\"\n        ></ng-template>\n      </ng-container>\n\n      <!-- OPTION GROUP -->\n      <li\n        class=\"gds-dropdown__options__label group\"\n        [attr.data-thook]=\"thook + '-option-group'\"\n        *ngIf=\"isGroup(item)\"\n      >\n        <!-- group label template (default or custom) -->\n        <ng-template\n          *ngTemplateOutlet=\"listGroupTemplate; context: { $implicit: item }\"\n        ></ng-template>\n\n        <ul [attr.aria-disabled]=\"item.disabled\" class=\"gds-reset\">\n          <ng-container *ngFor=\"let option of castGroup(item).options\">\n            <ng-template\n              *ngTemplateOutlet=\"\n                listItemTemplate;\n                context: { $implicit: option }\n              \"\n            ></ng-template>\n          </ng-container>\n        </ul>\n      </li>\n    </ng-container>\n  </ul>\n\n  <!-- TEMPLATE -->\n  <ng-template #listItemTemplate let-option>\n    <li\n      #optionRefs\n      *ngIf=\"!optionContentTpl\"\n      tabindex=\"-1\"\n      [id]=\"id + '-option-' + option.key\"\n      class=\"gds-dropdown__options__label option\"\n      role=\"option\"\n      #liElem\n      [attr.data-thook]=\"thook + '-option-' + option.key\"\n      [attr.aria-disabled]=\"option.disabled\"\n      [attr.aria-selected]=\"\n        option.key === selectedValue?.key && !!selectedValue?.key\n      \"\n      [attr.aria-focus]=\"option.key === state?.key && !option.disabled\"\n      [nggvTooltip]=\"isOverflow(liElem) ? t(option.label) : undefined\"\n      (click)=\"updateState(option, $event)\"\n    >\n      <ng-template\n        *ngTemplateOutlet=\"\n          basicOptionContentTpl;\n          context: { $implicit: option }\n        \"\n      >\n      </ng-template>\n    </li>\n    <!-- Checking overflow on custom templates do not work skip adding nggvToolTip if custom template is provided -->\n    <li\n      #optionRefs\n      *ngIf=\"!!optionContentTpl\"\n      tabindex=\"-1\"\n      [id]=\"id + '-option-' + option.key\"\n      class=\"gds-dropdown__options__label option\"\n      role=\"option\"\n      #liElem\n      [attr.data-thook]=\"thook + '-option-' + option.key\"\n      [attr.aria-disabled]=\"option.disabled\"\n      [attr.aria-selected]=\"\n        option.key === selectedValue?.key && !!selectedValue?.key\n      \"\n      [attr.aria-focus]=\"option.key === state?.key && !option.disabled\"\n      (click)=\"updateState(option, $event)\"\n    >\n      <ng-template\n        *ngTemplateOutlet=\"optionContentTpl; context: { $implicit: option }\"\n      >\n      </ng-template>\n    </li>\n  </ng-template>\n\n  <ng-template #listGroupTemplate let-item>\n    <ng-container *ngIf=\"groupLabelTpl\">\n      <ng-template\n        *ngTemplateOutlet=\"groupLabelTpl; context: { $implicit: item }\"\n      ></ng-template>\n    </ng-container>\n    <ng-container *ngIf=\"!groupLabelTpl\">\n      <ng-template\n        *ngTemplateOutlet=\"basicGroupLabelTpl; context: { $implicit: item }\"\n      ></ng-template>\n    </ng-container>\n  </ng-template>\n\n  <ng-template #basicOptionContentTpl let-option>\n    <nggv-typeahead-highlight\n      *ngIf=\"!!textToHighlight\"\n      [textToHighlight]=\"textToHighlight\"\n      [textContent]=\"t(option.label)\"\n    >\n    </nggv-typeahead-highlight>\n    <ng-container *ngIf=\"!textToHighlight\">\n      {{ t(option.label) }}\n    </ng-container>\n  </ng-template>\n\n  <ng-template #basicGroupLabelTpl let-item>\n    <div class=\"nggv-group-label\">{{ t(item.label) }}</div>\n  </ng-template>\n</ng-container>\n"]}
|
|
288
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dropdown-list.component.js","sourceRoot":"","sources":["../../../../../../../../libs/angular/src/v-angular/dropdown/dropdown-list/dropdown-list.component.ts","../../../../../../../../libs/angular/src/v-angular/dropdown/dropdown-list/dropdown-list.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ElementRef,\n  EventEmitter,\n  HostBinding,\n  Inject,\n  Input,\n  OnChanges,\n  OnInit,\n  Optional,\n  Output,\n  QueryList,\n  SimpleChanges,\n  TemplateRef,\n  ViewChildren,\n} from '@angular/core'\nimport { TRANSLOCO_SCOPE, TranslocoScope } from '@jsverse/transloco'\nimport { filter, fromEvent, Subject, Subscription, takeUntil } from 'rxjs'\nimport scrollIntoView from 'scroll-into-view-if-needed'\n\nimport {\n  DropdownUtils,\n  Option,\n  OptionBase,\n} from '@sebgroup/green-angular/src/v-angular/core'\n\n@Component({\n  selector: 'nggv-dropdown-list',\n  templateUrl: './dropdown-list.component.html',\n  styleUrls: ['./dropdown-list.component.scss'],\n})\nexport class NggvDropdownListComponent implements OnInit, OnChanges {\n  @Input() set expanded(state: boolean) {\n    this.setExpanded(state)\n  }\n  get expanded(): boolean {\n    return this._expanded\n  }\n  @Input() state: any\n\n  /** The additional amount to show when option is scrolled into view. */\n  @Input() scrollOffset = 24\n\n  @Input() optionContentTpl: TemplateRef<OptionBase<any>> | undefined\n  @Input() groupLabelTpl: TemplateRef<OptionBase<any>> | undefined\n\n  /** @internal List of references to the option elements. */\n  @ViewChildren('optionRefs') optionRefs:\n    | QueryList<ElementRef<HTMLLIElement>>\n    | undefined\n\n  /** Id of the host element and is accessible by the children, automatically generated if not provided. */\n  @HostBinding('attr.id') @Input() id = (window as any).nggv?.nextId()\n\n  /** Special property used for selecting DOM elements during automated UI testing. */\n  @HostBinding('attr.data-thook') @Input() thook: string | null | undefined =\n    'dropdown'\n\n  @Input() options!: any[]\n\n  @Input() textToHighlight?: string\n\n  /**\n   * Used to control if \"selectedValueChanged\" only should emit distinct changes, or each time a value is selected\n   * When true, value is not emitted if there's no distinct change\n   * When false, value is emitted every time an option is selected\n   * */\n  @Input() onlyEmitDistinctChanges = true\n\n  @Output() selectedValueChanged = new EventEmitter<any>()\n\n  @Output() closed = new EventEmitter<void>()\n\n  /** The current active option based on numeric index. */\n  public activeIndex = -1\n\n  scope: string | undefined\n\n  private dropdownUtils: DropdownUtils<string | null, string, any> =\n    new DropdownUtils<string | null, string, any>()\n  private _expanded = false\n  private closed$ = new Subject<boolean>()\n  public selectedValue?: Option<string, any>\n\n  /** Subscribe if dropdown expanded to listen to click outside to close dropdown. */\n  protected onClickSubscription: Subscription | undefined\n  /** Subscribe to to keyboard events only when list is open. */\n  private onKeyDownSubscription: Subscription | undefined\n  private onKeyUpSubscription: Subscription | undefined\n  private _flattenedOptions: any[] = []\n\n  constructor(\n    @Optional()\n    @Inject(TRANSLOCO_SCOPE)\n    protected translocoScope: TranslocoScope,\n  ) {\n    if (this.translocoScope) this.scope = this.translocoScope.toString()\n  }\n\n  ngOnInit(): void {\n    if (this.state)\n      this.activeIndex = this.options.findIndex(\n        (option) => option.key === this.state.key,\n      )\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (!!changes.state && !changes.state.firstChange)\n      this.selectedValue = changes.state.currentValue\n    if (!!changes.options?.currentValue?.length && this.expanded)\n      this.refreshSelectedOption()\n  }\n\n  /**\n   * Returns true if argument is an {@link OptionGroup}.\n   * @param option the object to check.\n   */\n  isGroup(option: any): boolean {\n    return 'options' in option\n  }\n\n  /** @internal */\n  updateState(option: any, event: Event) {\n    if (option.disabled) return\n\n    if (\n      !this.onlyEmitDistinctChanges ||\n      !this.dropdownUtils.deepEqual(this.selectedValue, option)\n    ) {\n      this.selectedValue = option\n      this.state = option\n      this.selectedValueChanged.emit(option)\n    }\n\n    this.setExpanded(false)\n    event.stopPropagation()\n  }\n\n  /**\n   * @internal\n   */\n  setExpanded(expanded = true) {\n    // update expanded state\n    this._expanded = expanded\n\n    if (expanded) {\n      this.refreshSelectedOption()\n      this.subscribeToKeyUpEvents()\n      this.subscribeToKeyDownEvents()\n    } else {\n      this.closed$.next(true)\n      this.onClickSubscription?.unsubscribe()\n      this.onKeyDownSubscription?.unsubscribe()\n      this.onKeyUpSubscription?.unsubscribe()\n      // to trigger gc removal\n      this.onKeyDownSubscription = undefined\n      this.onKeyUpSubscription = undefined\n    }\n  }\n\n  /**\n   * @internal\n   */\n  refreshSelectedOption() {\n    this._flattenedOptions = this.dropdownUtils.flattenOptions(\n      this.options,\n      !this.optionContentTpl,\n    )\n    this.activeIndex = this.getActiveIndex()\n    this.state = this._flattenedOptions[this.activeIndex]\n    this.scrollToResult(this.state)\n  }\n\n  /**\n   * @internal\n   * @returns The active index (number) if option is found, -1 otherwise.\n   * - If a selectedValue exists that's not nullish and that options is found, return that index\n   * - Else, return first non nullish index\n   * - If none of the above criterias are met, -1 are returned\n   */\n  getActiveIndex(): number {\n    if (!!this.selectedValue && this.selectedValue?.key != null) {\n      const selectedIndex = this._flattenedOptions.findIndex(\n        (option) =>\n          option.key != null && option.key === this.selectedValue?.key,\n      )\n      if (selectedIndex > -1) return selectedIndex\n    }\n    return this._flattenedOptions.findIndex((option) => option.key != null)\n  }\n\n  /**\n   * @internal\n   * evaluates wether the HTML element overflows\n   * @param elem The HTMLElement to evaluate\n   * */\n  isOverflow(elem: HTMLElement) {\n    return elem.offsetWidth < elem.scrollWidth\n  }\n\n  /**\n   * Typecast anything to an {@link OptionGroup}.\n   * @param group the object to typecast.\n   */\n  castGroup(group: any): any {\n    return group\n  }\n\n  /**\n   * @internal\n   * Disables default events.\n   * @param event fired containing which key was pressed.\n   */\n  onKeyDown(event: KeyboardEvent) {\n    switch (event.key) {\n      case ' ': // Space - placed here to ensure the dropdown-list closes after selecting using \"Space\"\n      case 'ArrowUp': // Disable scrolling up\n      case 'ArrowDown': // Disable scrolling down\n        event.preventDefault()\n        event.stopPropagation()\n        return false\n      case 'Enter': // Disable form submission and select the currently active value\n        event.preventDefault()\n        event.stopPropagation()\n        if (this.expanded) {\n          const option = this._flattenedOptions[this.activeIndex]\n          this.updateState(option, event)\n        }\n        return false\n    }\n    return true\n  }\n\n  /**\n   * @internal\n   * Enter toggles the dropdown, home, end, and, arrows change the index.\n   * @param event fired containing which key was released.\n   */\n  onKeyUp(event: KeyboardEvent) {\n    if (!this.expanded) return\n    let option\n\n    switch (event.key) {\n      case 'Tab':\n      case 'Escape':\n        this.setExpanded(false)\n        this.closed.emit()\n        break\n\n      case ' ': // Space - select the currently chosen value\n        option = this._flattenedOptions[this.activeIndex]\n        this.updateState(option, event)\n        break\n\n      case 'Home': // Move to the first option\n        this.activeIndex = 0\n\n        option = this._flattenedOptions[this.activeIndex]\n        this.state = option\n        this.scrollToResult(option)\n        break\n\n      case 'ArrowUp': // Move up one step to the previous option\n        if (this.activeIndex > 0) this.activeIndex--\n        else if (this.activeIndex === 0)\n          this.activeIndex = this._flattenedOptions.length - 1\n\n        option = this._flattenedOptions[this.activeIndex]\n        this.state = option\n        this.scrollToResult(option)\n        break\n\n      case 'ArrowDown': // Move down one step to the next option\n        if (this._flattenedOptions.length > this.activeIndex + 1)\n          this.activeIndex++\n        else if (this.activeIndex === this._flattenedOptions.length - 1)\n          this.activeIndex = 0\n\n        option = this._flattenedOptions[this.activeIndex]\n        this.state = option\n        this.scrollToResult(option)\n        break\n\n      case 'End': // Move to the last options\n        this.activeIndex = this._flattenedOptions.length - 1\n\n        option = this._flattenedOptions[this.activeIndex]\n        this.state = option\n        this.scrollToResult(option)\n        break\n    }\n  }\n\n  /** @internal */\n  subscribeToKeyUpEvents() {\n    this.onKeyUpSubscription = fromEvent<KeyboardEvent>(document, 'keyup')\n      .pipe(\n        filter(() => this.expanded),\n        takeUntil(this.closed$),\n      )\n      .subscribe((event) => this.onKeyUp(event))\n  }\n\n  /** @internal */\n  subscribeToKeyDownEvents() {\n    this.onKeyDownSubscription = fromEvent<KeyboardEvent>(document, 'keydown')\n      .pipe(\n        filter(() => this.expanded),\n        takeUntil(this.closed$),\n      )\n      .subscribe((event) => this.onKeyDown(event))\n  }\n\n  /**\n   * Scrolls focused result into view with a specified offset.\n   * @param key the result index which to scroll to.\n   */\n  scrollToResult(option: any) {\n    if (!this.optionRefs || !option) return\n    const optionRef = this.optionRefs.find(\n      (li) => li.nativeElement.id === this.id + '-option-' + option.key,\n    )\n    const offset = this.scrollOffset\n    if (optionRef) {\n      let delta = window.scrollY || document.documentElement.scrollTop\n\n      // The list seems not to be visible at the time of scrolling, but this setTimeout \"hack\" makes it work...\n      setTimeout(() => {\n        scrollIntoView(optionRef.nativeElement, {\n          scrollMode: 'if-needed',\n          block: 'nearest',\n        })\n\n        delta -= window.scrollY || document.documentElement.scrollTop\n        if (delta) window.scrollBy(0, delta > 0 ? -offset : offset)\n      }, 0)\n    }\n  }\n}\n","<ng-container *transloco=\"let t; read: scope\">\n  <ul\n    class=\"gds-dropdown__options card options gds-reset\"\n    [class.gds-dropdown__options-expanded]=\"expanded\"\n    role=\"listbox\"\n    tabindex=\"-1\"\n    [attr.data-thook]=\"thook + '-options'\"\n    [attr.aria-labelledby]=\"id + '-label'\"\n    [attr.aria-activedescendant]=\"\n      state ? id + '-option-' + state?.key : undefined\n    \"\n  >\n    <ng-container *ngFor=\"let item of options\">\n      <!-- OPTION -->\n      <ng-container *ngIf=\"!isGroup(item)\">\n        <ng-template\n          *ngTemplateOutlet=\"listItemTemplate; context: { $implicit: item }\"\n        ></ng-template>\n      </ng-container>\n\n      <!-- OPTION GROUP -->\n      <li\n        class=\"gds-dropdown__options__label group\"\n        [attr.data-thook]=\"thook + '-option-group'\"\n        *ngIf=\"isGroup(item)\"\n      >\n        <!-- group label template (default or custom) -->\n        <ng-template\n          *ngTemplateOutlet=\"listGroupTemplate; context: { $implicit: item }\"\n        ></ng-template>\n\n        <ul [attr.aria-disabled]=\"item.disabled\" class=\"gds-reset\">\n          <ng-container *ngFor=\"let option of castGroup(item).options\">\n            <ng-template\n              *ngTemplateOutlet=\"\n                listItemTemplate;\n                context: { $implicit: option }\n              \"\n            ></ng-template>\n          </ng-container>\n        </ul>\n      </li>\n    </ng-container>\n  </ul>\n\n  <!-- TEMPLATE -->\n  <ng-template #listItemTemplate let-option>\n    <li\n      #optionRefs\n      *ngIf=\"!optionContentTpl\"\n      tabindex=\"-1\"\n      [id]=\"id + '-option-' + option.key\"\n      class=\"gds-dropdown__options__label option\"\n      role=\"option\"\n      #liElem\n      [attr.data-thook]=\"thook + '-option-' + option.key\"\n      [attr.aria-disabled]=\"option.disabled\"\n      [attr.aria-selected]=\"\n        option.key === selectedValue?.key && !!selectedValue?.key\n      \"\n      [attr.aria-focus]=\"option.key === state?.key && !option.disabled\"\n      [nggvTooltip]=\"isOverflow(liElem) ? t(option.label) : undefined\"\n      (click)=\"updateState(option, $event)\"\n    >\n      <ng-template\n        *ngTemplateOutlet=\"\n          basicOptionContentTpl;\n          context: { $implicit: option }\n        \"\n      >\n      </ng-template>\n    </li>\n    <!-- Checking overflow on custom templates do not work skip adding nggvToolTip if custom template is provided -->\n    <li\n      #optionRefs\n      *ngIf=\"!!optionContentTpl\"\n      tabindex=\"-1\"\n      [id]=\"id + '-option-' + option.key\"\n      class=\"gds-dropdown__options__label option\"\n      role=\"option\"\n      #liElem\n      [attr.data-thook]=\"thook + '-option-' + option.key\"\n      [attr.aria-disabled]=\"option.disabled\"\n      [attr.aria-selected]=\"\n        option.key === selectedValue?.key && !!selectedValue?.key\n      \"\n      [attr.aria-focus]=\"option.key === state?.key && !option.disabled\"\n      (click)=\"updateState(option, $event)\"\n    >\n      <ng-template\n        *ngTemplateOutlet=\"optionContentTpl; context: { $implicit: option }\"\n      >\n      </ng-template>\n    </li>\n  </ng-template>\n\n  <ng-template #listGroupTemplate let-item>\n    <ng-container *ngIf=\"groupLabelTpl\">\n      <ng-template\n        *ngTemplateOutlet=\"groupLabelTpl; context: { $implicit: item }\"\n      ></ng-template>\n    </ng-container>\n    <ng-container *ngIf=\"!groupLabelTpl\">\n      <ng-template\n        *ngTemplateOutlet=\"basicGroupLabelTpl; context: { $implicit: item }\"\n      ></ng-template>\n    </ng-container>\n  </ng-template>\n\n  <ng-template #basicOptionContentTpl let-option>\n    <nggv-typeahead-highlight\n      *ngIf=\"!!textToHighlight\"\n      [textToHighlight]=\"textToHighlight\"\n      [textContent]=\"t(option.label)\"\n    >\n    </nggv-typeahead-highlight>\n    <ng-container *ngIf=\"!textToHighlight\">\n      {{ t(option.label) }}\n    </ng-container>\n  </ng-template>\n\n  <ng-template #basicGroupLabelTpl let-item>\n    <div class=\"nggv-group-label\">{{ t(item.label) }}</div>\n  </ng-template>\n</ng-container>\n"]}
|
|
@@ -195,11 +195,11 @@ export class NggvDropdownComponent extends NggvBaseControlValueAccessorComponent
|
|
|
195
195
|
return !('options' in option);
|
|
196
196
|
}
|
|
197
197
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: NggvDropdownComponent, deps: [{ token: i1.NgControl, optional: true, self: true }, { token: TRANSLOCO_SCOPE, optional: true }, { token: i0.ChangeDetectorRef }, { token: i2.DropdownUtils }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
198
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: NggvDropdownComponent, selector: "nggv-dropdown", inputs: { thook: "thook", placeholder: "placeholder", ariaLabel: "ariaLabel", options: "options", scrollOffset: "scrollOffset", allowControlNullishOption: "allowControlNullishOption", textToHighlight: "textToHighlight", selectOnSingleOption: "selectOnSingleOption" }, outputs: { expandedChange: "expandedChange" }, host: { listeners: { "keyup": "onKeyUp($event)" }, properties: { "attr.data-thook": "this.thook" } }, queries: [{ propertyName: "selectedContentTpl", first: true, predicate: ["selectedTpl"], descendants: true, read: TemplateRef }, { propertyName: "optionContentTpl", first: true, predicate: ["optionTpl"], descendants: true, read: TemplateRef }, { propertyName: "groupLabelTpl", first: true, predicate: ["groupLabelTpl"], descendants: true, read: TemplateRef }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<!-- LABEL -->\n<ng-container *transloco=\"let t; read: scope\">\n <label\n [id]=\"id + '-label'\"\n class=\"gds-field-label hide-if-empty\"\n [attr.for]=\"id + '-toggle'\"\n >\n <ng-template\n *ngTemplateOutlet=\"labelContentTpl || basicLabelContentTpl\"\n ></ng-template>\n <ng-template #basicLabelContentTpl>\n <!-- to trigger css:empty if no label was added -->\n <ng-container *ngIf=\"label\">\n {{ label }}\n <span\n *ngIf=\"optional === true || (required !== true && optional !== false)\"\n class=\"gds-field-label--optional\"\n >\n ({{ t('label.optional') }})\n </span>\n </ng-container>\n </ng-template>\n </label>\n\n <!-- DESCRIPTION -->\n <div class=\"description\">{{ description }}</div>\n\n <!-- LOCKED INPUT -->\n <ng-container *ngIf=\"locked\">\n <div\n [id]=\"id + '-input'\"\n class=\"nggv-field--locked\"\n [attr.name]=\"name\"\n [attr.value]=\"state\"\n [attr.role]=\"role\"\n [attr.aria-labelledby]=\"id + '-label ' + id + '-input'\"\n >\n <span *ngIf=\"!state\" class=\"unset-state\">-</span>\n <ng-container *ngIf=\"state\">\n <ng-template\n *ngTemplateOutlet=\"\n selectedContentTpl || defaultSelectedContentTpl;\n context: { $implicit: state }\n \"\n >\n </ng-template>\n </ng-container>\n </div>\n </ng-container>\n\n <!-- INPUT -->\n <ng-container *ngIf=\"!locked\">\n <div #input [id]=\"id + '-input'\" class=\"dropdown\">\n <button\n [id]=\"id + '-toggle'\"\n [disabled]=\"disabled\"\n type=\"button\"\n class=\"nggv-field-dropdown__label toggle\"\n [class.nggv-field--error]=\"invalid\"\n aria-haspopup=\"listbox\"\n [attr.data-thook]=\"thook + '-toggle'\"\n [attr.aria-expanded]=\"expanded\"\n [attr.aria-labelledby]=\"\n ariaLabel ? null : id + '-label ' + id + '-toggle'\n \"\n [attr.aria-label]=\"ariaLabel || null\"\n (click)=\"toggleDropdown()\"\n >\n <ng-template\n *ngTemplateOutlet=\"\n selectedContentTpl || defaultSelectedContentTpl;\n context: { $implicit: state }\n \"\n >\n </ng-template>\n </button>\n\n <nggv-dropdown-list\n #dropDownList\n [options]=\"options\"\n [scrollOffset]=\"scrollOffset\"\n [state]=\"state\"\n [expanded]=\"expanded\"\n [optionContentTpl]=\"optionContentTpl\"\n [groupLabelTpl]=\"groupLabelTpl\"\n [textToHighlight]=\"textToHighlight\"\n (closed)=\"setExpanded(false)\"\n (selectedValueChanged)=\"onSelectChange($event)\"\n >\n </nggv-dropdown-list>\n </div>\n <!-- ERRORS -->\n <div class=\"gds-form-item__footer error-wrapper\">\n <span\n class=\"form-info form-info--error\"\n [attr.for]=\"id + '-input'\"\n *ngIf=\"invalid && (error || ngControl?.invalid)\"\n >\n <span class=\"error-icon\">\n <gds-icon-triangle-exclamation\n width=\"16\"\n height=\"16\"\n [solid]=\"true\"\n *nggCoreElement\n ></gds-icon-triangle-exclamation>\n </span>\n <span\n *ngIf=\"error; else errorsRef\"\n [attr.data-thook]=\"thook + '-errorlabel'\"\n >{{ error }}</span\n >\n </span>\n <ng-template #errorsRef>\n <span\n *ngIf=\"firstError as error\"\n [attr.data-thook]=\"thook + '-errorlabel'\"\n >\n {{ t('error.field' + error?.code, error?.params) }}\n </span>\n </ng-template>\n </div>\n\n <!-- CHILDREN -->\n <ng-content></ng-content>\n </ng-container>\n\n <ng-template #defaultSelectedContentTpl let-state>\n <!-- eslint-disable-next-line @angular-eslint/template/eqeqeq -->\n {{ state?.key != null && state?.label ? t(state.label) : placeholder }}\n </ng-template>\n</ng-container>\n", styles: [":host{--sg-border-radius: .25rem;--sg-border-width: 1px;--sg-border-color: #868686;--text-primary-color: #333;--sg-form-control-bg: #fff;--grey-000: hsl(0 0% 100%);--grey-100: hsl(0, 0%, 97%);--grey-200: hsl(0, 0%, 91%);--grey-300: hsl(0, 0%, 87%);--grey-400: hsl(0, 0%, 81%);--grey-500: hsl(0, 0%, 68%);--grey-600: hsl(0, 0%, 53%);--grey-700: hsl(0, 0%, 29%);--grey-800: hsl(0, 0%, 20%);--grey-900: hsl(0, 0%, 10%);--grey-1000: hsl(0 0% 0%)}:host label{display:block;font-weight:500;line-height:1.25rem;margin-bottom:.5rem;width:100%}:host .description{margin-bottom:.5rem;line-height:1.25rem;width:100%}:host .gds-field-label--optional{font-weight:400}:host button{background-color:transparent;border:0;cursor:pointer;font-family:inherit;padding:0;padding:.75rem 1rem;border-radius:var(--sg-border-radius);border:solid var(--sg-border-width) var(--sg-border-color);--border-color: var(--grey-600);--sg-border-color: var(--grey-600);background:var(--sg-form-control-bg);color:var(--text-primary-color);min-height:2.75rem;display:flex;flex-wrap:nowrap;justify-content:space-between;align-items:center;max-width:100%;font-size:1rem;font-weight:400;line-height:1.125;width:100%}:host button:focus{outline-color:var(--gds-sys-color-focus-outline);outline-style:solid;outline-width:.125rem;outline-offset:.125rem}@media (max-width: 35.98em){:host button{min-width:100%}}:host button>span{white-space:nowrap;overflow:hidden;text-overflow:ellipsis}:host button:after{margin-left:.5rem;margin-right:.5rem;border-bottom:solid 2px var(--text-primary-color);border-left:solid 2px var(--text-primary-color);content:\"\";display:block;height:.5rem;width:.5rem;position:relative;top:-.15rem;transform:translate(75%) rotate3d(0,0,1,-45deg) scaleZ(-1);transition:transform .3s ease-in;flex-shrink:0}:host button[aria-expanded=true]:after{transform:translate(75%,6px) rotate3d(0,0,1,-45deg) scale3d(-1,-1,1)}:host button:hover:after{border-color:currentColor}:host button:disabled{--background: var(--grey-500)}:host button.small{font-size:.875rem}:host button:hover{background:#e7e7e7}:host button:active{background:inherit;color:inherit;border-color:inherit}:host button:disabled,:host button.disabled{--text-primary-color: var(--grey-600);background:var(--grey-300);color:var(--grey-600);cursor:not-allowed}:host button.nggv-field--error{border-bottom:2px solid #9f000a}:host .gds-form-item__footer .form-info{font-weight:500;font-size:.875rem}:host .gds-form-item__footer .form-info--error{display:flex;align-items:flex-start;gap:.5em;color:#9f000a}:host .gds-form-item__footer .form-info--error .error-icon{margin-top:.128rem;align-items:center}:host .dropdown{width:100%;position:relative}:host .dropdown nggv-dropdown-list{width:100%}\n"], dependencies: [{ kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i4.NggCoreElementDirective, selector: "[nggCoreElement]" }, { kind: "directive", type: i5.TranslocoDirective, selector: "[transloco]", inputs: ["transloco", "translocoParams", "translocoScope", "translocoRead", "translocoPrefix", "translocoLang", "translocoLoadingTpl"] }, { kind: "component", type: i6.NggvDropdownListComponent, selector: "nggv-dropdown-list", inputs: ["expanded", "state", "scrollOffset", "optionContentTpl", "groupLabelTpl", "id", "thook", "options", "textToHighlight", "onlyEmitDistinctChanges"], outputs: ["selectedValueChanged", "closed"] }] }); }
|
|
198
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: NggvDropdownComponent, selector: "nggv-dropdown", inputs: { thook: "thook", placeholder: "placeholder", ariaLabel: "ariaLabel", options: "options", scrollOffset: "scrollOffset", allowControlNullishOption: "allowControlNullishOption", textToHighlight: "textToHighlight", selectOnSingleOption: "selectOnSingleOption" }, outputs: { expandedChange: "expandedChange" }, host: { listeners: { "keyup": "onKeyUp($event)" }, properties: { "attr.data-thook": "this.thook" } }, queries: [{ propertyName: "selectedContentTpl", first: true, predicate: ["selectedTpl"], descendants: true, read: TemplateRef }, { propertyName: "optionContentTpl", first: true, predicate: ["optionTpl"], descendants: true, read: TemplateRef }, { propertyName: "groupLabelTpl", first: true, predicate: ["groupLabelTpl"], descendants: true, read: TemplateRef }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<!-- LABEL -->\n<ng-container *transloco=\"let t; read: scope\">\n <label\n [id]=\"id + '-label'\"\n class=\"gds-field-label hide-if-empty\"\n [attr.for]=\"id + '-toggle'\"\n >\n <ng-template\n *ngTemplateOutlet=\"labelContentTpl || basicLabelContentTpl\"\n ></ng-template>\n <ng-template #basicLabelContentTpl>\n <!-- to trigger css:empty if no label was added -->\n <ng-container *ngIf=\"label\">\n {{ label }}\n <span\n *ngIf=\"optional === true || (required !== true && optional !== false)\"\n class=\"gds-field-label--optional\"\n >\n ({{ t('label.optional') }})\n </span>\n </ng-container>\n </ng-template>\n </label>\n\n <!-- DESCRIPTION -->\n <div class=\"description\">{{ description }}</div>\n\n <!-- LOCKED INPUT -->\n <ng-container *ngIf=\"locked\">\n <div\n [id]=\"id + '-input'\"\n class=\"nggv-field--locked\"\n [attr.name]=\"name\"\n [attr.value]=\"state\"\n [attr.role]=\"role\"\n [attr.aria-labelledby]=\"id + '-label ' + id + '-input'\"\n >\n <span *ngIf=\"!state\" class=\"unset-state\">-</span>\n <ng-container *ngIf=\"state\">\n <ng-template\n *ngTemplateOutlet=\"\n selectedContentTpl || defaultSelectedContentTpl;\n context: { $implicit: state }\n \"\n >\n </ng-template>\n </ng-container>\n </div>\n </ng-container>\n\n <!-- INPUT -->\n <ng-container *ngIf=\"!locked\">\n <div class=\"gds-input-wrapper\">\n <div #input [id]=\"id + '-input'\" class=\"dropdown\">\n <button\n [id]=\"id + '-toggle'\"\n [disabled]=\"disabled\"\n type=\"button\"\n class=\"nggv-field-dropdown__label toggle\"\n [class.nggv-field--error]=\"invalid\"\n aria-haspopup=\"listbox\"\n [attr.data-thook]=\"thook + '-toggle'\"\n [attr.aria-expanded]=\"expanded\"\n [attr.aria-labelledby]=\"\n ariaLabel ? null : id + '-label ' + id + '-toggle'\n \"\n [attr.aria-label]=\"ariaLabel || null\"\n (click)=\"toggleDropdown()\"\n >\n <span>\n <ng-template\n *ngTemplateOutlet=\"\n selectedContentTpl || defaultSelectedContentTpl;\n context: { $implicit: state }\n \"\n >\n </ng-template>\n </span>\n </button>\n <nggv-dropdown-list\n #dropDownList\n [options]=\"options\"\n [scrollOffset]=\"scrollOffset\"\n [state]=\"state\"\n [expanded]=\"expanded\"\n [optionContentTpl]=\"optionContentTpl\"\n [groupLabelTpl]=\"groupLabelTpl\"\n [textToHighlight]=\"textToHighlight\"\n (closed)=\"setExpanded(false)\"\n (selectedValueChanged)=\"onSelectChange($event)\"\n >\n </nggv-dropdown-list>\n </div>\n <!-- ERRORS -->\n <div class=\"gds-form-item__footer error-wrapper\">\n <span\n class=\"form-info form-info--error\"\n [attr.for]=\"id + '-input'\"\n *ngIf=\"invalid && (error || ngControl?.invalid)\"\n >\n <span class=\"error-icon\">\n <gds-icon-triangle-exclamation\n width=\"16\"\n height=\"16\"\n [solid]=\"true\"\n *nggCoreElement\n ></gds-icon-triangle-exclamation>\n </span>\n <span\n *ngIf=\"error; else errorsRef\"\n [attr.data-thook]=\"thook + '-errorlabel'\"\n >{{ error }}</span\n >\n </span>\n <ng-template #errorsRef>\n <span\n *ngIf=\"firstError as error\"\n [attr.data-thook]=\"thook + '-errorlabel'\"\n >\n {{ t('error.field' + error?.code, error?.params) }}\n </span>\n </ng-template>\n </div>\n <!-- CHILDREN -->\n <ng-content></ng-content>\n </div>\n </ng-container>\n\n <ng-template #defaultSelectedContentTpl let-state>\n <!-- eslint-disable-next-line @angular-eslint/template/eqeqeq -->\n {{ state?.key != null && state?.label ? t(state.label) : placeholder }}\n </ng-template>\n</ng-container>\n", styles: [":host{--sg-border-radius: .25rem;--sg-border-width: 1px;--sg-border-color: #868686;--text-primary-color: #333;--sg-form-control-bg: #fff;--grey-000: hsl(0 0% 100%);--grey-100: hsl(0, 0%, 97%);--grey-200: hsl(0, 0%, 91%);--grey-300: hsl(0, 0%, 87%);--grey-400: hsl(0, 0%, 81%);--grey-500: hsl(0, 0%, 68%);--grey-600: hsl(0, 0%, 53%);--grey-700: hsl(0, 0%, 29%);--grey-800: hsl(0, 0%, 20%);--grey-900: hsl(0, 0%, 10%);--grey-1000: hsl(0 0% 0%);display:flex;flex-direction:column;max-width:100%;position:relative;width:100%;z-index:0}:host label{display:block;font-weight:500;line-height:1.25rem;width:100%}:host .description{font-size:.875rem;margin-bottom:.5rem;line-height:1.25rem;width:100%}:host:not(:last-child){margin-bottom:1.5rem}:host .gds-form-item__header{display:flex}:host .gds-form-item__header .form-info{font-weight:400}:host .gds-form-item__header button.icon.small{margin-top:-.5rem;margin-right:-.5rem}:host .gds-form-item__labels{flex:1;margin-bottom:.5rem}:host .gds-form-item__labels .form-info{margin-bottom:0}:host .gds-form-item__labels .form-info a:link:not(.button,[aria-disabled]){color:#0062bc}:host .gds-form-item__labels>*{width:100%;display:block}:host .gds-form-item__expandable-info{overflow:hidden;font-size:.875rem;line-height:1.25rem;transition:height .3s cubic-bezier(.23,1,.32,1)}:host .gds-form-item__expandable-info>div{padding-bottom:.5rem}:host .gds-form-item__backdrop{position:absolute;inset:0;background:var(--gds-ref-pallet-base100);border-radius:2px;z-index:-1;margin:-1rem;opacity:0;transition:all .3s cubic-bezier(.23,1,.32,1);border:1px solid transparent}@media (prefers-reduced-motion: reduce){:host .gds-form-item__backdrop{transition:none}}:host:has([aria-expanded=true]) .gds-form-item__backdrop{opacity:1;border-radius:.25rem;border-color:var(--gds-ref-pallet-base600)}:host .gds-form-item__footer:not(:empty){margin-top:.5rem;display:flex;column-gap:.5rem}:host .gds-form-item__footer:not(:empty)>span,:host .gds-form-item__footer:not(:empty)>.form-info{font-weight:500;line-height:1.125}:host .gds-field-label--optional{font-weight:400}:host button{background-color:transparent;border:0;cursor:pointer;font-family:inherit;padding:0;padding:.75rem 1rem;border-radius:var(--sg-border-radius);border:solid var(--sg-border-width) var(--sg-border-color);--border-color: var(--grey-600);--sg-border-color: var(--grey-600);background:var(--sg-form-control-bg);color:var(--text-primary-color);min-height:2.75rem;display:flex;flex-wrap:nowrap;justify-content:space-between;align-items:center;max-width:100%;font-size:1rem;font-weight:400;line-height:1.125;text-align:left;width:100%}:host button:focus{outline-color:var(--gds-sys-color-focus-outline);outline-style:solid;outline-width:.125rem;outline-offset:.125rem}@media (max-width: 35.98em){:host button{min-width:100%}}:host button>span{white-space:nowrap;overflow:hidden;text-overflow:ellipsis}:host button:after{margin-left:.5rem;margin-right:.5rem;border-bottom:solid 2px var(--text-primary-color);border-left:solid 2px var(--text-primary-color);content:\"\";display:block;height:.5rem;width:.5rem;position:relative;top:-.15rem;transform:translate(75%) rotate3d(0,0,1,-45deg) scaleZ(-1);transition:transform .3s ease-in;flex-shrink:0}:host button[aria-expanded=true]:after{transform:translate(75%,6px) rotate3d(0,0,1,-45deg) scale3d(-1,-1,1)}:host button:hover:after{border-color:currentColor}:host button:disabled{--background: var(--grey-500)}:host button span{width:100%;white-space:nowrap;display:block;text-overflow:ellipsis}:host button.small{font-size:.875rem}:host button:hover{background:#e7e7e7}:host button:active{background:inherit;color:inherit;border-color:inherit}:host button:disabled,:host button.disabled{--text-primary-color: var(--grey-600);background:var(--grey-300);color:var(--grey-600);cursor:not-allowed}:host button.nggv-field--error{border-bottom:2px solid #9f000a}:host .gds-form-item__footer .form-info{font-weight:500;font-size:.875rem}:host .gds-form-item__footer .form-info--error{display:flex;align-items:flex-start;gap:.5em;color:#9f000a}:host .gds-form-item__footer .form-info--error .error-icon{margin-top:.128rem;align-items:center}:host .dropdown{width:100%;position:relative}:host .dropdown nggv-dropdown-list{width:100%}\n"], dependencies: [{ kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i4.NggCoreElementDirective, selector: "[nggCoreElement]" }, { kind: "directive", type: i5.TranslocoDirective, selector: "[transloco]", inputs: ["transloco", "translocoParams", "translocoScope", "translocoRead", "translocoPrefix", "translocoLang", "translocoLoadingTpl"] }, { kind: "component", type: i6.NggvDropdownListComponent, selector: "nggv-dropdown-list", inputs: ["expanded", "state", "scrollOffset", "optionContentTpl", "groupLabelTpl", "id", "thook", "options", "textToHighlight", "onlyEmitDistinctChanges"], outputs: ["selectedValueChanged", "closed"] }] }); }
|
|
199
199
|
}
|
|
200
200
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: NggvDropdownComponent, decorators: [{
|
|
201
201
|
type: Component,
|
|
202
|
-
args: [{ selector: 'nggv-dropdown', template: "<!-- LABEL -->\n<ng-container *transloco=\"let t; read: scope\">\n <label\n [id]=\"id + '-label'\"\n class=\"gds-field-label hide-if-empty\"\n [attr.for]=\"id + '-toggle'\"\n >\n <ng-template\n *ngTemplateOutlet=\"labelContentTpl || basicLabelContentTpl\"\n ></ng-template>\n <ng-template #basicLabelContentTpl>\n <!-- to trigger css:empty if no label was added -->\n <ng-container *ngIf=\"label\">\n {{ label }}\n <span\n *ngIf=\"optional === true || (required !== true && optional !== false)\"\n class=\"gds-field-label--optional\"\n >\n ({{ t('label.optional') }})\n </span>\n </ng-container>\n </ng-template>\n </label>\n\n <!-- DESCRIPTION -->\n <div class=\"description\">{{ description }}</div>\n\n <!-- LOCKED INPUT -->\n <ng-container *ngIf=\"locked\">\n <div\n [id]=\"id + '-input'\"\n class=\"nggv-field--locked\"\n [attr.name]=\"name\"\n [attr.value]=\"state\"\n [attr.role]=\"role\"\n [attr.aria-labelledby]=\"id + '-label ' + id + '-input'\"\n >\n <span *ngIf=\"!state\" class=\"unset-state\">-</span>\n <ng-container *ngIf=\"state\">\n <ng-template\n *ngTemplateOutlet=\"\n selectedContentTpl || defaultSelectedContentTpl;\n context: { $implicit: state }\n \"\n >\n </ng-template>\n </ng-container>\n </div>\n </ng-container>\n\n <!-- INPUT -->\n <ng-container *ngIf=\"!locked\">\n <div #input [id]=\"id + '-input'\" class=\"dropdown\">\n
|
|
202
|
+
args: [{ selector: 'nggv-dropdown', template: "<!-- LABEL -->\n<ng-container *transloco=\"let t; read: scope\">\n <label\n [id]=\"id + '-label'\"\n class=\"gds-field-label hide-if-empty\"\n [attr.for]=\"id + '-toggle'\"\n >\n <ng-template\n *ngTemplateOutlet=\"labelContentTpl || basicLabelContentTpl\"\n ></ng-template>\n <ng-template #basicLabelContentTpl>\n <!-- to trigger css:empty if no label was added -->\n <ng-container *ngIf=\"label\">\n {{ label }}\n <span\n *ngIf=\"optional === true || (required !== true && optional !== false)\"\n class=\"gds-field-label--optional\"\n >\n ({{ t('label.optional') }})\n </span>\n </ng-container>\n </ng-template>\n </label>\n\n <!-- DESCRIPTION -->\n <div class=\"description\">{{ description }}</div>\n\n <!-- LOCKED INPUT -->\n <ng-container *ngIf=\"locked\">\n <div\n [id]=\"id + '-input'\"\n class=\"nggv-field--locked\"\n [attr.name]=\"name\"\n [attr.value]=\"state\"\n [attr.role]=\"role\"\n [attr.aria-labelledby]=\"id + '-label ' + id + '-input'\"\n >\n <span *ngIf=\"!state\" class=\"unset-state\">-</span>\n <ng-container *ngIf=\"state\">\n <ng-template\n *ngTemplateOutlet=\"\n selectedContentTpl || defaultSelectedContentTpl;\n context: { $implicit: state }\n \"\n >\n </ng-template>\n </ng-container>\n </div>\n </ng-container>\n\n <!-- INPUT -->\n <ng-container *ngIf=\"!locked\">\n <div class=\"gds-input-wrapper\">\n <div #input [id]=\"id + '-input'\" class=\"dropdown\">\n <button\n [id]=\"id + '-toggle'\"\n [disabled]=\"disabled\"\n type=\"button\"\n class=\"nggv-field-dropdown__label toggle\"\n [class.nggv-field--error]=\"invalid\"\n aria-haspopup=\"listbox\"\n [attr.data-thook]=\"thook + '-toggle'\"\n [attr.aria-expanded]=\"expanded\"\n [attr.aria-labelledby]=\"\n ariaLabel ? null : id + '-label ' + id + '-toggle'\n \"\n [attr.aria-label]=\"ariaLabel || null\"\n (click)=\"toggleDropdown()\"\n >\n <span>\n <ng-template\n *ngTemplateOutlet=\"\n selectedContentTpl || defaultSelectedContentTpl;\n context: { $implicit: state }\n \"\n >\n </ng-template>\n </span>\n </button>\n <nggv-dropdown-list\n #dropDownList\n [options]=\"options\"\n [scrollOffset]=\"scrollOffset\"\n [state]=\"state\"\n [expanded]=\"expanded\"\n [optionContentTpl]=\"optionContentTpl\"\n [groupLabelTpl]=\"groupLabelTpl\"\n [textToHighlight]=\"textToHighlight\"\n (closed)=\"setExpanded(false)\"\n (selectedValueChanged)=\"onSelectChange($event)\"\n >\n </nggv-dropdown-list>\n </div>\n <!-- ERRORS -->\n <div class=\"gds-form-item__footer error-wrapper\">\n <span\n class=\"form-info form-info--error\"\n [attr.for]=\"id + '-input'\"\n *ngIf=\"invalid && (error || ngControl?.invalid)\"\n >\n <span class=\"error-icon\">\n <gds-icon-triangle-exclamation\n width=\"16\"\n height=\"16\"\n [solid]=\"true\"\n *nggCoreElement\n ></gds-icon-triangle-exclamation>\n </span>\n <span\n *ngIf=\"error; else errorsRef\"\n [attr.data-thook]=\"thook + '-errorlabel'\"\n >{{ error }}</span\n >\n </span>\n <ng-template #errorsRef>\n <span\n *ngIf=\"firstError as error\"\n [attr.data-thook]=\"thook + '-errorlabel'\"\n >\n {{ t('error.field' + error?.code, error?.params) }}\n </span>\n </ng-template>\n </div>\n <!-- CHILDREN -->\n <ng-content></ng-content>\n </div>\n </ng-container>\n\n <ng-template #defaultSelectedContentTpl let-state>\n <!-- eslint-disable-next-line @angular-eslint/template/eqeqeq -->\n {{ state?.key != null && state?.label ? t(state.label) : placeholder }}\n </ng-template>\n</ng-container>\n", styles: [":host{--sg-border-radius: .25rem;--sg-border-width: 1px;--sg-border-color: #868686;--text-primary-color: #333;--sg-form-control-bg: #fff;--grey-000: hsl(0 0% 100%);--grey-100: hsl(0, 0%, 97%);--grey-200: hsl(0, 0%, 91%);--grey-300: hsl(0, 0%, 87%);--grey-400: hsl(0, 0%, 81%);--grey-500: hsl(0, 0%, 68%);--grey-600: hsl(0, 0%, 53%);--grey-700: hsl(0, 0%, 29%);--grey-800: hsl(0, 0%, 20%);--grey-900: hsl(0, 0%, 10%);--grey-1000: hsl(0 0% 0%);display:flex;flex-direction:column;max-width:100%;position:relative;width:100%;z-index:0}:host label{display:block;font-weight:500;line-height:1.25rem;width:100%}:host .description{font-size:.875rem;margin-bottom:.5rem;line-height:1.25rem;width:100%}:host:not(:last-child){margin-bottom:1.5rem}:host .gds-form-item__header{display:flex}:host .gds-form-item__header .form-info{font-weight:400}:host .gds-form-item__header button.icon.small{margin-top:-.5rem;margin-right:-.5rem}:host .gds-form-item__labels{flex:1;margin-bottom:.5rem}:host .gds-form-item__labels .form-info{margin-bottom:0}:host .gds-form-item__labels .form-info a:link:not(.button,[aria-disabled]){color:#0062bc}:host .gds-form-item__labels>*{width:100%;display:block}:host .gds-form-item__expandable-info{overflow:hidden;font-size:.875rem;line-height:1.25rem;transition:height .3s cubic-bezier(.23,1,.32,1)}:host .gds-form-item__expandable-info>div{padding-bottom:.5rem}:host .gds-form-item__backdrop{position:absolute;inset:0;background:var(--gds-ref-pallet-base100);border-radius:2px;z-index:-1;margin:-1rem;opacity:0;transition:all .3s cubic-bezier(.23,1,.32,1);border:1px solid transparent}@media (prefers-reduced-motion: reduce){:host .gds-form-item__backdrop{transition:none}}:host:has([aria-expanded=true]) .gds-form-item__backdrop{opacity:1;border-radius:.25rem;border-color:var(--gds-ref-pallet-base600)}:host .gds-form-item__footer:not(:empty){margin-top:.5rem;display:flex;column-gap:.5rem}:host .gds-form-item__footer:not(:empty)>span,:host .gds-form-item__footer:not(:empty)>.form-info{font-weight:500;line-height:1.125}:host .gds-field-label--optional{font-weight:400}:host button{background-color:transparent;border:0;cursor:pointer;font-family:inherit;padding:0;padding:.75rem 1rem;border-radius:var(--sg-border-radius);border:solid var(--sg-border-width) var(--sg-border-color);--border-color: var(--grey-600);--sg-border-color: var(--grey-600);background:var(--sg-form-control-bg);color:var(--text-primary-color);min-height:2.75rem;display:flex;flex-wrap:nowrap;justify-content:space-between;align-items:center;max-width:100%;font-size:1rem;font-weight:400;line-height:1.125;text-align:left;width:100%}:host button:focus{outline-color:var(--gds-sys-color-focus-outline);outline-style:solid;outline-width:.125rem;outline-offset:.125rem}@media (max-width: 35.98em){:host button{min-width:100%}}:host button>span{white-space:nowrap;overflow:hidden;text-overflow:ellipsis}:host button:after{margin-left:.5rem;margin-right:.5rem;border-bottom:solid 2px var(--text-primary-color);border-left:solid 2px var(--text-primary-color);content:\"\";display:block;height:.5rem;width:.5rem;position:relative;top:-.15rem;transform:translate(75%) rotate3d(0,0,1,-45deg) scaleZ(-1);transition:transform .3s ease-in;flex-shrink:0}:host button[aria-expanded=true]:after{transform:translate(75%,6px) rotate3d(0,0,1,-45deg) scale3d(-1,-1,1)}:host button:hover:after{border-color:currentColor}:host button:disabled{--background: var(--grey-500)}:host button span{width:100%;white-space:nowrap;display:block;text-overflow:ellipsis}:host button.small{font-size:.875rem}:host button:hover{background:#e7e7e7}:host button:active{background:inherit;color:inherit;border-color:inherit}:host button:disabled,:host button.disabled{--text-primary-color: var(--grey-600);background:var(--grey-300);color:var(--grey-600);cursor:not-allowed}:host button.nggv-field--error{border-bottom:2px solid #9f000a}:host .gds-form-item__footer .form-info{font-weight:500;font-size:.875rem}:host .gds-form-item__footer .form-info--error{display:flex;align-items:flex-start;gap:.5em;color:#9f000a}:host .gds-form-item__footer .form-info--error .error-icon{margin-top:.128rem;align-items:center}:host .dropdown{width:100%;position:relative}:host .dropdown nggv-dropdown-list{width:100%}\n"] }]
|
|
203
203
|
}], ctorParameters: () => [{ type: i1.NgControl, decorators: [{
|
|
204
204
|
type: Self
|
|
205
205
|
}, {
|
|
@@ -243,4 +243,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
243
243
|
type: HostListener,
|
|
244
244
|
args: ['keyup', ['$event']]
|
|
245
245
|
}] } });
|
|
246
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dropdown.component.js","sourceRoot":"","sources":["../../../../../../../libs/angular/src/v-angular/dropdown/dropdown.component.ts","../../../../../../../libs/angular/src/v-angular/dropdown/dropdown.component.html"],"names":[],"mappings":"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","sourcesContent":["import '@sebgroup/green-core/components/icon/icons/triangle-exclamation.js'\n\nimport {\n  ChangeDetectorRef,\n  Component,\n  ContentChild,\n  EventEmitter,\n  HostBinding,\n  HostListener,\n  Inject,\n  Input,\n  OnChanges,\n  OnDestroy,\n  Optional,\n  Output,\n  Self,\n  SimpleChanges,\n  TemplateRef,\n} from '@angular/core'\nimport { NgControl } from '@angular/forms'\nimport { TRANSLOCO_SCOPE, TranslocoScope } from '@jsverse/transloco'\nimport { fromEvent, Subscription } from 'rxjs'\n\nimport { NggvBaseControlValueAccessorComponent } from '@sebgroup/green-angular/src/v-angular/base-control-value-accessor'\nimport {\n  DropdownUtils,\n  Option,\n  OptionBase,\n  OptionGroup,\n} from '@sebgroup/green-angular/src/v-angular/core'\n\n/**\n * A dropdown allows the user to select an option from a list.\n * Dropdowns enables users to make a quick selection of the available options for a specific entry.\n * https://designlibrary.sebgroup.com/components/component-dropdown\n */\n@Component({\n  selector: 'nggv-dropdown',\n  templateUrl: './dropdown.component.html',\n  styleUrls: ['./dropdown.component.scss'],\n})\nexport class NggvDropdownComponent<\n    K = string | null | undefined,\n    V = string | null | undefined,\n    T extends Option<K, V> = Option<K, V>,\n  >\n  extends NggvBaseControlValueAccessorComponent\n  implements OnDestroy, OnChanges\n{\n  /** Custom template for displaying options and groups. */\n  @ContentChild('selectedTpl', { read: TemplateRef }) selectedContentTpl:\n    | TemplateRef<OptionBase<T>>\n    | undefined\n  @ContentChild('optionTpl', { read: TemplateRef }) optionContentTpl:\n    | TemplateRef<OptionBase<any>>\n    | undefined\n  @ContentChild('groupLabelTpl', { read: TemplateRef }) groupLabelTpl:\n    | TemplateRef<OptionBase<any>>\n    | undefined\n\n  /** Special property used for selecting DOM elements during automated UI testing. */\n  @HostBinding('attr.data-thook') @Input() thook: string | null | undefined =\n    'dropdown'\n\n  /** Text shown before an option is selected. */\n  @Input() placeholder?: string\n  /** Specific value for aria-label. If not provided, label/labelTpl will be used. */\n  @Input() ariaLabel?: string\n  /** List of {@link Option} and {@link OptionGroup} listed when dropdown is expanded. */\n  @Input() set options(value: OptionBase<T>[]) {\n    // update options\n    this._options = value\n    // already has a null/undefined key\n    const nullishOption = this.dropdownUtils\n      .flattenOptions(value, false)\n      .find((option) => option.key == null)\n    // if the dropdown is optional, add a null value to deselect option\n    if (!this.required && !nullishOption && this.allowControlNullishOption) {\n      this._options = [this.defaultNullishOption].concat(this._options)\n    }\n\n    // set default value and emit if there is only one option\n    if (value.length === 1 && this.required && this.selectOnSingleOption) {\n      const onlyOption = this.dropdownUtils.flattenOptions(value, false)[0]\n      this.updateModel(onlyOption)\n      return\n    }\n\n    // don't update local state if we shouldn't control nullish value and we cant find the selected value amongst the options\n    const matchingOption = this.dropdownUtils\n      .flattenOptions(value, false)\n      .find((option) => option.key == this.ngControl?.value)\n    if (!this.allowControlNullishOption && !matchingOption) return\n\n    // Update local state\n    this.writeValue(this.ngControl?.value)\n  }\n\n  get options(): OptionBase<T>[] {\n    return this._options\n  }\n\n  /** The additional amount to show when option is scrolled into view. */\n  @Input() scrollOffset = 24\n\n  /**\n   * Allow this component to add/ remove nullish options based on wether the control is required or not\n   * Defaults to true.\n   */\n  @Input() allowControlNullishOption = true\n\n  /** Text to highlight in option */\n  @Input() textToHighlight?: string\n\n  /**\n   * If only one option exists in options[], default is to select it.\n   * This input can be used to alter that behaviour so it doesn't automatically\n   * select a value if it's the only one.\n   * Defaults to true.\n   */\n  @Input() selectOnSingleOption = true\n\n  /**\n   * Emits changes of the expanded state of the dropdown\n   */\n  @Output() expandedChange = new EventEmitter<boolean>()\n\n  /** @internal nullish option. */\n  get defaultNullishOption(): OptionBase<any> {\n    return { key: null, label: this.placeholder ?? '\\u00A0' }\n  }\n\n  /** The current expanded state of the dropdown options. */\n  public expanded = false\n  /** The current option selected based on numeric index. */\n  public activeIndex = -1\n  /** Subscribe if dropdown expanded to listen to click outside to close dropdown. */\n  private onClickSubscription: Subscription | undefined\n\n  public keyEvent: KeyboardEvent = {} as KeyboardEvent\n  private _options: OptionBase<T>[] = []\n\n  constructor(\n    @Self() @Optional() public ngControl: NgControl,\n    @Optional()\n    @Inject(TRANSLOCO_SCOPE)\n    protected translocoScope: TranslocoScope,\n    protected cdr: ChangeDetectorRef,\n    protected dropdownUtils: DropdownUtils<K, V, T>,\n  ) {\n    super(ngControl, translocoScope, cdr)\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.required?.currentValue !== undefined) {\n      const isRequired = changes.required.currentValue\n      // remove nullish option\n      const hasNullishOption =\n        this.dropdownUtils.flattenOptions(this._options, false)[0]?.key == null\n      // if required, remove nullish option\n      if (isRequired && hasNullishOption && this.allowControlNullishOption) {\n        this._options = this._options.slice(1)\n        return\n      }\n      // if not required, add nullish option\n      if (!isRequired && !hasNullishOption && this.allowControlNullishOption) {\n        this._options = [this.defaultNullishOption].concat(this._options)\n        return\n      }\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.onClickSubscription?.unsubscribe()\n  }\n\n  /** @internal override to correctly set state from form value */\n  writeValue(value: any): void {\n    const options = this.dropdownUtils.flattenOptions(this._options, false)\n    this.state = options.find((option) => option.key === value)\n  }\n\n  // ----------------------------------------------------------------------------\n  // EVENTS\n  // ----------------------------------------------------------------------------\n\n  /** @internal */\n  onSelectChange(option: T) {\n    if (option.disabled) return\n    this.updateModel(option)\n    this.setExpanded(false)\n  }\n\n  // /**\n  //  * @internal\n  //  * Enter toggles the dropdown, home, end, and, arrows change the index.\n  //  * @param event fired containing which key was released.\n  //  */\n  @HostListener('keyup', ['$event'])\n  onKeyUp(event: KeyboardEvent) {\n    this.keyEvent = event\n  }\n\n  /**\n   * Closes the dropdown on click outside.\n   */\n  subscribeToOutsideClickEvent() {\n    this.onClickSubscription = fromEvent(document, 'click').subscribe({\n      next: (event: Event) => {\n        if (\n          this.expanded &&\n          !this.inputRef?.nativeElement.contains(event.target)\n        ) {\n          this.toggleDropdown()\n          this.onClickSubscription?.unsubscribe()\n        }\n      },\n    })\n  }\n\n  // ----------------------------------------------------------------------------\n  // HELPERS\n  // ----------------------------------------------------------------------------\n\n  /**\n   * Set the dropdown value to given option.\n   * @param value the dropdown option to select.\n   */\n  private updateModel(value: T) {\n    this.state = value\n    this.onChange(value.key)\n  }\n\n  /** Toggle the expanded state of the dropdown options. */\n  toggleDropdown() {\n    this.setExpanded(!this.expanded)\n    this.cdr.detectChanges()\n  }\n\n  /**\n   * Set the expanded state of the dropdown options. If true the options are shown below the field.\n   * Activate on click event to be able to close dropdown on click outside.\n   * @param state the expanded state which to set.\n   */\n  setExpanded(state = true) {\n    this.expanded = state\n    this.expandedChange.emit(this.expanded)\n    if (this.expanded) this.subscribeToOutsideClickEvent()\n    if (!this.expanded) this.onTouched()\n  }\n\n  /* TYPE CASTS */\n\n  /**\n   * Typecast anything to an {@link Option}.\n   * @param option the object to typecast.\n   */\n  castOption(option: any): T {\n    return option\n  }\n\n  /**\n   * Typecast anything to an {@link OptionGroup}.\n   * @param group the object to typecast.\n   */\n  castGroup(group: any): OptionGroup<T> {\n    return group\n  }\n\n  /* TYPE CHECKS */\n\n  /**\n   * Returns true if argument is an {@link Option}.\n   * @param option the object to check.\n   */\n  isOption(option: OptionBase<T>): option is OptionGroup<T> {\n    return !('options' in option)\n  }\n}\n","<!-- LABEL -->\n<ng-container *transloco=\"let t; read: scope\">\n  <label\n    [id]=\"id + '-label'\"\n    class=\"gds-field-label hide-if-empty\"\n    [attr.for]=\"id + '-toggle'\"\n  >\n    <ng-template\n      *ngTemplateOutlet=\"labelContentTpl || basicLabelContentTpl\"\n    ></ng-template>\n    <ng-template #basicLabelContentTpl>\n      <!-- to trigger css:empty if no label was added -->\n      <ng-container *ngIf=\"label\">\n        {{ label }}\n        <span\n          *ngIf=\"optional === true || (required !== true && optional !== false)\"\n          class=\"gds-field-label--optional\"\n        >\n          ({{ t('label.optional') }})\n        </span>\n      </ng-container>\n    </ng-template>\n  </label>\n\n  <!-- DESCRIPTION -->\n  <div class=\"description\">{{ description }}</div>\n\n  <!-- LOCKED INPUT -->\n  <ng-container *ngIf=\"locked\">\n    <div\n      [id]=\"id + '-input'\"\n      class=\"nggv-field--locked\"\n      [attr.name]=\"name\"\n      [attr.value]=\"state\"\n      [attr.role]=\"role\"\n      [attr.aria-labelledby]=\"id + '-label ' + id + '-input'\"\n    >\n      <span *ngIf=\"!state\" class=\"unset-state\">-</span>\n      <ng-container *ngIf=\"state\">\n        <ng-template\n          *ngTemplateOutlet=\"\n            selectedContentTpl || defaultSelectedContentTpl;\n            context: { $implicit: state }\n          \"\n        >\n        </ng-template>\n      </ng-container>\n    </div>\n  </ng-container>\n\n  <!-- INPUT -->\n  <ng-container *ngIf=\"!locked\">\n    <div #input [id]=\"id + '-input'\" class=\"dropdown\">\n      <button\n        [id]=\"id + '-toggle'\"\n        [disabled]=\"disabled\"\n        type=\"button\"\n        class=\"nggv-field-dropdown__label toggle\"\n        [class.nggv-field--error]=\"invalid\"\n        aria-haspopup=\"listbox\"\n        [attr.data-thook]=\"thook + '-toggle'\"\n        [attr.aria-expanded]=\"expanded\"\n        [attr.aria-labelledby]=\"\n          ariaLabel ? null : id + '-label ' + id + '-toggle'\n        \"\n        [attr.aria-label]=\"ariaLabel || null\"\n        (click)=\"toggleDropdown()\"\n      >\n        <ng-template\n          *ngTemplateOutlet=\"\n            selectedContentTpl || defaultSelectedContentTpl;\n            context: { $implicit: state }\n          \"\n        >\n        </ng-template>\n      </button>\n\n      <nggv-dropdown-list\n        #dropDownList\n        [options]=\"options\"\n        [scrollOffset]=\"scrollOffset\"\n        [state]=\"state\"\n        [expanded]=\"expanded\"\n        [optionContentTpl]=\"optionContentTpl\"\n        [groupLabelTpl]=\"groupLabelTpl\"\n        [textToHighlight]=\"textToHighlight\"\n        (closed)=\"setExpanded(false)\"\n        (selectedValueChanged)=\"onSelectChange($event)\"\n      >\n      </nggv-dropdown-list>\n    </div>\n    <!-- ERRORS -->\n    <div class=\"gds-form-item__footer error-wrapper\">\n      <span\n        class=\"form-info form-info--error\"\n        [attr.for]=\"id + '-input'\"\n        *ngIf=\"invalid && (error || ngControl?.invalid)\"\n      >\n        <span class=\"error-icon\">\n          <gds-icon-triangle-exclamation\n            width=\"16\"\n            height=\"16\"\n            [solid]=\"true\"\n            *nggCoreElement\n          ></gds-icon-triangle-exclamation>\n        </span>\n        <span\n          *ngIf=\"error; else errorsRef\"\n          [attr.data-thook]=\"thook + '-errorlabel'\"\n          >{{ error }}</span\n        >\n      </span>\n      <ng-template #errorsRef>\n        <span\n          *ngIf=\"firstError as error\"\n          [attr.data-thook]=\"thook + '-errorlabel'\"\n        >\n          {{ t('error.field' + error?.code, error?.params) }}\n        </span>\n      </ng-template>\n    </div>\n\n    <!-- CHILDREN -->\n    <ng-content></ng-content>\n  </ng-container>\n\n  <ng-template #defaultSelectedContentTpl let-state>\n    <!-- eslint-disable-next-line @angular-eslint/template/eqeqeq -->\n    {{ state?.key != null && state?.label ? t(state.label) : placeholder }}\n  </ng-template>\n</ng-container>\n"]}
|
|
246
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dropdown.component.js","sourceRoot":"","sources":["../../../../../../../libs/angular/src/v-angular/dropdown/dropdown.component.ts","../../../../../../../libs/angular/src/v-angular/dropdown/dropdown.component.html"],"names":[],"mappings":"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","sourcesContent":["import '@sebgroup/green-core/components/icon/icons/triangle-exclamation.js'\n\nimport {\n  ChangeDetectorRef,\n  Component,\n  ContentChild,\n  EventEmitter,\n  HostBinding,\n  HostListener,\n  Inject,\n  Input,\n  OnChanges,\n  OnDestroy,\n  Optional,\n  Output,\n  Self,\n  SimpleChanges,\n  TemplateRef,\n} from '@angular/core'\nimport { NgControl } from '@angular/forms'\nimport { TRANSLOCO_SCOPE, TranslocoScope } from '@jsverse/transloco'\nimport { fromEvent, Subscription } from 'rxjs'\n\nimport { NggvBaseControlValueAccessorComponent } from '@sebgroup/green-angular/src/v-angular/base-control-value-accessor'\nimport {\n  DropdownUtils,\n  Option,\n  OptionBase,\n  OptionGroup,\n} from '@sebgroup/green-angular/src/v-angular/core'\n\n/**\n * A dropdown allows the user to select an option from a list.\n * Dropdowns enables users to make a quick selection of the available options for a specific entry.\n * https://designlibrary.sebgroup.com/components/component-dropdown\n */\n@Component({\n  selector: 'nggv-dropdown',\n  templateUrl: './dropdown.component.html',\n  styleUrls: ['./dropdown.component.scss'],\n})\nexport class NggvDropdownComponent<\n    K = string | null | undefined,\n    V = string | null | undefined,\n    T extends Option<K, V> = Option<K, V>,\n  >\n  extends NggvBaseControlValueAccessorComponent\n  implements OnDestroy, OnChanges\n{\n  /** Custom template for displaying options and groups. */\n  @ContentChild('selectedTpl', { read: TemplateRef }) selectedContentTpl:\n    | TemplateRef<OptionBase<T>>\n    | undefined\n  @ContentChild('optionTpl', { read: TemplateRef }) optionContentTpl:\n    | TemplateRef<OptionBase<any>>\n    | undefined\n  @ContentChild('groupLabelTpl', { read: TemplateRef }) groupLabelTpl:\n    | TemplateRef<OptionBase<any>>\n    | undefined\n\n  /** Special property used for selecting DOM elements during automated UI testing. */\n  @HostBinding('attr.data-thook') @Input() thook: string | null | undefined =\n    'dropdown'\n\n  /** Text shown before an option is selected. */\n  @Input() placeholder?: string\n  /** Specific value for aria-label. If not provided, label/labelTpl will be used. */\n  @Input() ariaLabel?: string\n  /** List of {@link Option} and {@link OptionGroup} listed when dropdown is expanded. */\n  @Input() set options(value: OptionBase<T>[]) {\n    // update options\n    this._options = value\n    // already has a null/undefined key\n    const nullishOption = this.dropdownUtils\n      .flattenOptions(value, false)\n      .find((option) => option.key == null)\n    // if the dropdown is optional, add a null value to deselect option\n    if (!this.required && !nullishOption && this.allowControlNullishOption) {\n      this._options = [this.defaultNullishOption].concat(this._options)\n    }\n\n    // set default value and emit if there is only one option\n    if (value.length === 1 && this.required && this.selectOnSingleOption) {\n      const onlyOption = this.dropdownUtils.flattenOptions(value, false)[0]\n      this.updateModel(onlyOption)\n      return\n    }\n\n    // don't update local state if we shouldn't control nullish value and we cant find the selected value amongst the options\n    const matchingOption = this.dropdownUtils\n      .flattenOptions(value, false)\n      .find((option) => option.key == this.ngControl?.value)\n    if (!this.allowControlNullishOption && !matchingOption) return\n\n    // Update local state\n    this.writeValue(this.ngControl?.value)\n  }\n\n  get options(): OptionBase<T>[] {\n    return this._options\n  }\n\n  /** The additional amount to show when option is scrolled into view. */\n  @Input() scrollOffset = 24\n\n  /**\n   * Allow this component to add/ remove nullish options based on wether the control is required or not\n   * Defaults to true.\n   */\n  @Input() allowControlNullishOption = true\n\n  /** Text to highlight in option */\n  @Input() textToHighlight?: string\n\n  /**\n   * If only one option exists in options[], default is to select it.\n   * This input can be used to alter that behaviour so it doesn't automatically\n   * select a value if it's the only one.\n   * Defaults to true.\n   */\n  @Input() selectOnSingleOption = true\n\n  /**\n   * Emits changes of the expanded state of the dropdown\n   */\n  @Output() expandedChange = new EventEmitter<boolean>()\n\n  /** @internal nullish option. */\n  get defaultNullishOption(): OptionBase<any> {\n    return { key: null, label: this.placeholder ?? '\\u00A0' }\n  }\n\n  /** The current expanded state of the dropdown options. */\n  public expanded = false\n  /** The current option selected based on numeric index. */\n  public activeIndex = -1\n  /** Subscribe if dropdown expanded to listen to click outside to close dropdown. */\n  private onClickSubscription: Subscription | undefined\n\n  public keyEvent: KeyboardEvent = {} as KeyboardEvent\n  private _options: OptionBase<T>[] = []\n\n  constructor(\n    @Self() @Optional() public ngControl: NgControl,\n    @Optional()\n    @Inject(TRANSLOCO_SCOPE)\n    protected translocoScope: TranslocoScope,\n    protected cdr: ChangeDetectorRef,\n    protected dropdownUtils: DropdownUtils<K, V, T>,\n  ) {\n    super(ngControl, translocoScope, cdr)\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.required?.currentValue !== undefined) {\n      const isRequired = changes.required.currentValue\n      // remove nullish option\n      const hasNullishOption =\n        this.dropdownUtils.flattenOptions(this._options, false)[0]?.key == null\n      // if required, remove nullish option\n      if (isRequired && hasNullishOption && this.allowControlNullishOption) {\n        this._options = this._options.slice(1)\n        return\n      }\n      // if not required, add nullish option\n      if (!isRequired && !hasNullishOption && this.allowControlNullishOption) {\n        this._options = [this.defaultNullishOption].concat(this._options)\n        return\n      }\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.onClickSubscription?.unsubscribe()\n  }\n\n  /** @internal override to correctly set state from form value */\n  writeValue(value: any): void {\n    const options = this.dropdownUtils.flattenOptions(this._options, false)\n    this.state = options.find((option) => option.key === value)\n  }\n\n  // ----------------------------------------------------------------------------\n  // EVENTS\n  // ----------------------------------------------------------------------------\n\n  /** @internal */\n  onSelectChange(option: T) {\n    if (option.disabled) return\n    this.updateModel(option)\n    this.setExpanded(false)\n  }\n\n  // /**\n  //  * @internal\n  //  * Enter toggles the dropdown, home, end, and, arrows change the index.\n  //  * @param event fired containing which key was released.\n  //  */\n  @HostListener('keyup', ['$event'])\n  onKeyUp(event: KeyboardEvent) {\n    this.keyEvent = event\n  }\n\n  /**\n   * Closes the dropdown on click outside.\n   */\n  subscribeToOutsideClickEvent() {\n    this.onClickSubscription = fromEvent(document, 'click').subscribe({\n      next: (event: Event) => {\n        if (\n          this.expanded &&\n          !this.inputRef?.nativeElement.contains(event.target)\n        ) {\n          this.toggleDropdown()\n          this.onClickSubscription?.unsubscribe()\n        }\n      },\n    })\n  }\n\n  // ----------------------------------------------------------------------------\n  // HELPERS\n  // ----------------------------------------------------------------------------\n\n  /**\n   * Set the dropdown value to given option.\n   * @param value the dropdown option to select.\n   */\n  private updateModel(value: T) {\n    this.state = value\n    this.onChange(value.key)\n  }\n\n  /** Toggle the expanded state of the dropdown options. */\n  toggleDropdown() {\n    this.setExpanded(!this.expanded)\n    this.cdr.detectChanges()\n  }\n\n  /**\n   * Set the expanded state of the dropdown options. If true the options are shown below the field.\n   * Activate on click event to be able to close dropdown on click outside.\n   * @param state the expanded state which to set.\n   */\n  setExpanded(state = true) {\n    this.expanded = state\n    this.expandedChange.emit(this.expanded)\n    if (this.expanded) this.subscribeToOutsideClickEvent()\n    if (!this.expanded) this.onTouched()\n  }\n\n  /* TYPE CASTS */\n\n  /**\n   * Typecast anything to an {@link Option}.\n   * @param option the object to typecast.\n   */\n  castOption(option: any): T {\n    return option\n  }\n\n  /**\n   * Typecast anything to an {@link OptionGroup}.\n   * @param group the object to typecast.\n   */\n  castGroup(group: any): OptionGroup<T> {\n    return group\n  }\n\n  /* TYPE CHECKS */\n\n  /**\n   * Returns true if argument is an {@link Option}.\n   * @param option the object to check.\n   */\n  isOption(option: OptionBase<T>): option is OptionGroup<T> {\n    return !('options' in option)\n  }\n}\n","<!-- LABEL -->\n<ng-container *transloco=\"let t; read: scope\">\n  <label\n    [id]=\"id + '-label'\"\n    class=\"gds-field-label hide-if-empty\"\n    [attr.for]=\"id + '-toggle'\"\n  >\n    <ng-template\n      *ngTemplateOutlet=\"labelContentTpl || basicLabelContentTpl\"\n    ></ng-template>\n    <ng-template #basicLabelContentTpl>\n      <!-- to trigger css:empty if no label was added -->\n      <ng-container *ngIf=\"label\">\n        {{ label }}\n        <span\n          *ngIf=\"optional === true || (required !== true && optional !== false)\"\n          class=\"gds-field-label--optional\"\n        >\n          ({{ t('label.optional') }})\n        </span>\n      </ng-container>\n    </ng-template>\n  </label>\n\n  <!-- DESCRIPTION -->\n  <div class=\"description\">{{ description }}</div>\n\n  <!-- LOCKED INPUT -->\n  <ng-container *ngIf=\"locked\">\n    <div\n      [id]=\"id + '-input'\"\n      class=\"nggv-field--locked\"\n      [attr.name]=\"name\"\n      [attr.value]=\"state\"\n      [attr.role]=\"role\"\n      [attr.aria-labelledby]=\"id + '-label ' + id + '-input'\"\n    >\n      <span *ngIf=\"!state\" class=\"unset-state\">-</span>\n      <ng-container *ngIf=\"state\">\n        <ng-template\n          *ngTemplateOutlet=\"\n            selectedContentTpl || defaultSelectedContentTpl;\n            context: { $implicit: state }\n          \"\n        >\n        </ng-template>\n      </ng-container>\n    </div>\n  </ng-container>\n\n  <!-- INPUT -->\n  <ng-container *ngIf=\"!locked\">\n    <div class=\"gds-input-wrapper\">\n      <div #input [id]=\"id + '-input'\" class=\"dropdown\">\n        <button\n          [id]=\"id + '-toggle'\"\n          [disabled]=\"disabled\"\n          type=\"button\"\n          class=\"nggv-field-dropdown__label toggle\"\n          [class.nggv-field--error]=\"invalid\"\n          aria-haspopup=\"listbox\"\n          [attr.data-thook]=\"thook + '-toggle'\"\n          [attr.aria-expanded]=\"expanded\"\n          [attr.aria-labelledby]=\"\n            ariaLabel ? null : id + '-label ' + id + '-toggle'\n          \"\n          [attr.aria-label]=\"ariaLabel || null\"\n          (click)=\"toggleDropdown()\"\n        >\n          <span>\n            <ng-template\n              *ngTemplateOutlet=\"\n                selectedContentTpl || defaultSelectedContentTpl;\n                context: { $implicit: state }\n              \"\n            >\n            </ng-template>\n          </span>\n        </button>\n        <nggv-dropdown-list\n          #dropDownList\n          [options]=\"options\"\n          [scrollOffset]=\"scrollOffset\"\n          [state]=\"state\"\n          [expanded]=\"expanded\"\n          [optionContentTpl]=\"optionContentTpl\"\n          [groupLabelTpl]=\"groupLabelTpl\"\n          [textToHighlight]=\"textToHighlight\"\n          (closed)=\"setExpanded(false)\"\n          (selectedValueChanged)=\"onSelectChange($event)\"\n        >\n        </nggv-dropdown-list>\n      </div>\n      <!-- ERRORS -->\n      <div class=\"gds-form-item__footer error-wrapper\">\n        <span\n          class=\"form-info form-info--error\"\n          [attr.for]=\"id + '-input'\"\n          *ngIf=\"invalid && (error || ngControl?.invalid)\"\n        >\n          <span class=\"error-icon\">\n            <gds-icon-triangle-exclamation\n              width=\"16\"\n              height=\"16\"\n              [solid]=\"true\"\n              *nggCoreElement\n            ></gds-icon-triangle-exclamation>\n          </span>\n          <span\n            *ngIf=\"error; else errorsRef\"\n            [attr.data-thook]=\"thook + '-errorlabel'\"\n            >{{ error }}</span\n          >\n        </span>\n        <ng-template #errorsRef>\n          <span\n            *ngIf=\"firstError as error\"\n            [attr.data-thook]=\"thook + '-errorlabel'\"\n          >\n            {{ t('error.field' + error?.code, error?.params) }}\n          </span>\n        </ng-template>\n      </div>\n      <!-- CHILDREN -->\n      <ng-content></ng-content>\n    </div>\n  </ng-container>\n\n  <ng-template #defaultSelectedContentTpl let-state>\n    <!-- eslint-disable-next-line @angular-eslint/template/eqeqeq -->\n    {{ state?.key != null && state?.label ? t(state.label) : placeholder }}\n  </ng-template>\n</ng-container>\n"]}
|