@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
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
;
|
|
2
|
+
(() => {
|
|
3
|
+
// Make sure there is an incremental ID each component can use
|
|
4
|
+
if (typeof window !== 'undefined' && !window.nggv) {
|
|
5
|
+
window.nggv = {
|
|
6
|
+
ids: { default: -1 },
|
|
7
|
+
nextId(namespace = 'default') {
|
|
8
|
+
let id = this.ids[namespace] || 0;
|
|
9
|
+
if (typeof this.ids[namespace] === 'number')
|
|
10
|
+
id++;
|
|
11
|
+
this.ids[namespace] = id;
|
|
12
|
+
return namespace === 'default'
|
|
13
|
+
? `nggv-${id}`
|
|
14
|
+
: `nggv-${namespace}-${id}`;
|
|
15
|
+
},
|
|
16
|
+
};
|
|
17
|
+
}
|
|
18
|
+
})();
|
|
19
|
+
export {};
|
|
20
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29yZS5nbG9iYWxzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9hbmd1bGFyL3NyYy92LWFuZ3VsYXIvY29yZS9jb3JlLmdsb2JhbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBY0EsQ0FBQztBQUFBLENBQUMsR0FBRyxFQUFFO0lBQ0wsOERBQThEO0lBQzlELElBQUksT0FBTyxNQUFNLEtBQUssV0FBVyxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksRUFBRSxDQUFDO1FBQ2xELE1BQU0sQ0FBQyxJQUFJLEdBQUc7WUFDWixHQUFHLEVBQUUsRUFBRSxPQUFPLEVBQUUsQ0FBQyxDQUFDLEVBQUU7WUFDcEIsTUFBTSxDQUFDLFNBQVMsR0FBRyxTQUFTO2dCQUMxQixJQUFJLEVBQUUsR0FBRyxJQUFJLENBQUMsR0FBRyxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsQ0FBQTtnQkFDakMsSUFBSSxPQUFPLElBQUksQ0FBQyxHQUFHLENBQUMsU0FBUyxDQUFDLEtBQUssUUFBUTtvQkFBRSxFQUFFLEVBQUUsQ0FBQTtnQkFDakQsSUFBSSxDQUFDLEdBQUcsQ0FBQyxTQUFTLENBQUMsR0FBRyxFQUFFLENBQUE7Z0JBQ3hCLE9BQU8sU0FBUyxLQUFLLFNBQVM7b0JBQzVCLENBQUMsQ0FBQyxRQUFRLEVBQUUsRUFBRTtvQkFDZCxDQUFDLENBQUMsUUFBUSxTQUFTLElBQUksRUFBRSxFQUFFLENBQUE7WUFDL0IsQ0FBQztTQUNGLENBQUE7SUFDSCxDQUFDO0FBQ0gsQ0FBQyxDQUFDLEVBQUUsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbIi8vIFdlIG11c3QgZm9yY2UgdHNjIHRvIGludGVycHJldCB0aGlzIGZpbGUgYXMgYSBtb2R1bGUsIHJlc29sdmVzXG4vLyBcIkF1Z21lbnRhdGlvbnMgZm9yIHRoZSBnbG9iYWwgc2NvcGUgY2FuIG9ubHkgYmUgZGlyZWN0bHkgbmVzdGVkIGluIGV4dGVybmFsIG1vZHVsZXMgb3IgYW1iaWVudCBtb2R1bGUgZGVjbGFyYXRpb25zLlwiXG5leHBvcnQge31cblxuZGVjbGFyZSBnbG9iYWwge1xuICBpbnRlcmZhY2UgV2luZG93IHtcbiAgICAvKiogQ291bnRlciBmb3IgdW5pcXVlIGlkZW50aWZpZXJzICovXG4gICAgbmdndjoge1xuICAgICAgaWRzOiB7IFtuYW1lc3BhY2U6IHN0cmluZ106IG51bWJlcjsgZGVmYXVsdDogbnVtYmVyIH1cbiAgICAgIG5leHRJZDogKG5hbWVzcGFjZT86IHN0cmluZykgPT4gc3RyaW5nXG4gICAgfVxuICB9XG59XG5cbjsoKCkgPT4ge1xuICAvLyBNYWtlIHN1cmUgdGhlcmUgaXMgYW4gaW5jcmVtZW50YWwgSUQgZWFjaCBjb21wb25lbnQgY2FuIHVzZVxuICBpZiAodHlwZW9mIHdpbmRvdyAhPT0gJ3VuZGVmaW5lZCcgJiYgIXdpbmRvdy5uZ2d2KSB7XG4gICAgd2luZG93Lm5nZ3YgPSB7XG4gICAgICBpZHM6IHsgZGVmYXVsdDogLTEgfSxcbiAgICAgIG5leHRJZChuYW1lc3BhY2UgPSAnZGVmYXVsdCcpOiBzdHJpbmcge1xuICAgICAgICBsZXQgaWQgPSB0aGlzLmlkc1tuYW1lc3BhY2VdIHx8IDBcbiAgICAgICAgaWYgKHR5cGVvZiB0aGlzLmlkc1tuYW1lc3BhY2VdID09PSAnbnVtYmVyJykgaWQrK1xuICAgICAgICB0aGlzLmlkc1tuYW1lc3BhY2VdID0gaWRcbiAgICAgICAgcmV0dXJuIG5hbWVzcGFjZSA9PT0gJ2RlZmF1bHQnXG4gICAgICAgICAgPyBgbmdndi0ke2lkfWBcbiAgICAgICAgICA6IGBuZ2d2LSR7bmFtZXNwYWNlfS0ke2lkfWBcbiAgICAgIH0sXG4gICAgfVxuICB9XG59KSgpXG4iXX0=
|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
export class DropdownUtils {
|
|
4
|
+
flattenOptions(options, mustHaveLabel) {
|
|
5
|
+
if (!options)
|
|
6
|
+
return [];
|
|
7
|
+
return options.reduce((acc, current) => {
|
|
8
|
+
if (this.isGroup(current))
|
|
9
|
+
return acc.concat(this.flattenOptions(current.options, mustHaveLabel));
|
|
10
|
+
if (!current.disabled && !(mustHaveLabel && !current.label))
|
|
11
|
+
acc.push(current); // skip disabled
|
|
12
|
+
return acc;
|
|
13
|
+
}, []);
|
|
14
|
+
}
|
|
15
|
+
/**
|
|
16
|
+
* Returns true if argument is an {@link OptionGroup}.
|
|
17
|
+
* @param option the object to check.
|
|
18
|
+
*/
|
|
19
|
+
isGroup(option) {
|
|
20
|
+
return 'options' in option;
|
|
21
|
+
}
|
|
22
|
+
deepEqual(original, change) {
|
|
23
|
+
if (original === change)
|
|
24
|
+
return true;
|
|
25
|
+
// breaks early if either original or change is a primitive value
|
|
26
|
+
if (typeof original !== 'object' ||
|
|
27
|
+
typeof change !== 'object' ||
|
|
28
|
+
original == null ||
|
|
29
|
+
change == null)
|
|
30
|
+
return false;
|
|
31
|
+
const keysOriginal = Object.keys(original);
|
|
32
|
+
const keysChange = Object.keys(change);
|
|
33
|
+
if (keysOriginal.length !== keysChange.length)
|
|
34
|
+
return false;
|
|
35
|
+
for (const key of keysOriginal) {
|
|
36
|
+
if (!keysChange.includes(key))
|
|
37
|
+
return false;
|
|
38
|
+
switch (true) {
|
|
39
|
+
case Array.isArray(original[key]) && Array.isArray(change[key]):
|
|
40
|
+
if (JSON.stringify(original[key].sort()) !==
|
|
41
|
+
JSON.stringify(change[key].sort()))
|
|
42
|
+
return false;
|
|
43
|
+
break;
|
|
44
|
+
case typeof original[key] === 'function':
|
|
45
|
+
case typeof change[key] === 'function':
|
|
46
|
+
if (original[key].toString() !== change[key].toString())
|
|
47
|
+
return false;
|
|
48
|
+
break;
|
|
49
|
+
default:
|
|
50
|
+
if (!this.deepEqual(original[key], change[key]))
|
|
51
|
+
return false;
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
return true;
|
|
55
|
+
}
|
|
56
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: DropdownUtils, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
57
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: DropdownUtils }); }
|
|
58
|
+
}
|
|
59
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: DropdownUtils, decorators: [{
|
|
60
|
+
type: Injectable
|
|
61
|
+
}] });
|
|
62
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29yZS51dGlscy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvYW5ndWxhci9zcmMvdi1hbmd1bGFyL2NvcmUvY29yZS51dGlscy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFBOztBQWlCMUMsTUFBTSxPQUFPLGFBQWE7SUFLakIsY0FBYyxDQUNuQixPQUFvQyxFQUNwQyxhQUFzQjtRQUV0QixJQUFJLENBQUMsT0FBTztZQUFFLE9BQU8sRUFBUyxDQUFBO1FBQzlCLE9BQU8sT0FBTyxDQUFDLE1BQU0sQ0FBQyxDQUFDLEdBQUcsRUFBRSxPQUFPLEVBQUUsRUFBRTtZQUNyQyxJQUFJLElBQUksQ0FBQyxPQUFPLENBQUMsT0FBTyxDQUFDO2dCQUN2QixPQUFPLEdBQUcsQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLGNBQWMsQ0FBQyxPQUFPLENBQUMsT0FBTyxFQUFFLGFBQWEsQ0FBQyxDQUFDLENBQUE7WUFDeEUsSUFBSSxDQUFDLE9BQU8sQ0FBQyxRQUFRLElBQUksQ0FBQyxDQUFDLGFBQWEsSUFBSSxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUM7Z0JBQ3pELEdBQUcsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUEsQ0FBQyxnQkFBZ0I7WUFDcEMsT0FBTyxHQUFHLENBQUE7UUFDWixDQUFDLEVBQUUsRUFBUyxDQUFDLENBQUE7SUFDZixDQUFDO0lBRUQ7OztPQUdHO0lBQ0ksT0FBTyxDQUFDLE1BQXFCO1FBQ2xDLE9BQU8sU0FBUyxJQUFJLE1BQU0sQ0FBQTtJQUM1QixDQUFDO0lBRU0sU0FBUyxDQUFDLFFBQWEsRUFBRSxNQUFXO1FBQ3pDLElBQUksUUFBUSxLQUFLLE1BQU07WUFBRSxPQUFPLElBQUksQ0FBQTtRQUVwQyxpRUFBaUU7UUFDakUsSUFDRSxPQUFPLFFBQVEsS0FBSyxRQUFRO1lBQzVCLE9BQU8sTUFBTSxLQUFLLFFBQVE7WUFDMUIsUUFBUSxJQUFJLElBQUk7WUFDaEIsTUFBTSxJQUFJLElBQUk7WUFFZCxPQUFPLEtBQUssQ0FBQTtRQUVkLE1BQU0sWUFBWSxHQUFHLE1BQU0sQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLENBQUE7UUFDMUMsTUFBTSxVQUFVLEdBQUcsTUFBTSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQTtRQUV0QyxJQUFJLFlBQVksQ0FBQyxNQUFNLEtBQUssVUFBVSxDQUFDLE1BQU07WUFBRSxPQUFPLEtBQUssQ0FBQTtRQUUzRCxLQUFLLE1BQU0sR0FBRyxJQUFJLFlBQVksRUFBRSxDQUFDO1lBQy9CLElBQUksQ0FBQyxVQUFVLENBQUMsUUFBUSxDQUFDLEdBQUcsQ0FBQztnQkFBRSxPQUFPLEtBQUssQ0FBQTtZQUUzQyxRQUFRLElBQUksRUFBRSxDQUFDO2dCQUNiLEtBQUssS0FBSyxDQUFDLE9BQU8sQ0FBQyxRQUFRLENBQUMsR0FBRyxDQUFDLENBQUMsSUFBSSxLQUFLLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxHQUFHLENBQUMsQ0FBQztvQkFDN0QsSUFDRSxJQUFJLENBQUMsU0FBUyxDQUFDLFFBQVEsQ0FBQyxHQUFHLENBQUMsQ0FBQyxJQUFJLEVBQUUsQ0FBQzt3QkFDcEMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxNQUFNLENBQUMsR0FBRyxDQUFDLENBQUMsSUFBSSxFQUFFLENBQUM7d0JBRWxDLE9BQU8sS0FBSyxDQUFBO29CQUNkLE1BQUs7Z0JBQ1AsS0FBSyxPQUFPLFFBQVEsQ0FBQyxHQUFHLENBQUMsS0FBSyxVQUFVLENBQUM7Z0JBQ3pDLEtBQUssT0FBTyxNQUFNLENBQUMsR0FBRyxDQUFDLEtBQUssVUFBVTtvQkFDcEMsSUFBSSxRQUFRLENBQUMsR0FBRyxDQUFDLENBQUMsUUFBUSxFQUFFLEtBQUssTUFBTSxDQUFDLEdBQUcsQ0FBQyxDQUFDLFFBQVEsRUFBRTt3QkFBRSxPQUFPLEtBQUssQ0FBQTtvQkFDckUsTUFBSztnQkFDUDtvQkFDRSxJQUFJLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxRQUFRLENBQUMsR0FBRyxDQUFDLEVBQUUsTUFBTSxDQUFDLEdBQUcsQ0FBQyxDQUFDO3dCQUFFLE9BQU8sS0FBSyxDQUFBO1lBQ2pFLENBQUM7UUFDSCxDQUFDO1FBRUQsT0FBTyxJQUFJLENBQUE7SUFDYixDQUFDOytHQWpFVSxhQUFhO21IQUFiLGFBQWE7OzRGQUFiLGFBQWE7a0JBRHpCLFVBQVUiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3RhYmxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSdcblxuZXhwb3J0IGludGVyZmFjZSBPcHRpb248SywgVj4ge1xuICBrZXk6IEtcbiAgbGFiZWw6IFZcbiAgZGlzYWJsZWQ/OiBib29sZWFuXG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgT3B0aW9uR3JvdXA8VCBleHRlbmRzIE9wdGlvbjxhbnksIGFueT4+IHtcbiAgbGFiZWw6IHN0cmluZ1xuICBvcHRpb25zOiBUW11cbiAgZGlzYWJsZWQ/OiBib29sZWFuXG59XG5cbmV4cG9ydCB0eXBlIE9wdGlvbkJhc2U8VCBleHRlbmRzIE9wdGlvbjxhbnksIGFueT4+ID0gVCB8IE9wdGlvbkdyb3VwPFQ+XG5cbkBJbmplY3RhYmxlKClcbmV4cG9ydCBjbGFzcyBEcm9wZG93blV0aWxzPFxuICBLID0gc3RyaW5nIHwgbnVsbCxcbiAgViA9IHN0cmluZyxcbiAgVCBleHRlbmRzIE9wdGlvbjxLLCBWPiA9IE9wdGlvbjxLLCBWPixcbj4ge1xuICBwdWJsaWMgZmxhdHRlbk9wdGlvbnMoXG4gICAgb3B0aW9uczogT3B0aW9uQmFzZTxUPltdIHwgdW5kZWZpbmVkLFxuICAgIG11c3RIYXZlTGFiZWw6IGJvb2xlYW4sXG4gICk6IFRbXSB7XG4gICAgaWYgKCFvcHRpb25zKSByZXR1cm4gW10gYXMgVFtdXG4gICAgcmV0dXJuIG9wdGlvbnMucmVkdWNlKChhY2MsIGN1cnJlbnQpID0+IHtcbiAgICAgIGlmICh0aGlzLmlzR3JvdXAoY3VycmVudCkpXG4gICAgICAgIHJldHVybiBhY2MuY29uY2F0KHRoaXMuZmxhdHRlbk9wdGlvbnMoY3VycmVudC5vcHRpb25zLCBtdXN0SGF2ZUxhYmVsKSlcbiAgICAgIGlmICghY3VycmVudC5kaXNhYmxlZCAmJiAhKG11c3RIYXZlTGFiZWwgJiYgIWN1cnJlbnQubGFiZWwpKVxuICAgICAgICBhY2MucHVzaChjdXJyZW50KSAvLyBza2lwIGRpc2FibGVkXG4gICAgICByZXR1cm4gYWNjXG4gICAgfSwgW10gYXMgVFtdKVxuICB9XG5cbiAgLyoqXG4gICAqIFJldHVybnMgdHJ1ZSBpZiBhcmd1bWVudCBpcyBhbiB7QGxpbmsgT3B0aW9uR3JvdXB9LlxuICAgKiBAcGFyYW0gb3B0aW9uIHRoZSBvYmplY3QgdG8gY2hlY2suXG4gICAqL1xuICBwdWJsaWMgaXNHcm91cChvcHRpb246IE9wdGlvbkJhc2U8VD4pOiBvcHRpb24gaXMgT3B0aW9uR3JvdXA8VD4ge1xuICAgIHJldHVybiAnb3B0aW9ucycgaW4gb3B0aW9uXG4gIH1cblxuICBwdWJsaWMgZGVlcEVxdWFsKG9yaWdpbmFsOiBhbnksIGNoYW5nZTogYW55KTogYm9vbGVhbiB7XG4gICAgaWYgKG9yaWdpbmFsID09PSBjaGFuZ2UpIHJldHVybiB0cnVlXG5cbiAgICAvLyBicmVha3MgZWFybHkgaWYgZWl0aGVyIG9yaWdpbmFsIG9yIGNoYW5nZSBpcyBhIHByaW1pdGl2ZSB2YWx1ZVxuICAgIGlmIChcbiAgICAgIHR5cGVvZiBvcmlnaW5hbCAhPT0gJ29iamVjdCcgfHxcbiAgICAgIHR5cGVvZiBjaGFuZ2UgIT09ICdvYmplY3QnIHx8XG4gICAgICBvcmlnaW5hbCA9PSBudWxsIHx8XG4gICAgICBjaGFuZ2UgPT0gbnVsbFxuICAgIClcbiAgICAgIHJldHVybiBmYWxzZVxuXG4gICAgY29uc3Qga2V5c09yaWdpbmFsID0gT2JqZWN0LmtleXMob3JpZ2luYWwpXG4gICAgY29uc3Qga2V5c0NoYW5nZSA9IE9iamVjdC5rZXlzKGNoYW5nZSlcblxuICAgIGlmIChrZXlzT3JpZ2luYWwubGVuZ3RoICE9PSBrZXlzQ2hhbmdlLmxlbmd0aCkgcmV0dXJuIGZhbHNlXG5cbiAgICBmb3IgKGNvbnN0IGtleSBvZiBrZXlzT3JpZ2luYWwpIHtcbiAgICAgIGlmICgha2V5c0NoYW5nZS5pbmNsdWRlcyhrZXkpKSByZXR1cm4gZmFsc2VcblxuICAgICAgc3dpdGNoICh0cnVlKSB7XG4gICAgICAgIGNhc2UgQXJyYXkuaXNBcnJheShvcmlnaW5hbFtrZXldKSAmJiBBcnJheS5pc0FycmF5KGNoYW5nZVtrZXldKTpcbiAgICAgICAgICBpZiAoXG4gICAgICAgICAgICBKU09OLnN0cmluZ2lmeShvcmlnaW5hbFtrZXldLnNvcnQoKSkgIT09XG4gICAgICAgICAgICBKU09OLnN0cmluZ2lmeShjaGFuZ2Vba2V5XS5zb3J0KCkpXG4gICAgICAgICAgKVxuICAgICAgICAgICAgcmV0dXJuIGZhbHNlXG4gICAgICAgICAgYnJlYWtcbiAgICAgICAgY2FzZSB0eXBlb2Ygb3JpZ2luYWxba2V5XSA9PT0gJ2Z1bmN0aW9uJzpcbiAgICAgICAgY2FzZSB0eXBlb2YgY2hhbmdlW2tleV0gPT09ICdmdW5jdGlvbic6XG4gICAgICAgICAgaWYgKG9yaWdpbmFsW2tleV0udG9TdHJpbmcoKSAhPT0gY2hhbmdlW2tleV0udG9TdHJpbmcoKSkgcmV0dXJuIGZhbHNlXG4gICAgICAgICAgYnJlYWtcbiAgICAgICAgZGVmYXVsdDpcbiAgICAgICAgICBpZiAoIXRoaXMuZGVlcEVxdWFsKG9yaWdpbmFsW2tleV0sIGNoYW5nZVtrZXldKSkgcmV0dXJuIGZhbHNlXG4gICAgICB9XG4gICAgfVxuXG4gICAgcmV0dXJuIHRydWVcbiAgfVxufVxuIl19
|
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
export * from './core.globals';
|
|
2
|
+
export * from './core.utils';
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9saWJzL2FuZ3VsYXIvc3JjL3YtYW5ndWxhci9jb3JlL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsZ0JBQWdCLENBQUE7QUFDOUIsY0FBYyxjQUFjLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2NvcmUuZ2xvYmFscydcbmV4cG9ydCAqIGZyb20gJy4vY29yZS51dGlscydcbiJdfQ==
|
|
@@ -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"]}
|