@skyux/popovers 8.7.0 → 9.0.0-alpha.0
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/{esm2020 → esm2022}/lib/modules/dropdown/dropdown-button.component.mjs +4 -4
- package/esm2022/lib/modules/dropdown/dropdown-item.component.mjs +68 -0
- package/esm2022/lib/modules/dropdown/dropdown-menu.component.mjs +314 -0
- package/esm2022/lib/modules/dropdown/dropdown.component.mjs +391 -0
- package/{esm2020 → esm2022}/lib/modules/dropdown/dropdown.module.mjs +21 -21
- package/{esm2020 → esm2022}/lib/modules/popover/popover-adapter.service.mjs +4 -4
- package/esm2022/lib/modules/popover/popover-content.component.mjs +273 -0
- package/esm2022/lib/modules/popover/popover.component.mjs +236 -0
- package/esm2022/lib/modules/popover/popover.directive.mjs +217 -0
- package/{esm2020 → esm2022}/lib/modules/popover/popover.module.mjs +19 -19
- package/esm2022/lib/modules/shared/sky-popovers-resources.module.mjs +47 -0
- package/esm2022/testing/dropdown/dropdown-fixture.mjs +138 -0
- package/{esm2020 → esm2022}/testing/dropdown/dropdown-testing.module.mjs +5 -5
- package/esm2022/testing/dropdown/harness/dropdown-harness.mjs +95 -0
- package/esm2022/testing/dropdown/harness/dropdown-item-harness.mjs +38 -0
- package/esm2022/testing/dropdown/harness/dropdown-menu-harness.mjs +63 -0
- package/esm2022/testing/popover/harness/popover-body-harness.mjs +42 -0
- package/esm2022/testing/popover/harness/popover-content-harness.mjs +94 -0
- package/esm2022/testing/popover/harness/popover-harness.mjs +51 -0
- package/esm2022/testing/popover/popover-fixture.mjs +88 -0
- package/{esm2020 → esm2022}/testing/popover/popover-testing.module.mjs +11 -11
- package/fesm2022/skyux-popovers-testing.mjs +651 -0
- package/{fesm2020 → fesm2022}/skyux-popovers-testing.mjs.map +1 -1
- package/fesm2022/skyux-popovers.mjs +1827 -0
- package/{fesm2020 → fesm2022}/skyux-popovers.mjs.map +1 -1
- package/lib/modules/dropdown/dropdown-item.component.d.ts +1 -1
- package/lib/modules/dropdown/dropdown-menu.component.d.ts +1 -1
- package/lib/modules/dropdown/dropdown.component.d.ts +1 -1
- package/lib/modules/popover/popover.component.d.ts +1 -1
- package/lib/modules/popover/popover.directive.d.ts +1 -1
- package/package.json +18 -26
- package/esm2020/lib/modules/dropdown/dropdown-item.component.mjs +0 -70
- package/esm2020/lib/modules/dropdown/dropdown-menu.component.mjs +0 -307
- package/esm2020/lib/modules/dropdown/dropdown.component.mjs +0 -375
- package/esm2020/lib/modules/popover/popover-content.component.mjs +0 -268
- package/esm2020/lib/modules/popover/popover.component.mjs +0 -233
- package/esm2020/lib/modules/popover/popover.directive.mjs +0 -209
- package/esm2020/lib/modules/shared/sky-popovers-resources.module.mjs +0 -47
- package/esm2020/testing/dropdown/dropdown-fixture.mjs +0 -137
- package/esm2020/testing/dropdown/harness/dropdown-harness.mjs +0 -102
- package/esm2020/testing/dropdown/harness/dropdown-item-harness.mjs +0 -44
- package/esm2020/testing/dropdown/harness/dropdown-menu-harness.mjs +0 -69
- package/esm2020/testing/popover/harness/popover-body-harness.mjs +0 -42
- package/esm2020/testing/popover/harness/popover-content-harness.mjs +0 -100
- package/esm2020/testing/popover/harness/popover-harness.mjs +0 -56
- package/esm2020/testing/popover/popover-fixture.mjs +0 -74
- package/fesm2015/skyux-popovers-testing.mjs +0 -739
- package/fesm2015/skyux-popovers-testing.mjs.map +0 -1
- package/fesm2015/skyux-popovers.mjs +0 -1807
- package/fesm2015/skyux-popovers.mjs.map +0 -1
- package/fesm2020/skyux-popovers-testing.mjs +0 -660
- package/fesm2020/skyux-popovers.mjs +0 -1785
- /package/{esm2020 → esm2022}/index.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/dropdown/dropdown-extensions.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/dropdown/types/dropdown-button-type.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/dropdown/types/dropdown-horizontal-alignment.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/dropdown/types/dropdown-menu-change.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/dropdown/types/dropdown-message-type.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/dropdown/types/dropdown-message.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/dropdown/types/dropdown-trigger-type.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/popover/popover-animation-state.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/popover/popover-animation.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/popover/popover-context.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/popover/popover-extensions.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/popover/types/popover-adapter-arrow-coordinates.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/popover/types/popover-adapter-elements.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/popover/types/popover-alignment.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/popover/types/popover-message-type.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/popover/types/popover-message.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/popover/types/popover-placement.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/popover/types/popover-position.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/popover/types/popover-trigger.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/popover/types/popover-type.mjs +0 -0
- /package/{esm2020 → esm2022}/skyux-popovers.mjs +0 -0
- /package/{esm2020 → esm2022}/testing/dropdown/harness/dropdown-harness.filters.mjs +0 -0
- /package/{esm2020 → esm2022}/testing/dropdown/harness/dropdown-item-harness.filters.mjs +0 -0
- /package/{esm2020 → esm2022}/testing/dropdown/harness/dropdown-menu-harness.filters.mjs +0 -0
- /package/{esm2020 → esm2022}/testing/dropdown/popovers-fixture-dropdown-item.mjs +0 -0
- /package/{esm2020 → esm2022}/testing/dropdown/popovers-fixture-dropdown-menu.mjs +0 -0
- /package/{esm2020 → esm2022}/testing/dropdown/popovers-fixture-dropdown.mjs +0 -0
- /package/{esm2020 → esm2022}/testing/popover/harness/popover-content-harness-filters.mjs +0 -0
- /package/{esm2020 → esm2022}/testing/popover/harness/popover-harness-filters.mjs +0 -0
- /package/{esm2020 → esm2022}/testing/public-api.mjs +0 -0
- /package/{esm2020 → esm2022}/testing/skyux-popovers-testing.mjs +0 -0
|
@@ -4,11 +4,11 @@ import * as i0 from "@angular/core";
|
|
|
4
4
|
* Specifies the button for the dropdown menu.
|
|
5
5
|
*/
|
|
6
6
|
export class SkyDropdownButtonComponent {
|
|
7
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.7", ngImport: i0, type: SkyDropdownButtonComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
8
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.7", type: SkyDropdownButtonComponent, selector: "sky-dropdown-button", ngImport: i0, template: "<ng-content></ng-content>\n" }); }
|
|
7
9
|
}
|
|
8
|
-
|
|
9
|
-
SkyDropdownButtonComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: SkyDropdownButtonComponent, selector: "sky-dropdown-button", ngImport: i0, template: "<ng-content></ng-content>\n" });
|
|
10
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: SkyDropdownButtonComponent, decorators: [{
|
|
10
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.7", ngImport: i0, type: SkyDropdownButtonComponent, decorators: [{
|
|
11
11
|
type: Component,
|
|
12
12
|
args: [{ selector: 'sky-dropdown-button', template: "<ng-content></ng-content>\n" }]
|
|
13
13
|
}] });
|
|
14
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
14
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZHJvcGRvd24tYnV0dG9uLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvY29tcG9uZW50cy9wb3BvdmVycy9zcmMvbGliL21vZHVsZXMvZHJvcGRvd24vZHJvcGRvd24tYnV0dG9uLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvY29tcG9uZW50cy9wb3BvdmVycy9zcmMvbGliL21vZHVsZXMvZHJvcGRvd24vZHJvcGRvd24tYnV0dG9uLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUM7O0FBRTFDOztHQUVHO0FBS0gsTUFBTSxPQUFPLDBCQUEwQjs4R0FBMUIsMEJBQTBCO2tHQUExQiwwQkFBMEIsMkRDVHZDLDZCQUNBOzsyRkRRYSwwQkFBMEI7a0JBSnRDLFNBQVM7K0JBQ0UscUJBQXFCIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbi8qKlxuICogU3BlY2lmaWVzIHRoZSBidXR0b24gZm9yIHRoZSBkcm9wZG93biBtZW51LlxuICovXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdza3ktZHJvcGRvd24tYnV0dG9uJyxcbiAgdGVtcGxhdGVVcmw6ICcuL2Ryb3Bkb3duLWJ1dHRvbi5jb21wb25lbnQuaHRtbCcsXG59KVxuZXhwb3J0IGNsYXNzIFNreURyb3Bkb3duQnV0dG9uQ29tcG9uZW50IHt9XG4iLCI8bmctY29udGVudD48L25nLWNvbnRlbnQ+XG4iXX0=
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, Input, ViewEncapsulation, } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "@angular/common";
|
|
4
|
+
/**
|
|
5
|
+
* Specifies the items to display on the dropdown menu.
|
|
6
|
+
*/
|
|
7
|
+
export class SkyDropdownItemComponent {
|
|
8
|
+
/**
|
|
9
|
+
* The ARIA role for the dropdown menu item
|
|
10
|
+
* [to support accessibility](https://developer.blackbaud.com/skyux/learn/accessibility)
|
|
11
|
+
* by indicating how the item functions and what it controls. For information about
|
|
12
|
+
* how an ARIA role indicates what an item represents on a web page, see the
|
|
13
|
+
* [WAI-ARIA roles model](https://www.w3.org/WAI/PF/aria/#roles).
|
|
14
|
+
* @default "menuitem"
|
|
15
|
+
*/
|
|
16
|
+
set ariaRole(value) {
|
|
17
|
+
this.#_ariaRole = value ?? 'menuitem';
|
|
18
|
+
}
|
|
19
|
+
get ariaRole() {
|
|
20
|
+
return this.#_ariaRole;
|
|
21
|
+
}
|
|
22
|
+
#_ariaRole;
|
|
23
|
+
#changeDetector;
|
|
24
|
+
#renderer;
|
|
25
|
+
constructor(elementRef, changeDetector, renderer) {
|
|
26
|
+
this.elementRef = elementRef;
|
|
27
|
+
this.isActive = false;
|
|
28
|
+
this.#_ariaRole = 'menuitem';
|
|
29
|
+
this.#changeDetector = changeDetector;
|
|
30
|
+
this.#renderer = renderer;
|
|
31
|
+
}
|
|
32
|
+
ngAfterViewInit() {
|
|
33
|
+
// Make sure anchor elements are tab-able.
|
|
34
|
+
const buttonElement = this.#getButtonElement();
|
|
35
|
+
/* istanbul ignore else */
|
|
36
|
+
if (buttonElement) {
|
|
37
|
+
this.#renderer.setAttribute(buttonElement, 'tabIndex', '0');
|
|
38
|
+
}
|
|
39
|
+
this.#changeDetector.detectChanges();
|
|
40
|
+
}
|
|
41
|
+
focusElement(enableNativeFocus) {
|
|
42
|
+
this.isActive = true;
|
|
43
|
+
if (enableNativeFocus) {
|
|
44
|
+
this.#getButtonElement()?.focus();
|
|
45
|
+
}
|
|
46
|
+
this.#changeDetector.detectChanges();
|
|
47
|
+
}
|
|
48
|
+
isFocusable() {
|
|
49
|
+
const isFocusable = this.#getButtonElement()?.getAttribute('disabled') === null;
|
|
50
|
+
return isFocusable;
|
|
51
|
+
}
|
|
52
|
+
resetState() {
|
|
53
|
+
this.isActive = false;
|
|
54
|
+
this.#changeDetector.markForCheck();
|
|
55
|
+
}
|
|
56
|
+
#getButtonElement() {
|
|
57
|
+
return this.elementRef.nativeElement.querySelector('button,a');
|
|
58
|
+
}
|
|
59
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.7", ngImport: i0, type: SkyDropdownItemComponent, deps: [{ token: i0.ElementRef }, { token: i0.ChangeDetectorRef }, { token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
60
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.7", type: SkyDropdownItemComponent, selector: "sky-dropdown-item", inputs: { ariaRole: "ariaRole" }, ngImport: i0, template: "<div\n class=\"sky-dropdown-item\"\n [attr.role]=\"ariaRole\"\n [ngClass]=\"{\n 'sky-dropdown-item-active': isActive\n }\"\n>\n <ng-content> </ng-content>\n</div>\n", styles: [".sky-dropdown-item{background-color:transparent;border:none;display:block;margin:4px;min-width:160px;text-align:left}.sky-dropdown-item.sky-dropdown-item-active a,.sky-dropdown-item.sky-dropdown-item-active button,.sky-dropdown-item:hover a,.sky-dropdown-item:hover button{background-color:#eeeeef}.sky-dropdown-item button[disabled],.sky-dropdown-item a[disabled]{cursor:default}.sky-dropdown-item button[disabled]:hover,.sky-dropdown-item a[disabled]:hover{background-color:transparent}.sky-dropdown-item>a,.sky-dropdown-item>button{background-color:transparent;border:none;color:#212327;cursor:pointer;display:block;padding:3px 20px;text-align:left;width:100%}.sky-dropdown-item>a:hover,.sky-dropdown-item>button:hover{text-decoration:none}.sky-dropdown-item>a[disabled],.sky-dropdown-item>button[disabled]{color:#686c73}.sky-dropdown-item>a[disabled]:hover,.sky-dropdown-item>button[disabled]:hover{cursor:default}.sky-theme-modern .sky-dropdown-item{margin:0}.sky-theme-modern .sky-dropdown-item.sky-dropdown-item-active,.sky-theme-modern .sky-dropdown-item:hover{background-color:transparent}.sky-theme-modern .sky-dropdown-item>a,.sky-theme-modern .sky-dropdown-item>button{padding:10px 20px;border-radius:6px;transition:box-shadow .15s}.sky-theme-modern .sky-dropdown-item>a:hover,.sky-theme-modern .sky-dropdown-item>button:hover{outline:solid 1px #1870B8;outline-offset:-1px}.sky-theme-modern .sky-dropdown-item>a:focus-visible,.sky-theme-modern .sky-dropdown-item>a:active,.sky-theme-modern .sky-dropdown-item>button:focus-visible,.sky-theme-modern .sky-dropdown-item>button:active{outline:solid 2px #1870B8;outline-offset:-2px}.sky-theme-modern .sky-dropdown-item>a:focus-visible:not(:active),.sky-theme-modern .sky-dropdown-item>button:focus-visible:not(:active){box-shadow:0 1px 8px #0000004d}.sky-theme-modern.sky-theme-mode-dark .sky-dropdown-item>a,.sky-theme-modern.sky-theme-mode-dark .sky-dropdown-item>button{color:#fbfcfe}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
|
|
61
|
+
}
|
|
62
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.7", ngImport: i0, type: SkyDropdownItemComponent, decorators: [{
|
|
63
|
+
type: Component,
|
|
64
|
+
args: [{ selector: 'sky-dropdown-item', changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, template: "<div\n class=\"sky-dropdown-item\"\n [attr.role]=\"ariaRole\"\n [ngClass]=\"{\n 'sky-dropdown-item-active': isActive\n }\"\n>\n <ng-content> </ng-content>\n</div>\n", styles: [".sky-dropdown-item{background-color:transparent;border:none;display:block;margin:4px;min-width:160px;text-align:left}.sky-dropdown-item.sky-dropdown-item-active a,.sky-dropdown-item.sky-dropdown-item-active button,.sky-dropdown-item:hover a,.sky-dropdown-item:hover button{background-color:#eeeeef}.sky-dropdown-item button[disabled],.sky-dropdown-item a[disabled]{cursor:default}.sky-dropdown-item button[disabled]:hover,.sky-dropdown-item a[disabled]:hover{background-color:transparent}.sky-dropdown-item>a,.sky-dropdown-item>button{background-color:transparent;border:none;color:#212327;cursor:pointer;display:block;padding:3px 20px;text-align:left;width:100%}.sky-dropdown-item>a:hover,.sky-dropdown-item>button:hover{text-decoration:none}.sky-dropdown-item>a[disabled],.sky-dropdown-item>button[disabled]{color:#686c73}.sky-dropdown-item>a[disabled]:hover,.sky-dropdown-item>button[disabled]:hover{cursor:default}.sky-theme-modern .sky-dropdown-item{margin:0}.sky-theme-modern .sky-dropdown-item.sky-dropdown-item-active,.sky-theme-modern .sky-dropdown-item:hover{background-color:transparent}.sky-theme-modern .sky-dropdown-item>a,.sky-theme-modern .sky-dropdown-item>button{padding:10px 20px;border-radius:6px;transition:box-shadow .15s}.sky-theme-modern .sky-dropdown-item>a:hover,.sky-theme-modern .sky-dropdown-item>button:hover{outline:solid 1px #1870B8;outline-offset:-1px}.sky-theme-modern .sky-dropdown-item>a:focus-visible,.sky-theme-modern .sky-dropdown-item>a:active,.sky-theme-modern .sky-dropdown-item>button:focus-visible,.sky-theme-modern .sky-dropdown-item>button:active{outline:solid 2px #1870B8;outline-offset:-2px}.sky-theme-modern .sky-dropdown-item>a:focus-visible:not(:active),.sky-theme-modern .sky-dropdown-item>button:focus-visible:not(:active){box-shadow:0 1px 8px #0000004d}.sky-theme-modern.sky-theme-mode-dark .sky-dropdown-item>a,.sky-theme-modern.sky-theme-mode-dark .sky-dropdown-item>button{color:#fbfcfe}\n"] }]
|
|
65
|
+
}], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.ChangeDetectorRef }, { type: i0.Renderer2 }]; }, propDecorators: { ariaRole: [{
|
|
66
|
+
type: Input
|
|
67
|
+
}] } });
|
|
68
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZHJvcGRvd24taXRlbS5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9saWJzL2NvbXBvbmVudHMvcG9wb3ZlcnMvc3JjL2xpYi9tb2R1bGVzL2Ryb3Bkb3duL2Ryb3Bkb3duLWl0ZW0uY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9jb21wb25lbnRzL3BvcG92ZXJzL3NyYy9saWIvbW9kdWxlcy9kcm9wZG93bi9kcm9wZG93bi1pdGVtLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFFTCx1QkFBdUIsRUFFdkIsU0FBUyxFQUVULEtBQUssRUFFTCxpQkFBaUIsR0FDbEIsTUFBTSxlQUFlLENBQUM7OztBQUV2Qjs7R0FFRztBQVFILE1BQU0sT0FBTyx3QkFBd0I7SUFDbkM7Ozs7Ozs7T0FPRztJQUNILElBQ1csUUFBUSxDQUFDLEtBQXlCO1FBQzNDLElBQUksQ0FBQyxVQUFVLEdBQUcsS0FBSyxJQUFJLFVBQVUsQ0FBQztJQUN4QyxDQUFDO0lBRUQsSUFBVyxRQUFRO1FBQ2pCLE9BQU8sSUFBSSxDQUFDLFVBQVUsQ0FBQztJQUN6QixDQUFDO0lBSUQsVUFBVSxDQUFjO0lBRXhCLGVBQWUsQ0FBb0I7SUFDbkMsU0FBUyxDQUFZO0lBRXJCLFlBQ1MsVUFBc0IsRUFDN0IsY0FBaUMsRUFDakMsUUFBbUI7UUFGWixlQUFVLEdBQVYsVUFBVSxDQUFZO1FBUnhCLGFBQVEsR0FBRyxLQUFLLENBQUM7UUFFeEIsZUFBVSxHQUFHLFVBQVUsQ0FBQztRQVV0QixJQUFJLENBQUMsZUFBZSxHQUFHLGNBQWMsQ0FBQztRQUN0QyxJQUFJLENBQUMsU0FBUyxHQUFHLFFBQVEsQ0FBQztJQUM1QixDQUFDO0lBRU0sZUFBZTtRQUNwQiwwQ0FBMEM7UUFDMUMsTUFBTSxhQUFhLEdBQUcsSUFBSSxDQUFDLGlCQUFpQixFQUFFLENBQUM7UUFDL0MsMEJBQTBCO1FBQzFCLElBQUksYUFBYSxFQUFFO1lBQ2pCLElBQUksQ0FBQyxTQUFTLENBQUMsWUFBWSxDQUFDLGFBQWEsRUFBRSxVQUFVLEVBQUUsR0FBRyxDQUFDLENBQUM7U0FDN0Q7UUFFRCxJQUFJLENBQUMsZUFBZSxDQUFDLGFBQWEsRUFBRSxDQUFDO0lBQ3ZDLENBQUM7SUFFTSxZQUFZLENBQUMsaUJBQTBCO1FBQzVDLElBQUksQ0FBQyxRQUFRLEdBQUcsSUFBSSxDQUFDO1FBRXJCLElBQUksaUJBQWlCLEVBQUU7WUFDckIsSUFBSSxDQUFDLGlCQUFpQixFQUFFLEVBQUUsS0FBSyxFQUFFLENBQUM7U0FDbkM7UUFFRCxJQUFJLENBQUMsZUFBZSxDQUFDLGFBQWEsRUFBRSxDQUFDO0lBQ3ZDLENBQUM7SUFFTSxXQUFXO1FBQ2hCLE1BQU0sV0FBVyxHQUNmLElBQUksQ0FBQyxpQkFBaUIsRUFBRSxFQUFFLFlBQVksQ0FBQyxVQUFVLENBQUMsS0FBSyxJQUFJLENBQUM7UUFDOUQsT0FBTyxXQUFXLENBQUM7SUFDckIsQ0FBQztJQUVNLFVBQVU7UUFDZixJQUFJLENBQUMsUUFBUSxHQUFHLEtBQUssQ0FBQztRQUN0QixJQUFJLENBQUMsZUFBZSxDQUFDLFlBQVksRUFBRSxDQUFDO0lBQ3RDLENBQUM7SUFFRCxpQkFBaUI7UUFDZixPQUFPLElBQUksQ0FBQyxVQUFVLENBQUMsYUFBYSxDQUFDLGFBQWEsQ0FBQyxVQUFVLENBQUMsQ0FBQztJQUNqRSxDQUFDOzhHQXBFVSx3QkFBd0I7a0dBQXhCLHdCQUF3QiwyRkNyQnJDLDhLQVNBOzsyRkRZYSx3QkFBd0I7a0JBUHBDLFNBQVM7K0JBQ0UsbUJBQW1CLG1CQUdaLHVCQUF1QixDQUFDLE1BQU0saUJBQ2hDLGlCQUFpQixDQUFDLElBQUk7eUpBWTFCLFFBQVE7c0JBRGxCLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1xuICBBZnRlclZpZXdJbml0LFxuICBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneSxcbiAgQ2hhbmdlRGV0ZWN0b3JSZWYsXG4gIENvbXBvbmVudCxcbiAgRWxlbWVudFJlZixcbiAgSW5wdXQsXG4gIFJlbmRlcmVyMixcbiAgVmlld0VuY2Fwc3VsYXRpb24sXG59IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG4vKipcbiAqIFNwZWNpZmllcyB0aGUgaXRlbXMgdG8gZGlzcGxheSBvbiB0aGUgZHJvcGRvd24gbWVudS5cbiAqL1xuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnc2t5LWRyb3Bkb3duLWl0ZW0nLFxuICB0ZW1wbGF0ZVVybDogJy4vZHJvcGRvd24taXRlbS5jb21wb25lbnQuaHRtbCcsXG4gIHN0eWxlVXJsczogWycuL2Ryb3Bkb3duLWl0ZW0uY29tcG9uZW50LnNjc3MnXSxcbiAgY2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2gsXG4gIGVuY2Fwc3VsYXRpb246IFZpZXdFbmNhcHN1bGF0aW9uLk5vbmUsXG59KVxuZXhwb3J0IGNsYXNzIFNreURyb3Bkb3duSXRlbUNvbXBvbmVudCBpbXBsZW1lbnRzIEFmdGVyVmlld0luaXQge1xuICAvKipcbiAgICogVGhlIEFSSUEgcm9sZSBmb3IgdGhlIGRyb3Bkb3duIG1lbnUgaXRlbVxuICAgKiBbdG8gc3VwcG9ydCBhY2Nlc3NpYmlsaXR5XShodHRwczovL2RldmVsb3Blci5ibGFja2JhdWQuY29tL3NreXV4L2xlYXJuL2FjY2Vzc2liaWxpdHkpXG4gICAqIGJ5IGluZGljYXRpbmcgaG93IHRoZSBpdGVtIGZ1bmN0aW9ucyBhbmQgd2hhdCBpdCBjb250cm9scy4gRm9yIGluZm9ybWF0aW9uIGFib3V0XG4gICAqIGhvdyBhbiBBUklBIHJvbGUgaW5kaWNhdGVzIHdoYXQgYW4gaXRlbSByZXByZXNlbnRzIG9uIGEgd2ViIHBhZ2UsIHNlZSB0aGVcbiAgICogW1dBSS1BUklBIHJvbGVzIG1vZGVsXShodHRwczovL3d3dy53My5vcmcvV0FJL1BGL2FyaWEvI3JvbGVzKS5cbiAgICogQGRlZmF1bHQgXCJtZW51aXRlbVwiXG4gICAqL1xuICBASW5wdXQoKVxuICBwdWJsaWMgc2V0IGFyaWFSb2xlKHZhbHVlOiBzdHJpbmcgfCB1bmRlZmluZWQpIHtcbiAgICB0aGlzLiNfYXJpYVJvbGUgPSB2YWx1ZSA/PyAnbWVudWl0ZW0nO1xuICB9XG5cbiAgcHVibGljIGdldCBhcmlhUm9sZSgpOiBzdHJpbmcge1xuICAgIHJldHVybiB0aGlzLiNfYXJpYVJvbGU7XG4gIH1cblxuICBwdWJsaWMgaXNBY3RpdmUgPSBmYWxzZTtcblxuICAjX2FyaWFSb2xlID0gJ21lbnVpdGVtJztcblxuICAjY2hhbmdlRGV0ZWN0b3I6IENoYW5nZURldGVjdG9yUmVmO1xuICAjcmVuZGVyZXI6IFJlbmRlcmVyMjtcblxuICBjb25zdHJ1Y3RvcihcbiAgICBwdWJsaWMgZWxlbWVudFJlZjogRWxlbWVudFJlZixcbiAgICBjaGFuZ2VEZXRlY3RvcjogQ2hhbmdlRGV0ZWN0b3JSZWYsXG4gICAgcmVuZGVyZXI6IFJlbmRlcmVyMlxuICApIHtcbiAgICB0aGlzLiNjaGFuZ2VEZXRlY3RvciA9IGNoYW5nZURldGVjdG9yO1xuICAgIHRoaXMuI3JlbmRlcmVyID0gcmVuZGVyZXI7XG4gIH1cblxuICBwdWJsaWMgbmdBZnRlclZpZXdJbml0KCk6IHZvaWQge1xuICAgIC8vIE1ha2Ugc3VyZSBhbmNob3IgZWxlbWVudHMgYXJlIHRhYi1hYmxlLlxuICAgIGNvbnN0IGJ1dHRvbkVsZW1lbnQgPSB0aGlzLiNnZXRCdXR0b25FbGVtZW50KCk7XG4gICAgLyogaXN0YW5idWwgaWdub3JlIGVsc2UgKi9cbiAgICBpZiAoYnV0dG9uRWxlbWVudCkge1xuICAgICAgdGhpcy4jcmVuZGVyZXIuc2V0QXR0cmlidXRlKGJ1dHRvbkVsZW1lbnQsICd0YWJJbmRleCcsICcwJyk7XG4gICAgfVxuXG4gICAgdGhpcy4jY2hhbmdlRGV0ZWN0b3IuZGV0ZWN0Q2hhbmdlcygpO1xuICB9XG5cbiAgcHVibGljIGZvY3VzRWxlbWVudChlbmFibGVOYXRpdmVGb2N1czogYm9vbGVhbik6IHZvaWQge1xuICAgIHRoaXMuaXNBY3RpdmUgPSB0cnVlO1xuXG4gICAgaWYgKGVuYWJsZU5hdGl2ZUZvY3VzKSB7XG4gICAgICB0aGlzLiNnZXRCdXR0b25FbGVtZW50KCk/LmZvY3VzKCk7XG4gICAgfVxuXG4gICAgdGhpcy4jY2hhbmdlRGV0ZWN0b3IuZGV0ZWN0Q2hhbmdlcygpO1xuICB9XG5cbiAgcHVibGljIGlzRm9jdXNhYmxlKCk6IGJvb2xlYW4ge1xuICAgIGNvbnN0IGlzRm9jdXNhYmxlID1cbiAgICAgIHRoaXMuI2dldEJ1dHRvbkVsZW1lbnQoKT8uZ2V0QXR0cmlidXRlKCdkaXNhYmxlZCcpID09PSBudWxsO1xuICAgIHJldHVybiBpc0ZvY3VzYWJsZTtcbiAgfVxuXG4gIHB1YmxpYyByZXNldFN0YXRlKCk6IHZvaWQge1xuICAgIHRoaXMuaXNBY3RpdmUgPSBmYWxzZTtcbiAgICB0aGlzLiNjaGFuZ2VEZXRlY3Rvci5tYXJrRm9yQ2hlY2soKTtcbiAgfVxuXG4gICNnZXRCdXR0b25FbGVtZW50KCk6IEhUTUxCdXR0b25FbGVtZW50IHwgbnVsbCB7XG4gICAgcmV0dXJuIHRoaXMuZWxlbWVudFJlZi5uYXRpdmVFbGVtZW50LnF1ZXJ5U2VsZWN0b3IoJ2J1dHRvbixhJyk7XG4gIH1cbn1cbiIsIjxkaXZcbiAgY2xhc3M9XCJza3ktZHJvcGRvd24taXRlbVwiXG4gIFthdHRyLnJvbGVdPVwiYXJpYVJvbGVcIlxuICBbbmdDbGFzc109XCJ7XG4gICAgJ3NreS1kcm9wZG93bi1pdGVtLWFjdGl2ZSc6IGlzQWN0aXZlXG4gIH1cIlxuPlxuICA8bmctY29udGVudD4gPC9uZy1jb250ZW50PlxuPC9kaXY+XG4iXX0=
|
|
@@ -0,0 +1,314 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, ContentChildren, EventEmitter, Input, Optional, Output, ViewEncapsulation, } from '@angular/core';
|
|
2
|
+
import { Subject, fromEvent as observableFromEvent } from 'rxjs';
|
|
3
|
+
import { takeUntil } from 'rxjs/operators';
|
|
4
|
+
import { SkyDropdownItemComponent } from './dropdown-item.component';
|
|
5
|
+
import { SkyDropdownMessageType } from './types/dropdown-message-type';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "./dropdown.component";
|
|
8
|
+
let nextId = 0;
|
|
9
|
+
/**
|
|
10
|
+
* Creates a menu that contains dropdown menu items.
|
|
11
|
+
*
|
|
12
|
+
*/
|
|
13
|
+
export class SkyDropdownMenuComponent {
|
|
14
|
+
/**
|
|
15
|
+
* The ARIA role for the dropdown menu
|
|
16
|
+
* [to support accessibility](https://developer.blackbaud.com/skyux/learn/accessibility)
|
|
17
|
+
* by indicating how the dropdown menu functions and what it controls. The dropdown button
|
|
18
|
+
* inherits this value to set its `aria-haspopup` property. For information
|
|
19
|
+
* about how an ARIA role indicates what an item represents on a web page, see the
|
|
20
|
+
* [WAI-ARIA roles model](https://www.w3.org/WAI/PF/aria/#roles).
|
|
21
|
+
* @default "menu"
|
|
22
|
+
*/
|
|
23
|
+
set ariaRole(value) {
|
|
24
|
+
this.#_ariaRole = value ?? 'menu';
|
|
25
|
+
}
|
|
26
|
+
get ariaRole() {
|
|
27
|
+
return this.#_ariaRole;
|
|
28
|
+
}
|
|
29
|
+
/**
|
|
30
|
+
* Whether to use the browser's native focus function when users navigate through menu
|
|
31
|
+
* items with the keyboard. To disable the native focus function, set this property to `false`.
|
|
32
|
+
* For example, to let users interact with the dropdown menu but keep the keyboard focus on a
|
|
33
|
+
* different element, set this property to `false`.
|
|
34
|
+
* @default true
|
|
35
|
+
* @internal
|
|
36
|
+
*/
|
|
37
|
+
set useNativeFocus(value) {
|
|
38
|
+
this.#_useNativeFocus = value ?? true;
|
|
39
|
+
}
|
|
40
|
+
get useNativeFocus() {
|
|
41
|
+
return this.#_useNativeFocus;
|
|
42
|
+
}
|
|
43
|
+
set menuIndex(value) {
|
|
44
|
+
if (value < 0) {
|
|
45
|
+
value = this.menuItems.length - 1;
|
|
46
|
+
}
|
|
47
|
+
if (value >= this.menuItems.length) {
|
|
48
|
+
value = 0;
|
|
49
|
+
}
|
|
50
|
+
this.#_menuIndex = value;
|
|
51
|
+
}
|
|
52
|
+
get menuIndex() {
|
|
53
|
+
return this.#_menuIndex;
|
|
54
|
+
}
|
|
55
|
+
#ngUnsubscribe;
|
|
56
|
+
#_ariaRole;
|
|
57
|
+
#_menuIndex;
|
|
58
|
+
#_useNativeFocus;
|
|
59
|
+
#changeDetector;
|
|
60
|
+
#elementRef;
|
|
61
|
+
#dropdownComponent;
|
|
62
|
+
constructor(changeDetector, elementRef, dropdownComponent) {
|
|
63
|
+
/**
|
|
64
|
+
* Fires when the dropdown menu's active index or selected item changes. This event provides an
|
|
65
|
+
* observable to emit changes, and the response is of
|
|
66
|
+
* the SkyDropdownMenuChange type.
|
|
67
|
+
* @internal
|
|
68
|
+
*/
|
|
69
|
+
this.menuChanges = new EventEmitter();
|
|
70
|
+
this.dropdownMenuId = `sky-dropdown-menu-${++nextId}`;
|
|
71
|
+
this.#ngUnsubscribe = new Subject();
|
|
72
|
+
this.#_ariaRole = 'menu';
|
|
73
|
+
this.#_menuIndex = 0;
|
|
74
|
+
this.#_useNativeFocus = true;
|
|
75
|
+
this.#changeDetector = changeDetector;
|
|
76
|
+
this.#elementRef = elementRef;
|
|
77
|
+
this.#dropdownComponent = dropdownComponent;
|
|
78
|
+
}
|
|
79
|
+
ngAfterContentInit() {
|
|
80
|
+
/* istanbul ignore else */
|
|
81
|
+
if (this.#dropdownComponent) {
|
|
82
|
+
this.#dropdownComponent.menuId = this.dropdownMenuId;
|
|
83
|
+
this.#dropdownComponent.menuAriaRole = this.ariaRole;
|
|
84
|
+
this.#dropdownComponent.messageStream
|
|
85
|
+
?.pipe(takeUntil(this.#ngUnsubscribe))
|
|
86
|
+
.subscribe((message) => {
|
|
87
|
+
switch (message.type) {
|
|
88
|
+
case SkyDropdownMessageType.Open:
|
|
89
|
+
case SkyDropdownMessageType.Close:
|
|
90
|
+
this.reset();
|
|
91
|
+
break;
|
|
92
|
+
case SkyDropdownMessageType.FocusFirstItem:
|
|
93
|
+
this.focusFirstItem();
|
|
94
|
+
break;
|
|
95
|
+
case SkyDropdownMessageType.FocusNextItem:
|
|
96
|
+
this.focusNextItem();
|
|
97
|
+
break;
|
|
98
|
+
case SkyDropdownMessageType.FocusPreviousItem:
|
|
99
|
+
this.focusPreviousItem();
|
|
100
|
+
break;
|
|
101
|
+
case SkyDropdownMessageType.FocusLastItem:
|
|
102
|
+
this.focusLastItem();
|
|
103
|
+
break;
|
|
104
|
+
}
|
|
105
|
+
});
|
|
106
|
+
this.menuChanges
|
|
107
|
+
.pipe(takeUntil(this.#ngUnsubscribe))
|
|
108
|
+
.subscribe((change) => {
|
|
109
|
+
// Close the dropdown when a menu item is selected.
|
|
110
|
+
if (change.selectedItem) {
|
|
111
|
+
this.#sendMessage(SkyDropdownMessageType.Close);
|
|
112
|
+
this.#sendMessage(SkyDropdownMessageType.FocusTriggerButton);
|
|
113
|
+
}
|
|
114
|
+
if (change.items) {
|
|
115
|
+
// Update the popover style and position whenever the number of items changes.
|
|
116
|
+
this.#sendMessage(SkyDropdownMessageType.Reposition);
|
|
117
|
+
}
|
|
118
|
+
});
|
|
119
|
+
}
|
|
120
|
+
// Reset dropdown whenever the menu items change.
|
|
121
|
+
this.menuItems.changes
|
|
122
|
+
.pipe(takeUntil(this.#ngUnsubscribe))
|
|
123
|
+
.subscribe((items) => {
|
|
124
|
+
this.reset();
|
|
125
|
+
this.menuChanges.emit({
|
|
126
|
+
items: items.toArray(),
|
|
127
|
+
});
|
|
128
|
+
});
|
|
129
|
+
this.#addEventListeners();
|
|
130
|
+
}
|
|
131
|
+
ngOnDestroy() {
|
|
132
|
+
this.#ngUnsubscribe.next();
|
|
133
|
+
this.#ngUnsubscribe.complete();
|
|
134
|
+
}
|
|
135
|
+
focusFirstItem() {
|
|
136
|
+
if (!this.#hasFocusableItems()) {
|
|
137
|
+
return;
|
|
138
|
+
}
|
|
139
|
+
this.menuIndex = 0;
|
|
140
|
+
const firstItem = this.#getItemByIndex(this.menuIndex);
|
|
141
|
+
if (firstItem && firstItem.isFocusable()) {
|
|
142
|
+
this.#focusItem(firstItem);
|
|
143
|
+
}
|
|
144
|
+
else {
|
|
145
|
+
this.focusNextItem();
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
focusLastItem() {
|
|
149
|
+
if (!this.#hasFocusableItems()) {
|
|
150
|
+
return;
|
|
151
|
+
}
|
|
152
|
+
this.menuIndex = this.menuItems.length - 1;
|
|
153
|
+
const lastItem = this.#getItemByIndex(this.menuIndex);
|
|
154
|
+
if (lastItem && lastItem.isFocusable()) {
|
|
155
|
+
this.#focusItem(lastItem);
|
|
156
|
+
}
|
|
157
|
+
else {
|
|
158
|
+
this.focusPreviousItem();
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
focusPreviousItem() {
|
|
162
|
+
if (!this.#hasFocusableItems()) {
|
|
163
|
+
return;
|
|
164
|
+
}
|
|
165
|
+
this.menuIndex--;
|
|
166
|
+
const previousItem = this.#getItemByIndex(this.menuIndex);
|
|
167
|
+
if (previousItem && previousItem.isFocusable()) {
|
|
168
|
+
this.#focusItem(previousItem);
|
|
169
|
+
}
|
|
170
|
+
else {
|
|
171
|
+
this.focusPreviousItem();
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
focusNextItem() {
|
|
175
|
+
if (!this.#hasFocusableItems()) {
|
|
176
|
+
return;
|
|
177
|
+
}
|
|
178
|
+
this.menuIndex++;
|
|
179
|
+
const nextItem = this.#getItemByIndex(this.menuIndex);
|
|
180
|
+
if (nextItem && nextItem.isFocusable()) {
|
|
181
|
+
this.#focusItem(nextItem);
|
|
182
|
+
}
|
|
183
|
+
else {
|
|
184
|
+
this.focusNextItem();
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
reset() {
|
|
188
|
+
this.#_menuIndex = -1;
|
|
189
|
+
this.#resetItemsActiveState();
|
|
190
|
+
this.#changeDetector.markForCheck();
|
|
191
|
+
}
|
|
192
|
+
#resetItemsActiveState() {
|
|
193
|
+
this.menuItems.forEach((item) => {
|
|
194
|
+
item.resetState();
|
|
195
|
+
});
|
|
196
|
+
}
|
|
197
|
+
#focusItem(item) {
|
|
198
|
+
this.#resetItemsActiveState();
|
|
199
|
+
item.focusElement(this.useNativeFocus);
|
|
200
|
+
this.menuChanges.emit({
|
|
201
|
+
activeIndex: this.menuIndex,
|
|
202
|
+
});
|
|
203
|
+
}
|
|
204
|
+
#getItemByIndex(index) {
|
|
205
|
+
return this.menuItems.find((_, i) => i === index);
|
|
206
|
+
}
|
|
207
|
+
#selectItemByEventTarget(target) {
|
|
208
|
+
const selectedItem = this.menuItems.find((item, i) => {
|
|
209
|
+
const found = item.elementRef.nativeElement.contains(target);
|
|
210
|
+
if (found) {
|
|
211
|
+
this.menuIndex = i;
|
|
212
|
+
this.menuChanges.next({
|
|
213
|
+
activeIndex: this.menuIndex,
|
|
214
|
+
});
|
|
215
|
+
}
|
|
216
|
+
return found;
|
|
217
|
+
});
|
|
218
|
+
/* istanbul ignore else */
|
|
219
|
+
if (selectedItem) {
|
|
220
|
+
this.menuChanges.next({
|
|
221
|
+
selectedItem,
|
|
222
|
+
});
|
|
223
|
+
}
|
|
224
|
+
}
|
|
225
|
+
#sendMessage(type) {
|
|
226
|
+
this.#dropdownComponent?.messageStream?.next({ type });
|
|
227
|
+
}
|
|
228
|
+
#addEventListeners() {
|
|
229
|
+
const dropdownMenuElement = this.#elementRef.nativeElement;
|
|
230
|
+
observableFromEvent(dropdownMenuElement, 'click')
|
|
231
|
+
.pipe(takeUntil(this.#ngUnsubscribe))
|
|
232
|
+
.subscribe((event) => {
|
|
233
|
+
this.#selectItemByEventTarget(event.target);
|
|
234
|
+
});
|
|
235
|
+
observableFromEvent(dropdownMenuElement, 'keydown')
|
|
236
|
+
.pipe(takeUntil(this.#ngUnsubscribe))
|
|
237
|
+
.subscribe((event) => {
|
|
238
|
+
const key = event.key.toLowerCase();
|
|
239
|
+
switch (key) {
|
|
240
|
+
case 'escape':
|
|
241
|
+
this.#sendMessage(SkyDropdownMessageType.Close);
|
|
242
|
+
this.#sendMessage(SkyDropdownMessageType.FocusTriggerButton);
|
|
243
|
+
event.stopPropagation();
|
|
244
|
+
event.preventDefault();
|
|
245
|
+
break;
|
|
246
|
+
case 'arrowdown':
|
|
247
|
+
case 'down':
|
|
248
|
+
this.focusNextItem();
|
|
249
|
+
event.preventDefault();
|
|
250
|
+
break;
|
|
251
|
+
case 'arrowup':
|
|
252
|
+
case 'up':
|
|
253
|
+
this.focusPreviousItem();
|
|
254
|
+
event.preventDefault();
|
|
255
|
+
break;
|
|
256
|
+
case 'tab':
|
|
257
|
+
if (this.#dropdownComponent?.dismissOnBlur) {
|
|
258
|
+
this.#sendMessage(SkyDropdownMessageType.Close);
|
|
259
|
+
}
|
|
260
|
+
this.#sendMessage(SkyDropdownMessageType.FocusTriggerButton);
|
|
261
|
+
break;
|
|
262
|
+
}
|
|
263
|
+
});
|
|
264
|
+
observableFromEvent(dropdownMenuElement, 'mouseenter')
|
|
265
|
+
.pipe(takeUntil(this.#ngUnsubscribe))
|
|
266
|
+
.subscribe(() => {
|
|
267
|
+
/* istanbul ignore else */
|
|
268
|
+
if (this.#dropdownComponent) {
|
|
269
|
+
this.#dropdownComponent.isMouseEnter = true;
|
|
270
|
+
}
|
|
271
|
+
});
|
|
272
|
+
observableFromEvent(dropdownMenuElement, 'mouseleave')
|
|
273
|
+
.pipe(takeUntil(this.#ngUnsubscribe))
|
|
274
|
+
.subscribe(() => {
|
|
275
|
+
/* istanbul ignore else */
|
|
276
|
+
if (this.#dropdownComponent) {
|
|
277
|
+
this.#dropdownComponent.isMouseEnter = false;
|
|
278
|
+
// Allow the dropdown component to set isMouseEnter before checking if the close action
|
|
279
|
+
// should be taken.
|
|
280
|
+
setTimeout(() => {
|
|
281
|
+
if (this.#dropdownComponent &&
|
|
282
|
+
this.#dropdownComponent.trigger === 'hover' &&
|
|
283
|
+
this.#dropdownComponent.isMouseEnter === false) {
|
|
284
|
+
this.#sendMessage(SkyDropdownMessageType.Close);
|
|
285
|
+
}
|
|
286
|
+
});
|
|
287
|
+
}
|
|
288
|
+
});
|
|
289
|
+
}
|
|
290
|
+
#hasFocusableItems() {
|
|
291
|
+
const found = this.menuItems.find((item) => item.isFocusable());
|
|
292
|
+
return found !== undefined;
|
|
293
|
+
}
|
|
294
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.7", ngImport: i0, type: SkyDropdownMenuComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i0.ElementRef }, { token: i1.SkyDropdownComponent, optional: true }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
295
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.7", type: SkyDropdownMenuComponent, selector: "sky-dropdown-menu", inputs: { ariaLabelledBy: "ariaLabelledBy", ariaRole: "ariaRole", useNativeFocus: "useNativeFocus" }, outputs: { menuChanges: "menuChanges" }, queries: [{ propertyName: "menuItems", predicate: SkyDropdownItemComponent, descendants: true }], ngImport: i0, template: "<div\n class=\"sky-dropdown-menu sky-shadow sky-elevation-4\"\n [attr.aria-labelledby]=\"ariaLabelledBy\"\n [attr.role]=\"ariaRole\"\n [id]=\"dropdownMenuId\"\n>\n <ng-content> </ng-content>\n</div>\n", styles: [".sky-dropdown-menu{overflow-y:auto;overflow-x:hidden;min-height:35px;max-height:calc(50vh - 50px);display:flex;flex-direction:column;background-color:#fff}.sky-dropdown-menu button{overflow:hidden;text-overflow:ellipsis}.sky-theme-modern .sky-dropdown-menu{border-radius:6px;padding:10px 0}.sky-theme-modern.sky-theme-mode-dark .sky-dropdown-menu{background-color:#252a2e;color:#fff}\n"], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
|
|
296
|
+
}
|
|
297
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.7", ngImport: i0, type: SkyDropdownMenuComponent, decorators: [{
|
|
298
|
+
type: Component,
|
|
299
|
+
args: [{ selector: 'sky-dropdown-menu', changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, template: "<div\n class=\"sky-dropdown-menu sky-shadow sky-elevation-4\"\n [attr.aria-labelledby]=\"ariaLabelledBy\"\n [attr.role]=\"ariaRole\"\n [id]=\"dropdownMenuId\"\n>\n <ng-content> </ng-content>\n</div>\n", styles: [".sky-dropdown-menu{overflow-y:auto;overflow-x:hidden;min-height:35px;max-height:calc(50vh - 50px);display:flex;flex-direction:column;background-color:#fff}.sky-dropdown-menu button{overflow:hidden;text-overflow:ellipsis}.sky-theme-modern .sky-dropdown-menu{border-radius:6px;padding:10px 0}.sky-theme-modern.sky-theme-mode-dark .sky-dropdown-menu{background-color:#252a2e;color:#fff}\n"] }]
|
|
300
|
+
}], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }, { type: i0.ElementRef }, { type: i1.SkyDropdownComponent, decorators: [{
|
|
301
|
+
type: Optional
|
|
302
|
+
}] }]; }, propDecorators: { ariaLabelledBy: [{
|
|
303
|
+
type: Input
|
|
304
|
+
}], ariaRole: [{
|
|
305
|
+
type: Input
|
|
306
|
+
}], useNativeFocus: [{
|
|
307
|
+
type: Input
|
|
308
|
+
}], menuChanges: [{
|
|
309
|
+
type: Output
|
|
310
|
+
}], menuItems: [{
|
|
311
|
+
type: ContentChildren,
|
|
312
|
+
args: [SkyDropdownItemComponent, { descendants: true }]
|
|
313
|
+
}] } });
|
|
314
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dropdown-menu.component.js","sourceRoot":"","sources":["../../../../../../../../libs/components/popovers/src/lib/modules/dropdown/dropdown-menu.component.ts","../../../../../../../../libs/components/popovers/src/lib/modules/dropdown/dropdown-menu.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterContentInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ContentChildren,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnDestroy,\n  Optional,\n  Output,\n  QueryList,\n  ViewEncapsulation,\n} from '@angular/core';\n\nimport { Subject, fromEvent as observableFromEvent } from 'rxjs';\nimport { takeUntil } from 'rxjs/operators';\n\nimport { SkyDropdownItemComponent } from './dropdown-item.component';\nimport { SkyDropdownComponent } from './dropdown.component';\nimport { SkyDropdownMenuChange } from './types/dropdown-menu-change';\nimport { SkyDropdownMessage } from './types/dropdown-message';\nimport { SkyDropdownMessageType } from './types/dropdown-message-type';\n\nlet nextId = 0;\n/**\n * Creates a menu that contains dropdown menu items.\n *\n */\n@Component({\n  selector: 'sky-dropdown-menu',\n  templateUrl: './dropdown-menu.component.html',\n  styleUrls: ['./dropdown-menu.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  encapsulation: ViewEncapsulation.None,\n})\nexport class SkyDropdownMenuComponent implements AfterContentInit, OnDestroy {\n  /**\n   * The HTML element ID of the element that labels\n   * the dropdown menu. This sets the dropdown menu's `aria-labelledby` attribute to provide a text equivalent for\n   * [to support accessibility](https://developer.blackbaud.com/skyux/learn/accessibility).\n   * For more information about the `aria-labelledby` attribute, see the [WAI-ARIA definition](https://www.w3.org/TR/wai-aria/#aria-labelledby).\n   */\n  @Input()\n  public ariaLabelledBy: string | undefined;\n\n  /**\n   * The ARIA role for the dropdown menu\n   * [to support accessibility](https://developer.blackbaud.com/skyux/learn/accessibility)\n   * by indicating how the dropdown menu functions and what it controls. The dropdown button\n   * inherits this value to set its `aria-haspopup` property. For information\n   * about how an ARIA role indicates what an item represents on a web page, see the\n   * [WAI-ARIA roles model](https://www.w3.org/WAI/PF/aria/#roles).\n   * @default \"menu\"\n   */\n  @Input()\n  public set ariaRole(value: string | undefined) {\n    this.#_ariaRole = value ?? 'menu';\n  }\n\n  public get ariaRole(): string {\n    return this.#_ariaRole;\n  }\n\n  /**\n   * Whether to use the browser's native focus function when users navigate through menu\n   * items with the keyboard. To disable the native focus function, set this property to `false`.\n   * For example, to let users interact with the dropdown menu but keep the keyboard focus on a\n   * different element, set this property to `false`.\n   * @default true\n   * @internal\n   */\n  @Input()\n  public set useNativeFocus(value: boolean | undefined) {\n    this.#_useNativeFocus = value ?? true;\n  }\n\n  public get useNativeFocus(): boolean {\n    return this.#_useNativeFocus;\n  }\n\n  /**\n   * Fires when the dropdown menu's active index or selected item changes. This event provides an\n   * observable to emit changes, and the response is of\n   * the SkyDropdownMenuChange type.\n   * @internal\n   */\n  @Output()\n  public menuChanges = new EventEmitter<SkyDropdownMenuChange>();\n\n  public dropdownMenuId = `sky-dropdown-menu-${++nextId}`;\n\n  public set menuIndex(value: number) {\n    if (value < 0) {\n      value = this.menuItems.length - 1;\n    }\n\n    if (value >= this.menuItems.length) {\n      value = 0;\n    }\n\n    this.#_menuIndex = value;\n  }\n\n  public get menuIndex(): number {\n    return this.#_menuIndex;\n  }\n\n  @ContentChildren(SkyDropdownItemComponent, { descendants: true })\n  public menuItems!: QueryList<SkyDropdownItemComponent>;\n\n  #ngUnsubscribe = new Subject<void>();\n\n  #_ariaRole = 'menu';\n\n  #_menuIndex = 0;\n\n  #_useNativeFocus = true;\n\n  #changeDetector: ChangeDetectorRef;\n  #elementRef: ElementRef;\n  #dropdownComponent: SkyDropdownComponent | undefined;\n\n  constructor(\n    changeDetector: ChangeDetectorRef,\n    elementRef: ElementRef,\n    @Optional() dropdownComponent?: SkyDropdownComponent\n  ) {\n    this.#changeDetector = changeDetector;\n    this.#elementRef = elementRef;\n    this.#dropdownComponent = dropdownComponent;\n  }\n\n  public ngAfterContentInit(): void {\n    /* istanbul ignore else */\n    if (this.#dropdownComponent) {\n      this.#dropdownComponent.menuId = this.dropdownMenuId;\n      this.#dropdownComponent.menuAriaRole = this.ariaRole;\n\n      this.#dropdownComponent.messageStream\n        ?.pipe(takeUntil(this.#ngUnsubscribe))\n        .subscribe((message: SkyDropdownMessage) => {\n          switch (message.type) {\n            case SkyDropdownMessageType.Open:\n            case SkyDropdownMessageType.Close:\n              this.reset();\n              break;\n\n            case SkyDropdownMessageType.FocusFirstItem:\n              this.focusFirstItem();\n              break;\n\n            case SkyDropdownMessageType.FocusNextItem:\n              this.focusNextItem();\n              break;\n\n            case SkyDropdownMessageType.FocusPreviousItem:\n              this.focusPreviousItem();\n              break;\n\n            case SkyDropdownMessageType.FocusLastItem:\n              this.focusLastItem();\n              break;\n          }\n        });\n\n      this.menuChanges\n        .pipe(takeUntil(this.#ngUnsubscribe))\n        .subscribe((change: SkyDropdownMenuChange) => {\n          // Close the dropdown when a menu item is selected.\n          if (change.selectedItem) {\n            this.#sendMessage(SkyDropdownMessageType.Close);\n            this.#sendMessage(SkyDropdownMessageType.FocusTriggerButton);\n          }\n\n          if (change.items) {\n            // Update the popover style and position whenever the number of items changes.\n            this.#sendMessage(SkyDropdownMessageType.Reposition);\n          }\n        });\n    }\n\n    // Reset dropdown whenever the menu items change.\n    this.menuItems.changes\n      .pipe(takeUntil(this.#ngUnsubscribe))\n      .subscribe((items: QueryList<SkyDropdownItemComponent>) => {\n        this.reset();\n        this.menuChanges.emit({\n          items: items.toArray(),\n        });\n      });\n\n    this.#addEventListeners();\n  }\n\n  public ngOnDestroy(): void {\n    this.#ngUnsubscribe.next();\n    this.#ngUnsubscribe.complete();\n  }\n\n  public focusFirstItem(): void {\n    if (!this.#hasFocusableItems()) {\n      return;\n    }\n\n    this.menuIndex = 0;\n\n    const firstItem = this.#getItemByIndex(this.menuIndex);\n    if (firstItem && firstItem.isFocusable()) {\n      this.#focusItem(firstItem);\n    } else {\n      this.focusNextItem();\n    }\n  }\n\n  public focusLastItem(): void {\n    if (!this.#hasFocusableItems()) {\n      return;\n    }\n\n    this.menuIndex = this.menuItems.length - 1;\n\n    const lastItem = this.#getItemByIndex(this.menuIndex);\n    if (lastItem && lastItem.isFocusable()) {\n      this.#focusItem(lastItem);\n    } else {\n      this.focusPreviousItem();\n    }\n  }\n\n  public focusPreviousItem(): void {\n    if (!this.#hasFocusableItems()) {\n      return;\n    }\n\n    this.menuIndex--;\n\n    const previousItem = this.#getItemByIndex(this.menuIndex);\n    if (previousItem && previousItem.isFocusable()) {\n      this.#focusItem(previousItem);\n    } else {\n      this.focusPreviousItem();\n    }\n  }\n\n  public focusNextItem() {\n    if (!this.#hasFocusableItems()) {\n      return;\n    }\n\n    this.menuIndex++;\n\n    const nextItem = this.#getItemByIndex(this.menuIndex);\n    if (nextItem && nextItem.isFocusable()) {\n      this.#focusItem(nextItem);\n    } else {\n      this.focusNextItem();\n    }\n  }\n\n  public reset(): void {\n    this.#_menuIndex = -1;\n    this.#resetItemsActiveState();\n    this.#changeDetector.markForCheck();\n  }\n\n  #resetItemsActiveState() {\n    this.menuItems.forEach((item: SkyDropdownItemComponent) => {\n      item.resetState();\n    });\n  }\n\n  #focusItem(item: SkyDropdownItemComponent): void {\n    this.#resetItemsActiveState();\n    item.focusElement(this.useNativeFocus);\n    this.menuChanges.emit({\n      activeIndex: this.menuIndex,\n    });\n  }\n\n  #getItemByIndex(index: number): SkyDropdownItemComponent | undefined {\n    return this.menuItems.find((_, i) => i === index);\n  }\n\n  #selectItemByEventTarget(target: EventTarget): void {\n    const selectedItem = this.menuItems.find(\n      (item: SkyDropdownItemComponent, i: number) => {\n        const found = item.elementRef.nativeElement.contains(target);\n\n        if (found) {\n          this.menuIndex = i;\n          this.menuChanges.next({\n            activeIndex: this.menuIndex,\n          });\n        }\n\n        return found;\n      }\n    );\n\n    /* istanbul ignore else */\n    if (selectedItem) {\n      this.menuChanges.next({\n        selectedItem,\n      });\n    }\n  }\n\n  #sendMessage(type: SkyDropdownMessageType): void {\n    this.#dropdownComponent?.messageStream?.next({ type });\n  }\n\n  #addEventListeners(): void {\n    const dropdownMenuElement = this.#elementRef.nativeElement;\n\n    observableFromEvent<MouseEvent>(dropdownMenuElement, 'click')\n      .pipe(takeUntil(this.#ngUnsubscribe))\n      .subscribe((event) => {\n        this.#selectItemByEventTarget(event.target as EventTarget);\n      });\n\n    observableFromEvent<KeyboardEvent>(dropdownMenuElement, 'keydown')\n      .pipe(takeUntil(this.#ngUnsubscribe))\n      .subscribe((event) => {\n        const key = event.key.toLowerCase();\n\n        switch (key) {\n          case 'escape':\n            this.#sendMessage(SkyDropdownMessageType.Close);\n            this.#sendMessage(SkyDropdownMessageType.FocusTriggerButton);\n            event.stopPropagation();\n            event.preventDefault();\n            break;\n\n          case 'arrowdown':\n          case 'down':\n            this.focusNextItem();\n            event.preventDefault();\n            break;\n\n          case 'arrowup':\n          case 'up':\n            this.focusPreviousItem();\n            event.preventDefault();\n            break;\n\n          case 'tab':\n            if (this.#dropdownComponent?.dismissOnBlur) {\n              this.#sendMessage(SkyDropdownMessageType.Close);\n            }\n            this.#sendMessage(SkyDropdownMessageType.FocusTriggerButton);\n            break;\n        }\n      });\n\n    observableFromEvent(dropdownMenuElement, 'mouseenter')\n      .pipe(takeUntil(this.#ngUnsubscribe))\n      .subscribe(() => {\n        /* istanbul ignore else */\n        if (this.#dropdownComponent) {\n          this.#dropdownComponent.isMouseEnter = true;\n        }\n      });\n\n    observableFromEvent(dropdownMenuElement, 'mouseleave')\n      .pipe(takeUntil(this.#ngUnsubscribe))\n      .subscribe(() => {\n        /* istanbul ignore else */\n        if (this.#dropdownComponent) {\n          this.#dropdownComponent.isMouseEnter = false;\n          // Allow the dropdown component to set isMouseEnter before checking if the close action\n          // should be taken.\n          setTimeout(() => {\n            if (\n              this.#dropdownComponent &&\n              this.#dropdownComponent.trigger === 'hover' &&\n              this.#dropdownComponent.isMouseEnter === false\n            ) {\n              this.#sendMessage(SkyDropdownMessageType.Close);\n            }\n          });\n        }\n      });\n  }\n\n  #hasFocusableItems(): boolean {\n    const found = this.menuItems.find((item) => item.isFocusable());\n    return found !== undefined;\n  }\n}\n","<div\n  class=\"sky-dropdown-menu sky-shadow sky-elevation-4\"\n  [attr.aria-labelledby]=\"ariaLabelledBy\"\n  [attr.role]=\"ariaRole\"\n  [id]=\"dropdownMenuId\"\n>\n  <ng-content> </ng-content>\n</div>\n"]}
|