@design-system-rte/angular 1.2.1-rc5 → 1.4.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/esm2022/lib/components/dropdown/dropdown-item/dropdown-item.component.mjs +157 -14
- package/esm2022/lib/components/dropdown/dropdown-menu/dropdown-menu.component.mjs +71 -6
- package/esm2022/lib/components/dropdown/dropdown.directive.mjs +48 -39
- package/esm2022/lib/components/dropdown/dropdown.types.mjs +2 -0
- package/esm2022/lib/components/dropdown/index.mjs +1 -3
- package/esm2022/lib/components/icon/icon-map.mjs +112 -112
- package/esm2022/lib/components/modal/modal.component.mjs +3 -3
- package/esm2022/lib/components/segmented-control/segmented-control.component.mjs +5 -3
- package/esm2022/lib/components/side-nav/side-nav.component.mjs +1 -1
- package/esm2022/lib/components/stepper/stepper.component.mjs +3 -3
- package/esm2022/lib/components/tab/tab-item/tab-item.component.mjs +3 -3
- package/esm2022/lib/components/tab/tab.component.mjs +1 -1
- package/esm2022/lib/components/tag/tag.component.mjs +4 -3
- package/esm2022/public-api.mjs +9 -1
- package/fesm2022/design-system-rte-angular.mjs +431 -219
- package/fesm2022/design-system-rte-angular.mjs.map +1 -1
- package/lib/components/assistive-text/assistive-text.component.d.ts +1 -1
- package/lib/components/chip/chip.component.d.ts +1 -1
- package/lib/components/dropdown/dropdown-item/dropdown-item.component.d.ts +28 -16
- package/lib/components/dropdown/dropdown-menu/dropdown-menu.component.d.ts +14 -3
- package/lib/components/dropdown/dropdown.directive.d.ts +8 -4
- package/lib/components/dropdown/dropdown.types.d.ts +30 -0
- package/lib/components/dropdown/index.d.ts +1 -0
- package/lib/components/segmented-control/segmented-control.component.d.ts +3 -1
- package/lib/components/side-nav/side-nav.component.d.ts +2 -2
- package/lib/components/tab/tab.component.d.ts +1 -1
- package/lib/components/tag/tag.component.d.ts +2 -1
- package/lib/components/text-input/base-text-input/base-text-input.component.d.ts +1 -1
- package/package.json +2 -2
- package/public-api.d.ts +8 -0
|
@@ -1,17 +1,34 @@
|
|
|
1
1
|
import { CommonModule } from "@angular/common";
|
|
2
|
-
import { Component, computed, input, output } from "@angular/core";
|
|
2
|
+
import { ChangeDetectorRef, Component, computed, ElementRef, inject, input, output, signal, } from "@angular/core";
|
|
3
3
|
import { shouldDisplayBadge } from "@design-system-rte/core/components/badge/badge.utils";
|
|
4
|
+
import { DropdownManager } from "@design-system-rte/core/components/dropdown/DropdownManager";
|
|
4
5
|
import { ENTER_KEY, SPACE_KEY } from "@design-system-rte/core/constants/keyboard/keyboard.constants";
|
|
5
6
|
import { BadgeComponent } from "../../badge/badge.component";
|
|
6
7
|
import { DividerComponent } from "../../divider/divider.component";
|
|
7
8
|
import { IconComponent } from "../../icon/icon.component";
|
|
9
|
+
import { focusDropdownFirstElement } from "../dropdown.utils";
|
|
8
10
|
import * as i0 from "@angular/core";
|
|
9
11
|
import * as i1 from "@angular/common";
|
|
12
|
+
const SUB_MENU_CLOSE_DELAY_MS = 300;
|
|
10
13
|
export class DropdownItemComponent {
|
|
11
14
|
constructor() {
|
|
15
|
+
this.elementRef = inject(ElementRef);
|
|
16
|
+
this.cdr = inject(ChangeDetectorRef);
|
|
12
17
|
this.item = input();
|
|
13
18
|
this.menuId = input();
|
|
14
19
|
this.itemEvent = output();
|
|
20
|
+
this.submenuRequest = output();
|
|
21
|
+
this.subMenuOpen = signal(false);
|
|
22
|
+
this.subMenuRef = null;
|
|
23
|
+
this.closeSubMenuTimeout = null;
|
|
24
|
+
this.subMenuSubscriptions = [];
|
|
25
|
+
this.dropdownManagerUnsubscribe = null;
|
|
26
|
+
this.hasChildren = computed(() => (this.item()?.children?.length ?? 0) > 0);
|
|
27
|
+
this.childDropdownId = computed(() => {
|
|
28
|
+
const menuId = this.menuId();
|
|
29
|
+
const label = this.item()?.label ?? "";
|
|
30
|
+
return menuId && label ? `${menuId}-${label.replace(/\s+/g, "")}` : "";
|
|
31
|
+
});
|
|
15
32
|
this.shouldDisplayBadge = computed(() => {
|
|
16
33
|
const item = this.item();
|
|
17
34
|
return shouldDisplayBadge({
|
|
@@ -21,6 +38,7 @@ export class DropdownItemComponent {
|
|
|
21
38
|
badgeIcon: item?.badgeIcon,
|
|
22
39
|
});
|
|
23
40
|
});
|
|
41
|
+
this.boundHandleSubMenuMouseEnter = () => this.handleSubMenuMouseEnter();
|
|
24
42
|
}
|
|
25
43
|
handleClick(event) {
|
|
26
44
|
if (this.item()?.disabled) {
|
|
@@ -28,31 +46,156 @@ export class DropdownItemComponent {
|
|
|
28
46
|
event.stopPropagation();
|
|
29
47
|
return;
|
|
30
48
|
}
|
|
49
|
+
if (this.hasChildren()) {
|
|
50
|
+
event.preventDefault();
|
|
51
|
+
event.stopPropagation();
|
|
52
|
+
this.openSubMenu();
|
|
53
|
+
return;
|
|
54
|
+
}
|
|
31
55
|
this.itemEvent.emit({
|
|
32
56
|
event,
|
|
33
57
|
id: this.item()?.id || this.item()?.label || "",
|
|
58
|
+
item: this.item(),
|
|
34
59
|
});
|
|
35
60
|
}
|
|
36
61
|
handleKeyDown(event) {
|
|
37
62
|
event.preventDefault();
|
|
38
|
-
if ([SPACE_KEY, ENTER_KEY].includes(event.key)) {
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
63
|
+
if (this.item()?.link && [SPACE_KEY, ENTER_KEY].includes(event.key)) {
|
|
64
|
+
const link = event.target.closest("li")?.querySelector("a");
|
|
65
|
+
link?.click();
|
|
66
|
+
return;
|
|
67
|
+
}
|
|
68
|
+
if (this.hasChildren()) {
|
|
69
|
+
if (event.key === SPACE_KEY) {
|
|
70
|
+
this.openSubMenu();
|
|
71
|
+
const childId = this.childDropdownId();
|
|
72
|
+
if (childId) {
|
|
73
|
+
setTimeout(() => focusDropdownFirstElement(childId), 0);
|
|
74
|
+
}
|
|
48
75
|
}
|
|
76
|
+
return;
|
|
77
|
+
}
|
|
78
|
+
if ([SPACE_KEY, ENTER_KEY].includes(event.key)) {
|
|
79
|
+
this.itemEvent.emit({
|
|
80
|
+
event,
|
|
81
|
+
id: this.item()?.id || this.item()?.label || "",
|
|
82
|
+
item: this.item(),
|
|
83
|
+
});
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
handleMouseEnter() {
|
|
87
|
+
if (this.item()?.disabled || !this.hasChildren())
|
|
88
|
+
return;
|
|
89
|
+
const menuId = this.menuId();
|
|
90
|
+
const childId = this.childDropdownId();
|
|
91
|
+
if (menuId) {
|
|
92
|
+
DropdownManager.closeSubMenus(menuId, childId);
|
|
93
|
+
this.openSubMenu();
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
handleFocus() {
|
|
97
|
+
if (this.item()?.disabled || !this.hasChildren())
|
|
98
|
+
return;
|
|
99
|
+
const menuId = this.menuId();
|
|
100
|
+
const childId = this.childDropdownId();
|
|
101
|
+
if (menuId) {
|
|
102
|
+
DropdownManager.closeSubMenus(menuId, childId);
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
handleMouseLeave() {
|
|
106
|
+
if (!this.hasChildren())
|
|
107
|
+
return;
|
|
108
|
+
this.scheduleCloseSubMenu();
|
|
109
|
+
}
|
|
110
|
+
handleSubMenuMouseEnter() {
|
|
111
|
+
this.cancelCloseSubMenu();
|
|
112
|
+
}
|
|
113
|
+
openSubMenuForKeyboard() {
|
|
114
|
+
this.openSubMenu();
|
|
115
|
+
const childId = this.childDropdownId();
|
|
116
|
+
if (childId) {
|
|
117
|
+
setTimeout(() => focusDropdownFirstElement(childId), 0);
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
openSubMenu() {
|
|
121
|
+
if (this.subMenuRef)
|
|
122
|
+
return;
|
|
123
|
+
const children = this.item()?.children;
|
|
124
|
+
const childId = this.childDropdownId();
|
|
125
|
+
if (!children?.length || !childId)
|
|
126
|
+
return;
|
|
127
|
+
const triggerElement = this.elementRef.nativeElement.querySelector("li.rte-dropdown-item");
|
|
128
|
+
if (!triggerElement)
|
|
129
|
+
return;
|
|
130
|
+
DropdownManager.open(childId);
|
|
131
|
+
this.submenuRequest.emit({
|
|
132
|
+
children,
|
|
133
|
+
childId,
|
|
134
|
+
triggerElement,
|
|
135
|
+
onCreated: (result) => this.wireSubmenu(result),
|
|
136
|
+
});
|
|
137
|
+
}
|
|
138
|
+
wireSubmenu(result) {
|
|
139
|
+
if (this.subMenuRef)
|
|
140
|
+
return;
|
|
141
|
+
this.subMenuRef = result.componentRef;
|
|
142
|
+
const hostElement = result.hostElement;
|
|
143
|
+
const childId = this.childDropdownId();
|
|
144
|
+
this.dropdownManagerUnsubscribe = DropdownManager.subscribe(childId ?? "", () => this.destroySubMenu());
|
|
145
|
+
const menuInstance = this.subMenuRef.instance;
|
|
146
|
+
const itemSub = menuInstance.itemEvent.subscribe((emittedItemEvent) => {
|
|
147
|
+
this.itemEvent.emit({ ...emittedItemEvent, item: emittedItemEvent.item });
|
|
148
|
+
});
|
|
149
|
+
const closeSub = menuInstance.closingMenu.subscribe(() => {
|
|
150
|
+
this.destroySubMenu();
|
|
151
|
+
});
|
|
152
|
+
this.subMenuSubscriptions.push(() => {
|
|
153
|
+
itemSub.unsubscribe();
|
|
154
|
+
closeSub.unsubscribe();
|
|
155
|
+
});
|
|
156
|
+
hostElement.addEventListener("mouseenter", this.boundHandleSubMenuMouseEnter);
|
|
157
|
+
this.subMenuOpen.set(true);
|
|
158
|
+
this.cdr.markForCheck();
|
|
159
|
+
}
|
|
160
|
+
scheduleCloseSubMenu() {
|
|
161
|
+
this.cancelCloseSubMenu();
|
|
162
|
+
this.closeSubMenuTimeout = setTimeout(() => {
|
|
163
|
+
this.closeSubMenuTimeout = null;
|
|
164
|
+
this.destroySubMenu();
|
|
165
|
+
}, SUB_MENU_CLOSE_DELAY_MS);
|
|
166
|
+
}
|
|
167
|
+
cancelCloseSubMenu() {
|
|
168
|
+
if (this.closeSubMenuTimeout) {
|
|
169
|
+
clearTimeout(this.closeSubMenuTimeout);
|
|
170
|
+
this.closeSubMenuTimeout = null;
|
|
49
171
|
}
|
|
50
172
|
}
|
|
173
|
+
destroySubMenu() {
|
|
174
|
+
this.cancelCloseSubMenu();
|
|
175
|
+
this.subMenuSubscriptions.forEach((unsubscribe) => unsubscribe());
|
|
176
|
+
this.subMenuSubscriptions = [];
|
|
177
|
+
if (this.subMenuRef) {
|
|
178
|
+
const hostElement = this.subMenuRef.location.nativeElement;
|
|
179
|
+
hostElement.removeEventListener("mouseenter", this.boundHandleSubMenuMouseEnter);
|
|
180
|
+
const childId = this.childDropdownId();
|
|
181
|
+
this.dropdownManagerUnsubscribe?.();
|
|
182
|
+
this.dropdownManagerUnsubscribe = null;
|
|
183
|
+
if (childId)
|
|
184
|
+
DropdownManager.close(childId);
|
|
185
|
+
this.subMenuRef.destroy();
|
|
186
|
+
this.subMenuRef = null;
|
|
187
|
+
}
|
|
188
|
+
this.subMenuOpen.set(false);
|
|
189
|
+
this.cdr.markForCheck();
|
|
190
|
+
}
|
|
191
|
+
ngOnDestroy() {
|
|
192
|
+
this.destroySubMenu();
|
|
193
|
+
}
|
|
51
194
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DropdownItemComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
52
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "18.2.14", type: DropdownItemComponent, isStandalone: true, selector: "rte-dropdown-item", inputs: { item: { classPropertyName: "item", publicName: "item", isSignal: true, isRequired: false, transformFunction: null }, menuId: { classPropertyName: "menuId", publicName: "menuId", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { itemEvent: "itemEvent" }, ngImport: i0, template: "<ng-container>\n <li\n #itemRef\n class=\"rte-dropdown-item\"\n role=\"menuitem\"\n tabindex=\"0\"\n [attr.data-disabled]=\"item()?.disabled\"\n [attr.data-active]=\"item()?.selected\"\n (click)=\"handleClick($event)\"\n (keydown)=\"handleKeyDown($event)\"\n >\n <span *ngIf=\"item()?.hasIndent && !item()?.leftIcon\" style=\"width: 20px\"></span>\n <rte-icon *ngIf=\"item()?.leftIcon\" [name]=\"item()?.leftIcon ?? ''\" />\n <ng-container [ngSwitch]=\"!!item()?.link\">\n <a *ngSwitchCase=\"true\" style=\"flex: 2; text-decoration: none; color: inherit\" [href]=\"item()?.link\"\n >{{ item()?.label }}\n </a>\n <span *ngSwitchDefault style=\"flex: 2\">{{ item()?.label }}</span>\n </ng-container>\n <div *ngIf=\"item()?.trailingText\">{{ item()?.trailingText }}</div>\n <rte-badge\n *ngIf=\"shouldDisplayBadge()\"\n [badgeType]=\"item()?.badgeType!\"\n [badgeContent]=\"item()?.badgeContent!\"\n [count]=\"item()?.badgeCount\"\n [simpleBadge]=\"true\"\n [icon]=\"item()?.badgeIcon ?? ''\"\n [badgeSize]=\"item()?.badgeSize!\"\n ></rte-badge>\n </li>\n <div *ngIf=\"item()?.hasSeparator\" class=\"dropdown-divider\">\n <rte-divider />\n </div>\n</ng-container>\n", styles: [".rte-dropdown-item{font-family:Arial;font-size:14px;font-weight:400;line-height:20px;letter-spacing:0px;color:var(--content-secondary);display:flex;height:40px;padding:8px 12px;align-items:center;gap:12px;align-self:stretch;cursor:pointer;box-sizing:border-box}.rte-dropdown-item span{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.rte-dropdown-item:hover{background-color:var(--background-hover)}.rte-dropdown-item[data-disabled=true]{cursor:not-allowed;color:var(--content-disabled);background:var(--background-disabled)}.rte-dropdown-item[data-active=true]{background-color:var(--background-brand-inverse-pressed)}.rte-dropdown-item:focus-visible{outline:none;background-color:var(--background-hover)}.dropdown-divider{width:100%;padding:8px 0}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { kind: "directive", type: i1.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { kind: "directive", type: i1.NgSwitchDefault, selector: "[ngSwitchDefault]" }, { kind: "component", type: IconComponent, selector: "rte-icon", inputs: ["name", "size", "color", "classes", "appearance"] }, { kind: "component", type: DividerComponent, selector: "rte-divider", inputs: ["orientation", "thickness", "appearance", "endPoint"] }, { kind: "component", type: BadgeComponent, selector: "rte-badge", inputs: ["badgeType", "badgeSize", "badgeContent", "count", "icon", "simpleBadge"] }] }); }
|
|
195
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "18.2.14", type: DropdownItemComponent, isStandalone: true, selector: "rte-dropdown-item", inputs: { item: { classPropertyName: "item", publicName: "item", isSignal: true, isRequired: false, transformFunction: null }, menuId: { classPropertyName: "menuId", publicName: "menuId", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { itemEvent: "itemEvent", submenuRequest: "submenuRequest" }, ngImport: i0, template: "<ng-container>\n <li\n #itemRef\n class=\"rte-dropdown-item\"\n role=\"menuitem\"\n tabindex=\"0\"\n [attr.data-disabled]=\"item()?.disabled\"\n [attr.data-active]=\"hasChildren() ? subMenuOpen() : item()?.selected\"\n [attr.aria-haspopup]=\"hasChildren() ? 'menu' : null\"\n [attr.aria-expanded]=\"hasChildren() ? subMenuOpen() : null\"\n (click)=\"handleClick($event)\"\n (keydown)=\"handleKeyDown($event)\"\n (mouseenter)=\"handleMouseEnter()\"\n (mouseleave)=\"handleMouseLeave()\"\n (focus)=\"handleFocus()\"\n >\n <span *ngIf=\"item()?.hasIndent && !item()?.leftIcon\" style=\"width: 20px\"></span>\n <rte-icon *ngIf=\"item()?.leftIcon\" [name]=\"item()?.leftIcon ?? ''\" />\n <ng-container [ngSwitch]=\"!!item()?.link\">\n <a *ngSwitchCase=\"true\" style=\"flex: 2; text-decoration: none; color: inherit\" [href]=\"item()?.link\"\n >{{ item()?.label }}\n </a>\n <span *ngSwitchDefault style=\"flex: 2\">{{ item()?.label }}</span>\n </ng-container>\n <rte-icon *ngIf=\"hasChildren()\" name=\"arrow-chevron-right\" />\n <div *ngIf=\"item()?.trailingText\">{{ item()?.trailingText }}</div>\n <rte-badge\n *ngIf=\"shouldDisplayBadge()\"\n [badgeType]=\"item()?.badgeType!\"\n [badgeContent]=\"item()?.badgeContent!\"\n [count]=\"item()?.badgeCount\"\n [simpleBadge]=\"true\"\n [icon]=\"item()?.badgeIcon ?? ''\"\n [badgeSize]=\"item()?.badgeSize!\"\n ></rte-badge>\n </li>\n <div *ngIf=\"item()?.hasSeparator\" class=\"dropdown-divider\">\n <rte-divider />\n </div>\n</ng-container>\n", styles: [".rte-dropdown-item{font-family:Arial;font-size:14px;font-weight:400;line-height:20px;letter-spacing:0px;color:var(--content-secondary);display:flex;height:40px;padding:8px 12px;align-items:center;gap:12px;align-self:stretch;cursor:pointer;box-sizing:border-box}.rte-dropdown-item span{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.rte-dropdown-item:hover{background-color:var(--background-hover)}.rte-dropdown-item[data-disabled=true]{cursor:not-allowed;color:var(--content-disabled);background:var(--background-disabled)}.rte-dropdown-item[data-active=true]{background-color:var(--background-brand-inverse-pressed)}.rte-dropdown-item:focus-visible{outline:none;background-color:var(--background-hover)}.dropdown-divider{width:100%;padding:8px 0}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { kind: "directive", type: i1.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { kind: "directive", type: i1.NgSwitchDefault, selector: "[ngSwitchDefault]" }, { kind: "component", type: IconComponent, selector: "rte-icon", inputs: ["name", "size", "color", "classes", "appearance"] }, { kind: "component", type: DividerComponent, selector: "rte-divider", inputs: ["orientation", "thickness", "appearance", "endPoint"] }, { kind: "component", type: BadgeComponent, selector: "rte-badge", inputs: ["badgeType", "badgeSize", "badgeContent", "count", "icon", "simpleBadge"] }] }); }
|
|
53
196
|
}
|
|
54
197
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DropdownItemComponent, decorators: [{
|
|
55
198
|
type: Component,
|
|
56
|
-
args: [{ selector: "rte-dropdown-item", imports: [CommonModule, IconComponent, DividerComponent, BadgeComponent], standalone: true, template: "<ng-container>\n <li\n #itemRef\n class=\"rte-dropdown-item\"\n role=\"menuitem\"\n tabindex=\"0\"\n [attr.data-disabled]=\"item()?.disabled\"\n [attr.data-active]=\"item()?.selected\"\n (click)=\"handleClick($event)\"\n (keydown)=\"handleKeyDown($event)\"\n >\n <span *ngIf=\"item()?.hasIndent && !item()?.leftIcon\" style=\"width: 20px\"></span>\n <rte-icon *ngIf=\"item()?.leftIcon\" [name]=\"item()?.leftIcon ?? ''\" />\n <ng-container [ngSwitch]=\"!!item()?.link\">\n <a *ngSwitchCase=\"true\" style=\"flex: 2; text-decoration: none; color: inherit\" [href]=\"item()?.link\"\n >{{ item()?.label }}\n </a>\n <span *ngSwitchDefault style=\"flex: 2\">{{ item()?.label }}</span>\n </ng-container>\n <div *ngIf=\"item()?.trailingText\">{{ item()?.trailingText }}</div>\n <rte-badge\n *ngIf=\"shouldDisplayBadge()\"\n [badgeType]=\"item()?.badgeType!\"\n [badgeContent]=\"item()?.badgeContent!\"\n [count]=\"item()?.badgeCount\"\n [simpleBadge]=\"true\"\n [icon]=\"item()?.badgeIcon ?? ''\"\n [badgeSize]=\"item()?.badgeSize!\"\n ></rte-badge>\n </li>\n <div *ngIf=\"item()?.hasSeparator\" class=\"dropdown-divider\">\n <rte-divider />\n </div>\n</ng-container>\n", styles: [".rte-dropdown-item{font-family:Arial;font-size:14px;font-weight:400;line-height:20px;letter-spacing:0px;color:var(--content-secondary);display:flex;height:40px;padding:8px 12px;align-items:center;gap:12px;align-self:stretch;cursor:pointer;box-sizing:border-box}.rte-dropdown-item span{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.rte-dropdown-item:hover{background-color:var(--background-hover)}.rte-dropdown-item[data-disabled=true]{cursor:not-allowed;color:var(--content-disabled);background:var(--background-disabled)}.rte-dropdown-item[data-active=true]{background-color:var(--background-brand-inverse-pressed)}.rte-dropdown-item:focus-visible{outline:none;background-color:var(--background-hover)}.dropdown-divider{width:100%;padding:8px 0}\n"] }]
|
|
199
|
+
args: [{ selector: "rte-dropdown-item", imports: [CommonModule, IconComponent, DividerComponent, BadgeComponent], standalone: true, template: "<ng-container>\n <li\n #itemRef\n class=\"rte-dropdown-item\"\n role=\"menuitem\"\n tabindex=\"0\"\n [attr.data-disabled]=\"item()?.disabled\"\n [attr.data-active]=\"hasChildren() ? subMenuOpen() : item()?.selected\"\n [attr.aria-haspopup]=\"hasChildren() ? 'menu' : null\"\n [attr.aria-expanded]=\"hasChildren() ? subMenuOpen() : null\"\n (click)=\"handleClick($event)\"\n (keydown)=\"handleKeyDown($event)\"\n (mouseenter)=\"handleMouseEnter()\"\n (mouseleave)=\"handleMouseLeave()\"\n (focus)=\"handleFocus()\"\n >\n <span *ngIf=\"item()?.hasIndent && !item()?.leftIcon\" style=\"width: 20px\"></span>\n <rte-icon *ngIf=\"item()?.leftIcon\" [name]=\"item()?.leftIcon ?? ''\" />\n <ng-container [ngSwitch]=\"!!item()?.link\">\n <a *ngSwitchCase=\"true\" style=\"flex: 2; text-decoration: none; color: inherit\" [href]=\"item()?.link\"\n >{{ item()?.label }}\n </a>\n <span *ngSwitchDefault style=\"flex: 2\">{{ item()?.label }}</span>\n </ng-container>\n <rte-icon *ngIf=\"hasChildren()\" name=\"arrow-chevron-right\" />\n <div *ngIf=\"item()?.trailingText\">{{ item()?.trailingText }}</div>\n <rte-badge\n *ngIf=\"shouldDisplayBadge()\"\n [badgeType]=\"item()?.badgeType!\"\n [badgeContent]=\"item()?.badgeContent!\"\n [count]=\"item()?.badgeCount\"\n [simpleBadge]=\"true\"\n [icon]=\"item()?.badgeIcon ?? ''\"\n [badgeSize]=\"item()?.badgeSize!\"\n ></rte-badge>\n </li>\n <div *ngIf=\"item()?.hasSeparator\" class=\"dropdown-divider\">\n <rte-divider />\n </div>\n</ng-container>\n", styles: [".rte-dropdown-item{font-family:Arial;font-size:14px;font-weight:400;line-height:20px;letter-spacing:0px;color:var(--content-secondary);display:flex;height:40px;padding:8px 12px;align-items:center;gap:12px;align-self:stretch;cursor:pointer;box-sizing:border-box}.rte-dropdown-item span{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.rte-dropdown-item:hover{background-color:var(--background-hover)}.rte-dropdown-item[data-disabled=true]{cursor:not-allowed;color:var(--content-disabled);background:var(--background-disabled)}.rte-dropdown-item[data-active=true]{background-color:var(--background-brand-inverse-pressed)}.rte-dropdown-item:focus-visible{outline:none;background-color:var(--background-hover)}.dropdown-divider{width:100%;padding:8px 0}\n"] }]
|
|
57
200
|
}] });
|
|
58
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZHJvcGRvd24taXRlbS5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9kcy1ydGUtbGliL3NyYy9saWIvY29tcG9uZW50cy9kcm9wZG93bi9kcm9wZG93bi1pdGVtL2Ryb3Bkb3duLWl0ZW0uY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvZHMtcnRlLWxpYi9zcmMvbGliL2NvbXBvbmVudHMvZHJvcGRvd24vZHJvcGRvd24taXRlbS9kcm9wZG93bi1pdGVtLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsU0FBUyxFQUFFLFFBQVEsRUFBZ0IsS0FBSyxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUNqRixPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxzREFBc0QsQ0FBQztBQUUxRixPQUFPLEVBQUUsU0FBUyxFQUFFLFNBQVMsRUFBRSxNQUFNLCtEQUErRCxDQUFDO0FBRXJHLE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSw2QkFBNkIsQ0FBQztBQUM3RCxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSxpQ0FBaUMsQ0FBQztBQUNuRSxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sMkJBQTJCLENBQUM7OztBQXNCMUQsTUFBTSxPQUFPLHFCQUFxQjtJQVBsQztRQVFXLFNBQUksR0FBRyxLQUFLLEVBQXNCLENBQUM7UUFDbkMsV0FBTSxHQUFHLEtBQUssRUFBVSxDQUFDO1FBQ3pCLGNBQVMsR0FBRyxNQUFNLEVBQWdDLENBQUM7UUFFbkQsdUJBQWtCLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRTtZQUMxQyxNQUFNLElBQUksR0FBRyxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUM7WUFDekIsT0FBTyxrQkFBa0IsQ0FBQztnQkFDeEIsU0FBUyxFQUFFLENBQUMsQ0FBQyxJQUFJLEVBQUUsU0FBUztnQkFDNUIsWUFBWSxFQUFFLElBQUksRUFBRSxZQUFZO2dCQUNoQyxVQUFVLEVBQUUsSUFBSSxFQUFFLFVBQVU7Z0JBQzVCLFNBQVMsRUFBRSxJQUFJLEVBQUUsU0FBUzthQUMzQixDQUFDLENBQUM7UUFDTCxDQUFDLENBQUMsQ0FBQztLQTZCSjtJQTNCQyxXQUFXLENBQUMsS0FBWTtRQUN0QixJQUFJLElBQUksQ0FBQyxJQUFJLEVBQUUsRUFBRSxRQUFRLEVBQUUsQ0FBQztZQUMxQixLQUFLLENBQUMsY0FBYyxFQUFFLENBQUM7WUFDdkIsS0FBSyxDQUFDLGVBQWUsRUFBRSxDQUFDO1lBQ3hCLE9BQU87UUFDVCxDQUFDO1FBQ0QsSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUM7WUFDbEIsS0FBSztZQUNMLEVBQUUsRUFBRSxJQUFJLENBQUMsSUFBSSxFQUFFLEVBQUUsRUFBRSxJQUFJLElBQUksQ0FBQyxJQUFJLEVBQUUsRUFBRSxLQUFLLElBQUksRUFBRTtTQUNoRCxDQUFDLENBQUM7SUFDTCxDQUFDO0lBRUQsYUFBYSxDQUFDLEtBQW9CO1FBQ2hDLEtBQUssQ0FBQyxjQUFjLEVBQUUsQ0FBQztRQUV2QixJQUFJLENBQUMsU0FBUyxFQUFFLFNBQVMsQ0FBQyxDQUFDLFFBQVEsQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQztZQUMvQyxJQUFJLElBQUksQ0FBQyxJQUFJLEVBQUUsRUFBRSxJQUFJLEVBQUUsQ0FBQztnQkFDdEIsTUFBTSxJQUFJLEdBQUksS0FBSyxDQUFDLE1BQXNCLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxFQUFFLGFBQWEsQ0FBQyxHQUFHLENBQUMsQ0FBQztnQkFDN0UsSUFBSSxFQUFFLEtBQUssRUFBRSxDQUFDO1lBQ2hCLENBQUM7aUJBQU0sQ0FBQztnQkFDTixJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQztvQkFDbEIsS0FBSztvQkFDTCxFQUFFLEVBQUUsSUFBSSxDQUFDLElBQUksRUFBRSxFQUFFLEVBQUUsSUFBSSxJQUFJLENBQUMsSUFBSSxFQUFFLEVBQUUsS0FBSyxJQUFJLEVBQUU7aUJBQ2hELENBQUMsQ0FBQztZQUNMLENBQUM7UUFDSCxDQUFDO0lBQ0gsQ0FBQzsrR0F6Q1UscUJBQXFCO21HQUFyQixxQkFBcUIsOFdDOUJsQywwdkNBa0NBLGt6QkRUWSxZQUFZLCtZQUFFLGFBQWEsaUhBQUUsZ0JBQWdCLHdIQUFFLGNBQWM7OzRGQUs1RCxxQkFBcUI7a0JBUGpDLFNBQVM7K0JBQ0UsbUJBQW1CLFdBQ3BCLENBQUMsWUFBWSxFQUFFLGFBQWEsRUFBRSxnQkFBZ0IsRUFBRSxjQUFjLENBQUMsY0FDNUQsSUFBSSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gXCJAYW5ndWxhci9jb21tb25cIjtcbmltcG9ydCB7IENvbXBvbmVudCwgY29tcHV0ZWQsIEV2ZW50RW1pdHRlciwgaW5wdXQsIG91dHB1dCB9IGZyb20gXCJAYW5ndWxhci9jb3JlXCI7XG5pbXBvcnQgeyBzaG91bGREaXNwbGF5QmFkZ2UgfSBmcm9tIFwiQGRlc2lnbi1zeXN0ZW0tcnRlL2NvcmUvY29tcG9uZW50cy9iYWRnZS9iYWRnZS51dGlsc1wiO1xuaW1wb3J0IHsgRHJvcGRvd25JdGVtUHJvcHMgfSBmcm9tIFwiQGRlc2lnbi1zeXN0ZW0tcnRlL2NvcmUvY29tcG9uZW50cy9kcm9wZG93bi9kcm9wZG93bi5pbnRlcmZhY2VcIjtcbmltcG9ydCB7IEVOVEVSX0tFWSwgU1BBQ0VfS0VZIH0gZnJvbSBcIkBkZXNpZ24tc3lzdGVtLXJ0ZS9jb3JlL2NvbnN0YW50cy9rZXlib2FyZC9rZXlib2FyZC5jb25zdGFudHNcIjtcblxuaW1wb3J0IHsgQmFkZ2VDb21wb25lbnQgfSBmcm9tIFwiLi4vLi4vYmFkZ2UvYmFkZ2UuY29tcG9uZW50XCI7XG5pbXBvcnQgeyBEaXZpZGVyQ29tcG9uZW50IH0gZnJvbSBcIi4uLy4uL2RpdmlkZXIvZGl2aWRlci5jb21wb25lbnRcIjtcbmltcG9ydCB7IEljb25Db21wb25lbnQgfSBmcm9tIFwiLi4vLi4vaWNvbi9pY29uLmNvbXBvbmVudFwiO1xuXG5leHBvcnQgaW50ZXJmYWNlIERyb3Bkb3duSXRlbUNvbmZpZyBleHRlbmRzIE9taXQ8RHJvcGRvd25JdGVtUHJvcHMsIFwib25DbGlja1wiPiB7XG4gIGlkPzogc3RyaW5nO1xuICBzZWxlY3RlZD86IGJvb2xlYW47XG4gIGxhYmVsOiBzdHJpbmc7XG4gIGxlZnRJY29uPzogc3RyaW5nO1xuICB0cmFpbGluZ1RleHQ/OiBzdHJpbmc7XG4gIGRpc2FibGVkPzogYm9vbGVhbjtcbiAgaGFzU2VwYXJhdG9yPzogYm9vbGVhbjtcbiAgaGFzSW5kZW50PzogYm9vbGVhbjtcbiAgbGluaz86IHN0cmluZztcbiAgY2xpY2s/OiBFdmVudEVtaXR0ZXI8RXZlbnQ+O1xufVxuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6IFwicnRlLWRyb3Bkb3duLWl0ZW1cIixcbiAgaW1wb3J0czogW0NvbW1vbk1vZHVsZSwgSWNvbkNvbXBvbmVudCwgRGl2aWRlckNvbXBvbmVudCwgQmFkZ2VDb21wb25lbnRdLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICB0ZW1wbGF0ZVVybDogXCIuL2Ryb3Bkb3duLWl0ZW0uY29tcG9uZW50Lmh0bWxcIixcbiAgc3R5bGVVcmw6IFwiLi9kcm9wZG93bi1pdGVtLmNvbXBvbmVudC5zY3NzXCIsXG59KVxuZXhwb3J0IGNsYXNzIERyb3Bkb3duSXRlbUNvbXBvbmVudCB7XG4gIHJlYWRvbmx5IGl0ZW0gPSBpbnB1dDxEcm9wZG93bkl0ZW1Db25maWc+KCk7XG4gIHJlYWRvbmx5IG1lbnVJZCA9IGlucHV0PHN0cmluZz4oKTtcbiAgcmVhZG9ubHkgaXRlbUV2ZW50ID0gb3V0cHV0PHsgZXZlbnQ6IEV2ZW50OyBpZDogc3RyaW5nIH0+KCk7XG5cbiAgcmVhZG9ubHkgc2hvdWxkRGlzcGxheUJhZGdlID0gY29tcHV0ZWQoKCkgPT4ge1xuICAgIGNvbnN0IGl0ZW0gPSB0aGlzLml0ZW0oKTtcbiAgICByZXR1cm4gc2hvdWxkRGlzcGxheUJhZGdlKHtcbiAgICAgIHNob3dCYWRnZTogISFpdGVtPy5zaG93QmFkZ2UsXG4gICAgICBiYWRnZUNvbnRlbnQ6IGl0ZW0/LmJhZGdlQ29udGVudCxcbiAgICAgIGJhZGdlQ291bnQ6IGl0ZW0/LmJhZGdlQ291bnQsXG4gICAgICBiYWRnZUljb246IGl0ZW0/LmJhZGdlSWNvbixcbiAgICB9KTtcbiAgfSk7XG5cbiAgaGFuZGxlQ2xpY2soZXZlbnQ6IEV2ZW50KTogdm9pZCB7XG4gICAgaWYgKHRoaXMuaXRlbSgpPy5kaXNhYmxlZCkge1xuICAgICAgZXZlbnQucHJldmVudERlZmF1bHQoKTtcbiAgICAgIGV2ZW50LnN0b3BQcm9wYWdhdGlvbigpO1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICB0aGlzLml0ZW1FdmVudC5lbWl0KHtcbiAgICAgIGV2ZW50LFxuICAgICAgaWQ6IHRoaXMuaXRlbSgpPy5pZCB8fCB0aGlzLml0ZW0oKT8ubGFiZWwgfHwgXCJcIixcbiAgICB9KTtcbiAgfVxuXG4gIGhhbmRsZUtleURvd24oZXZlbnQ6IEtleWJvYXJkRXZlbnQpOiB2b2lkIHtcbiAgICBldmVudC5wcmV2ZW50RGVmYXVsdCgpO1xuXG4gICAgaWYgKFtTUEFDRV9LRVksIEVOVEVSX0tFWV0uaW5jbHVkZXMoZXZlbnQua2V5KSkge1xuICAgICAgaWYgKHRoaXMuaXRlbSgpPy5saW5rKSB7XG4gICAgICAgIGNvbnN0IGxpbmsgPSAoZXZlbnQudGFyZ2V0IGFzIEhUTUxFbGVtZW50KS5jbG9zZXN0KFwibGlcIik/LnF1ZXJ5U2VsZWN0b3IoXCJhXCIpO1xuICAgICAgICBsaW5rPy5jbGljaygpO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgdGhpcy5pdGVtRXZlbnQuZW1pdCh7XG4gICAgICAgICAgZXZlbnQsXG4gICAgICAgICAgaWQ6IHRoaXMuaXRlbSgpPy5pZCB8fCB0aGlzLml0ZW0oKT8ubGFiZWwgfHwgXCJcIixcbiAgICAgICAgfSk7XG4gICAgICB9XG4gICAgfVxuICB9XG59XG4iLCI8bmctY29udGFpbmVyPlxuICA8bGlcbiAgICAjaXRlbVJlZlxuICAgIGNsYXNzPVwicnRlLWRyb3Bkb3duLWl0ZW1cIlxuICAgIHJvbGU9XCJtZW51aXRlbVwiXG4gICAgdGFiaW5kZXg9XCIwXCJcbiAgICBbYXR0ci5kYXRhLWRpc2FibGVkXT1cIml0ZW0oKT8uZGlzYWJsZWRcIlxuICAgIFthdHRyLmRhdGEtYWN0aXZlXT1cIml0ZW0oKT8uc2VsZWN0ZWRcIlxuICAgIChjbGljayk9XCJoYW5kbGVDbGljaygkZXZlbnQpXCJcbiAgICAoa2V5ZG93bik9XCJoYW5kbGVLZXlEb3duKCRldmVudClcIlxuICA+XG4gICAgPHNwYW4gKm5nSWY9XCJpdGVtKCk/Lmhhc0luZGVudCAmJiAhaXRlbSgpPy5sZWZ0SWNvblwiIHN0eWxlPVwid2lkdGg6IDIwcHhcIj48L3NwYW4+XG4gICAgPHJ0ZS1pY29uICpuZ0lmPVwiaXRlbSgpPy5sZWZ0SWNvblwiIFtuYW1lXT1cIml0ZW0oKT8ubGVmdEljb24gPz8gJydcIiAvPlxuICAgIDxuZy1jb250YWluZXIgW25nU3dpdGNoXT1cIiEhaXRlbSgpPy5saW5rXCI+XG4gICAgICA8YSAqbmdTd2l0Y2hDYXNlPVwidHJ1ZVwiIHN0eWxlPVwiZmxleDogMjsgdGV4dC1kZWNvcmF0aW9uOiBub25lOyBjb2xvcjogaW5oZXJpdFwiIFtocmVmXT1cIml0ZW0oKT8ubGlua1wiXG4gICAgICAgID57eyBpdGVtKCk/LmxhYmVsIH19XG4gICAgICA8L2E+XG4gICAgICA8c3BhbiAqbmdTd2l0Y2hEZWZhdWx0IHN0eWxlPVwiZmxleDogMlwiPnt7IGl0ZW0oKT8ubGFiZWwgfX08L3NwYW4+XG4gICAgPC9uZy1jb250YWluZXI+XG4gICAgPGRpdiAqbmdJZj1cIml0ZW0oKT8udHJhaWxpbmdUZXh0XCI+e3sgaXRlbSgpPy50cmFpbGluZ1RleHQgfX08L2Rpdj5cbiAgICA8cnRlLWJhZGdlXG4gICAgICAqbmdJZj1cInNob3VsZERpc3BsYXlCYWRnZSgpXCJcbiAgICAgIFtiYWRnZVR5cGVdPVwiaXRlbSgpPy5iYWRnZVR5cGUhXCJcbiAgICAgIFtiYWRnZUNvbnRlbnRdPVwiaXRlbSgpPy5iYWRnZUNvbnRlbnQhXCJcbiAgICAgIFtjb3VudF09XCJpdGVtKCk/LmJhZGdlQ291bnRcIlxuICAgICAgW3NpbXBsZUJhZGdlXT1cInRydWVcIlxuICAgICAgW2ljb25dPVwiaXRlbSgpPy5iYWRnZUljb24gPz8gJydcIlxuICAgICAgW2JhZGdlU2l6ZV09XCJpdGVtKCk/LmJhZGdlU2l6ZSFcIlxuICAgID48L3J0ZS1iYWRnZT5cbiAgPC9saT5cbiAgPGRpdiAqbmdJZj1cIml0ZW0oKT8uaGFzU2VwYXJhdG9yXCIgY2xhc3M9XCJkcm9wZG93bi1kaXZpZGVyXCI+XG4gICAgPHJ0ZS1kaXZpZGVyIC8+XG4gIDwvZGl2PlxuPC9uZy1jb250YWluZXI+XG4iXX0=
|
|
201
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dropdown-item.component.js","sourceRoot":"","sources":["../../../../../../../projects/ds-rte-lib/src/lib/components/dropdown/dropdown-item/dropdown-item.component.ts","../../../../../../../projects/ds-rte-lib/src/lib/components/dropdown/dropdown-item/dropdown-item.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from \"@angular/common\";\nimport {\n  ChangeDetectorRef,\n  Component,\n  computed,\n  ElementRef,\n  inject,\n  input,\n  OnDestroy,\n  output,\n  signal,\n} from \"@angular/core\";\nimport { shouldDisplayBadge } from \"@design-system-rte/core/components/badge/badge.utils\";\nimport { DropdownManager } from \"@design-system-rte/core/components/dropdown/DropdownManager\";\nimport { ENTER_KEY, SPACE_KEY } from \"@design-system-rte/core/constants/keyboard/keyboard.constants\";\n\nimport { BadgeComponent } from \"../../badge/badge.component\";\nimport { DividerComponent } from \"../../divider/divider.component\";\nimport { IconComponent } from \"../../icon/icon.component\";\nimport { DropdownItemConfig, SubmenuCreatedResult, SubmenuRequestEvent } from \"../dropdown.types\";\nimport { focusDropdownFirstElement } from \"../dropdown.utils\";\n\nexport type { DropdownItemConfig };\n\nconst SUB_MENU_CLOSE_DELAY_MS = 300;\n\n@Component({\n  selector: \"rte-dropdown-item\",\n  imports: [CommonModule, IconComponent, DividerComponent, BadgeComponent],\n  standalone: true,\n  templateUrl: \"./dropdown-item.component.html\",\n  styleUrl: \"./dropdown-item.component.scss\",\n})\nexport class DropdownItemComponent implements OnDestroy {\n  readonly elementRef = inject(ElementRef);\n  private readonly cdr = inject(ChangeDetectorRef);\n\n  readonly item = input<DropdownItemConfig>();\n  readonly menuId = input<string>();\n  readonly itemEvent = output<{ event: Event; id: string; item?: DropdownItemConfig }>();\n  readonly submenuRequest = output<SubmenuRequestEvent>();\n\n  readonly subMenuOpen = signal<boolean>(false);\n\n  private subMenuRef: import(\"@angular/core\").ComponentRef<unknown> | null = null;\n  private closeSubMenuTimeout: ReturnType<typeof setTimeout> | null = null;\n  private subMenuSubscriptions: (() => void)[] = [];\n  private dropdownManagerUnsubscribe: (() => void) | null = null;\n\n  readonly hasChildren = computed(() => (this.item()?.children?.length ?? 0) > 0);\n  readonly childDropdownId = computed(() => {\n    const menuId = this.menuId();\n    const label = this.item()?.label ?? \"\";\n    return menuId && label ? `${menuId}-${label.replace(/\\s+/g, \"\")}` : \"\";\n  });\n\n  readonly shouldDisplayBadge = computed(() => {\n    const item = this.item();\n    return shouldDisplayBadge({\n      showBadge: !!item?.showBadge,\n      badgeContent: item?.badgeContent,\n      badgeCount: item?.badgeCount,\n      badgeIcon: item?.badgeIcon,\n    });\n  });\n\n  handleClick(event: Event): void {\n    if (this.item()?.disabled) {\n      event.preventDefault();\n      event.stopPropagation();\n      return;\n    }\n    if (this.hasChildren()) {\n      event.preventDefault();\n      event.stopPropagation();\n      this.openSubMenu();\n      return;\n    }\n    this.itemEvent.emit({\n      event,\n      id: this.item()?.id || this.item()?.label || \"\",\n      item: this.item(),\n    });\n  }\n\n  handleKeyDown(event: KeyboardEvent): void {\n    event.preventDefault();\n\n    if (this.item()?.link && [SPACE_KEY, ENTER_KEY].includes(event.key)) {\n      const link = (event.target as HTMLElement).closest(\"li\")?.querySelector(\"a\");\n      link?.click();\n      return;\n    }\n\n    if (this.hasChildren()) {\n      if (event.key === SPACE_KEY) {\n        this.openSubMenu();\n        const childId = this.childDropdownId();\n        if (childId) {\n          setTimeout(() => focusDropdownFirstElement(childId), 0);\n        }\n      }\n      return;\n    }\n\n    if ([SPACE_KEY, ENTER_KEY].includes(event.key)) {\n      this.itemEvent.emit({\n        event,\n        id: this.item()?.id || this.item()?.label || \"\",\n        item: this.item(),\n      });\n    }\n  }\n\n  handleMouseEnter(): void {\n    if (this.item()?.disabled || !this.hasChildren()) return;\n    const menuId = this.menuId();\n    const childId = this.childDropdownId();\n    if (menuId) {\n      DropdownManager.closeSubMenus(menuId, childId);\n      this.openSubMenu();\n    }\n  }\n\n  handleFocus(): void {\n    if (this.item()?.disabled || !this.hasChildren()) return;\n    const menuId = this.menuId();\n    const childId = this.childDropdownId();\n    if (menuId) {\n      DropdownManager.closeSubMenus(menuId, childId);\n    }\n  }\n\n  handleMouseLeave(): void {\n    if (!this.hasChildren()) return;\n    this.scheduleCloseSubMenu();\n  }\n\n  handleSubMenuMouseEnter(): void {\n    this.cancelCloseSubMenu();\n  }\n\n  openSubMenuForKeyboard(): void {\n    this.openSubMenu();\n    const childId = this.childDropdownId();\n    if (childId) {\n      setTimeout(() => focusDropdownFirstElement(childId), 0);\n    }\n  }\n\n  private openSubMenu(): void {\n    if (this.subMenuRef) return;\n    const children = this.item()?.children;\n    const childId = this.childDropdownId();\n    if (!children?.length || !childId) return;\n\n    const triggerElement = this.elementRef.nativeElement.querySelector(\"li.rte-dropdown-item\") as HTMLElement | null;\n    if (!triggerElement) return;\n\n    DropdownManager.open(childId);\n\n    this.submenuRequest.emit({\n      children,\n      childId,\n      triggerElement,\n      onCreated: (result) => this.wireSubmenu(result),\n    });\n  }\n\n  private wireSubmenu(result: SubmenuCreatedResult): void {\n    if (this.subMenuRef) return;\n    this.subMenuRef = result.componentRef;\n    const hostElement = result.hostElement;\n    const childId = this.childDropdownId();\n\n    this.dropdownManagerUnsubscribe = DropdownManager.subscribe(childId ?? \"\", () => this.destroySubMenu());\n\n    const menuInstance = this.subMenuRef.instance as {\n      itemEvent: {\n        subscribe: (callback: (event: { event: Event; id: string; item?: DropdownItemConfig }) => void) => {\n          unsubscribe: () => void;\n        };\n      };\n      closingMenu: { subscribe: (callback: () => void) => { unsubscribe: () => void } };\n    };\n    const itemSub = menuInstance.itemEvent.subscribe(\n      (emittedItemEvent: { event: Event; id: string; item?: DropdownItemConfig }) => {\n        this.itemEvent.emit({ ...emittedItemEvent, item: emittedItemEvent.item });\n      },\n    );\n    const closeSub = menuInstance.closingMenu.subscribe(() => {\n      this.destroySubMenu();\n    });\n    this.subMenuSubscriptions.push(() => {\n      itemSub.unsubscribe();\n      closeSub.unsubscribe();\n    });\n\n    hostElement.addEventListener(\"mouseenter\", this.boundHandleSubMenuMouseEnter);\n\n    this.subMenuOpen.set(true);\n    this.cdr.markForCheck();\n  }\n\n  private scheduleCloseSubMenu(): void {\n    this.cancelCloseSubMenu();\n    this.closeSubMenuTimeout = setTimeout(() => {\n      this.closeSubMenuTimeout = null;\n      this.destroySubMenu();\n    }, SUB_MENU_CLOSE_DELAY_MS);\n  }\n\n  private cancelCloseSubMenu(): void {\n    if (this.closeSubMenuTimeout) {\n      clearTimeout(this.closeSubMenuTimeout);\n      this.closeSubMenuTimeout = null;\n    }\n  }\n\n  private readonly boundHandleSubMenuMouseEnter = (): void => this.handleSubMenuMouseEnter();\n\n  private destroySubMenu(): void {\n    this.cancelCloseSubMenu();\n    this.subMenuSubscriptions.forEach((unsubscribe) => unsubscribe());\n    this.subMenuSubscriptions = [];\n    if (this.subMenuRef) {\n      const hostElement = this.subMenuRef.location.nativeElement as HTMLElement;\n      hostElement.removeEventListener(\"mouseenter\", this.boundHandleSubMenuMouseEnter);\n      const childId = this.childDropdownId();\n      this.dropdownManagerUnsubscribe?.();\n      this.dropdownManagerUnsubscribe = null;\n      if (childId) DropdownManager.close(childId);\n      this.subMenuRef.destroy();\n      this.subMenuRef = null;\n    }\n    this.subMenuOpen.set(false);\n    this.cdr.markForCheck();\n  }\n\n  ngOnDestroy(): void {\n    this.destroySubMenu();\n  }\n}\n","<ng-container>\n  <li\n    #itemRef\n    class=\"rte-dropdown-item\"\n    role=\"menuitem\"\n    tabindex=\"0\"\n    [attr.data-disabled]=\"item()?.disabled\"\n    [attr.data-active]=\"hasChildren() ? subMenuOpen() : item()?.selected\"\n    [attr.aria-haspopup]=\"hasChildren() ? 'menu' : null\"\n    [attr.aria-expanded]=\"hasChildren() ? subMenuOpen() : null\"\n    (click)=\"handleClick($event)\"\n    (keydown)=\"handleKeyDown($event)\"\n    (mouseenter)=\"handleMouseEnter()\"\n    (mouseleave)=\"handleMouseLeave()\"\n    (focus)=\"handleFocus()\"\n  >\n    <span *ngIf=\"item()?.hasIndent && !item()?.leftIcon\" style=\"width: 20px\"></span>\n    <rte-icon *ngIf=\"item()?.leftIcon\" [name]=\"item()?.leftIcon ?? ''\" />\n    <ng-container [ngSwitch]=\"!!item()?.link\">\n      <a *ngSwitchCase=\"true\" style=\"flex: 2; text-decoration: none; color: inherit\" [href]=\"item()?.link\"\n        >{{ item()?.label }}\n      </a>\n      <span *ngSwitchDefault style=\"flex: 2\">{{ item()?.label }}</span>\n    </ng-container>\n    <rte-icon *ngIf=\"hasChildren()\" name=\"arrow-chevron-right\" />\n    <div *ngIf=\"item()?.trailingText\">{{ item()?.trailingText }}</div>\n    <rte-badge\n      *ngIf=\"shouldDisplayBadge()\"\n      [badgeType]=\"item()?.badgeType!\"\n      [badgeContent]=\"item()?.badgeContent!\"\n      [count]=\"item()?.badgeCount\"\n      [simpleBadge]=\"true\"\n      [icon]=\"item()?.badgeIcon ?? ''\"\n      [badgeSize]=\"item()?.badgeSize!\"\n    ></rte-badge>\n  </li>\n  <div *ngIf=\"item()?.hasSeparator\" class=\"dropdown-divider\">\n    <rte-divider />\n  </div>\n</ng-container>\n"]}
|
|
@@ -1,9 +1,14 @@
|
|
|
1
1
|
import { CommonModule } from "@angular/common";
|
|
2
|
-
import { Component, computed, contentChild, ElementRef, HostListener, inject, input, output, viewChild, } from "@angular/core";
|
|
3
|
-
import {
|
|
2
|
+
import { Component, computed, contentChild, ElementRef, HostListener, inject, input, output, ViewContainerRef, viewChild, viewChildren, } from "@angular/core";
|
|
3
|
+
import { DropdownManager } from "@design-system-rte/core/components/dropdown/DropdownManager";
|
|
4
|
+
import { getAutoAlignment, getAutoPlacementDropdown, getCoordinates, } from "@design-system-rte/core/components/utils/auto-placement";
|
|
5
|
+
import { ARROW_DOWN_KEY, ARROW_UP_KEY, ESCAPE_KEY, TAB_KEY, } from "@design-system-rte/core/constants/keyboard/keyboard.constants";
|
|
6
|
+
import { logError, logWarn } from "@design-system-rte/core/utils/log-handlers";
|
|
4
7
|
import { DropdownService } from "../../../services/dropdown.service";
|
|
8
|
+
import { OverlayService } from "../../../services/overlay.service";
|
|
5
9
|
import { DividerComponent } from "../../divider/divider.component";
|
|
6
10
|
import { DropdownItemComponent } from "../dropdown-item/dropdown-item.component";
|
|
11
|
+
import { focusParentDropdownFirstElement } from "../dropdown.utils";
|
|
7
12
|
import { DropdownMenuFooterDirective } from "./dropdown-menu-footer.directive";
|
|
8
13
|
import { DropdownMenuHeaderDirective } from "./dropdown-menu-header.directive";
|
|
9
14
|
import * as i0 from "@angular/core";
|
|
@@ -11,7 +16,10 @@ import * as i1 from "@angular/common";
|
|
|
11
16
|
export class DropdownMenuComponent {
|
|
12
17
|
constructor() {
|
|
13
18
|
this.elementRef = inject(ElementRef);
|
|
19
|
+
this.pendingPositioningFrameId = null;
|
|
14
20
|
this.dropdownService = inject(DropdownService);
|
|
21
|
+
this.overlayService = inject(OverlayService);
|
|
22
|
+
this.viewContainerRef = inject(ViewContainerRef);
|
|
15
23
|
this.items = input([]);
|
|
16
24
|
this.menuId = input();
|
|
17
25
|
this.itemEvent = output();
|
|
@@ -28,6 +36,7 @@ export class DropdownMenuComponent {
|
|
|
28
36
|
this.footerTemplate = input(undefined);
|
|
29
37
|
this.headerContentRef = viewChild("headerContent");
|
|
30
38
|
this.footerContentRef = viewChild("footerContent");
|
|
39
|
+
this.itemComponents = viewChildren(DropdownItemComponent);
|
|
31
40
|
this.hasHeaderContent = computed(() => {
|
|
32
41
|
const hasTemplate = !!this.headerTemplate();
|
|
33
42
|
const hasProjectedContent = !!this.headerContentRef()?.nativeElement?.children.length;
|
|
@@ -39,36 +48,92 @@ export class DropdownMenuComponent {
|
|
|
39
48
|
return hasTemplate || hasProjectedContent;
|
|
40
49
|
});
|
|
41
50
|
}
|
|
51
|
+
static { this.SUB_MENU_OFFSET = 4; }
|
|
42
52
|
getChildMenuId(itemIndex) {
|
|
43
53
|
return `${this.menuId()}:${itemIndex + 1}`;
|
|
44
54
|
}
|
|
45
55
|
handleItemEvent(itemEvent) {
|
|
46
56
|
this.itemEvent.emit(itemEvent);
|
|
47
57
|
}
|
|
58
|
+
handleSubmenuRequest(event) {
|
|
59
|
+
const { children, childId, triggerElement, onCreated } = event;
|
|
60
|
+
const componentRef = this.overlayService.create(DropdownMenuComponent, this.viewContainerRef);
|
|
61
|
+
componentRef.setInput("items", children);
|
|
62
|
+
componentRef.setInput("menuId", childId);
|
|
63
|
+
componentRef.setInput("isOpen", true);
|
|
64
|
+
const hostElement = componentRef.location.nativeElement;
|
|
65
|
+
this.cancelPendingPositioningFrame();
|
|
66
|
+
this.pendingPositioningFrameId = requestAnimationFrame(() => {
|
|
67
|
+
this.pendingPositioningFrameId = null;
|
|
68
|
+
try {
|
|
69
|
+
const subMenuHost = componentRef.location?.nativeElement;
|
|
70
|
+
const menuElement = subMenuHost?.querySelector(".rte-dropdown-menu");
|
|
71
|
+
if (!triggerElement || !menuElement) {
|
|
72
|
+
logWarn("DropdownMenuComponent", "Unable to position submenu: required DOM elements not found.");
|
|
73
|
+
return;
|
|
74
|
+
}
|
|
75
|
+
const position = getAutoPlacementDropdown({
|
|
76
|
+
hostElement: triggerElement,
|
|
77
|
+
castedElement: menuElement,
|
|
78
|
+
defaultPosition: "right",
|
|
79
|
+
offset: DropdownMenuComponent.SUB_MENU_OFFSET,
|
|
80
|
+
hasDropdownParent: true,
|
|
81
|
+
});
|
|
82
|
+
const alignment = getAutoAlignment(triggerElement, menuElement, position);
|
|
83
|
+
const coords = getCoordinates(position, triggerElement, menuElement, DropdownMenuComponent.SUB_MENU_OFFSET, alignment);
|
|
84
|
+
hostElement.style.display = "block";
|
|
85
|
+
hostElement.style.position = "absolute";
|
|
86
|
+
hostElement.style.top = `${coords.top}px`;
|
|
87
|
+
hostElement.style.left = `${coords.left}px`;
|
|
88
|
+
hostElement.style.opacity = "1";
|
|
89
|
+
}
|
|
90
|
+
catch (error) {
|
|
91
|
+
logError("DropdownMenuComponent", "Error while positioning submenu:", error);
|
|
92
|
+
}
|
|
93
|
+
});
|
|
94
|
+
onCreated({ componentRef, hostElement });
|
|
95
|
+
}
|
|
96
|
+
ngOnDestroy() {
|
|
97
|
+
this.cancelPendingPositioningFrame();
|
|
98
|
+
}
|
|
99
|
+
cancelPendingPositioningFrame() {
|
|
100
|
+
if (this.pendingPositioningFrameId !== null) {
|
|
101
|
+
cancelAnimationFrame(this.pendingPositioningFrameId);
|
|
102
|
+
this.pendingPositioningFrameId = null;
|
|
103
|
+
}
|
|
104
|
+
}
|
|
48
105
|
onKeyDown(event) {
|
|
49
106
|
if (!event.target || !this.menuId()) {
|
|
50
107
|
return;
|
|
51
108
|
}
|
|
52
|
-
if ([ARROW_UP_KEY, ARROW_DOWN_KEY,
|
|
109
|
+
if ([ARROW_UP_KEY, ARROW_DOWN_KEY, TAB_KEY].includes(event.key)) {
|
|
53
110
|
event.preventDefault();
|
|
54
111
|
}
|
|
55
112
|
if (event.key === ESCAPE_KEY) {
|
|
56
113
|
this.closingMenu.emit();
|
|
57
114
|
}
|
|
58
115
|
const menuId = this.menuId();
|
|
116
|
+
if (event.key === TAB_KEY && event.shiftKey) {
|
|
117
|
+
const parentMenuId = DropdownManager.getParentDropdownId(menuId);
|
|
118
|
+
if (parentMenuId) {
|
|
119
|
+
focusParentDropdownFirstElement(menuId);
|
|
120
|
+
this.closingMenu.emit();
|
|
121
|
+
return;
|
|
122
|
+
}
|
|
123
|
+
}
|
|
59
124
|
this.dropdownService.handleKeyboardInput(event.key, {
|
|
60
125
|
menuElement: this.elementRef,
|
|
61
126
|
menuId,
|
|
62
127
|
});
|
|
63
128
|
}
|
|
64
129
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DropdownMenuComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
65
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: DropdownMenuComponent, isStandalone: true, selector: "rte-dropdown-menu", inputs: { items: { classPropertyName: "items", publicName: "items", isSignal: true, isRequired: false, transformFunction: null }, menuId: { classPropertyName: "menuId", publicName: "menuId", isSignal: true, isRequired: false, transformFunction: null }, isOpen: { classPropertyName: "isOpen", publicName: "isOpen", isSignal: true, isRequired: false, transformFunction: null }, width: { classPropertyName: "width", publicName: "width", isSignal: true, isRequired: false, transformFunction: null }, headerTemplate: { classPropertyName: "headerTemplate", publicName: "headerTemplate", isSignal: true, isRequired: false, transformFunction: null }, footerTemplate: { classPropertyName: "footerTemplate", publicName: "footerTemplate", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { itemEvent: "itemEvent", closingMenu: "closingMenu" }, host: { listeners: { "keydown": "onKeyDown($event)" }, properties: { "attr.data-menu-id": "menuId()" } }, queries: [{ propertyName: "headerDirective", first: true, predicate: DropdownMenuHeaderDirective, descendants: true, isSignal: true }, { propertyName: "footerDirective", first: true, predicate: DropdownMenuFooterDirective, descendants: true, isSignal: true }], viewQueries: [{ propertyName: "headerContentRef", first: true, predicate: ["headerContent"], descendants: true, isSignal: true }, { propertyName: "footerContentRef", first: true, predicate: ["footerContent"], descendants: true, isSignal: true }], ngImport: i0, template: "<div class=\"rte-dropdown-menu {{ isOpen() ? 'open' : 'closed' }}\" [ngStyle]=\"menuStyle()\">\n @if (hasHeaderContent()) {\n <div class=\"rte-dropdown-menu-header\">\n @if (headerTemplate() || headerDirective()?.templateRef) {\n <ng-container [ngTemplateOutlet]=\"(headerTemplate() || headerDirective()?.templateRef) ?? null\"></ng-container>\n } @else {\n <div #headerContent>\n <ng-content select=\"[dropdown-menu-header]\"></ng-content>\n </div>\n }\n <rte-divider />\n </div>\n }\n <div class=\"rte-dropdown-menu-content\">\n <ul class=\"rte-dropdown-items\" role=\"menu\" [attr.aria-activedescendant]=\"menuId()\">\n @for (item of items(); track item.label; let i = $index) {\n <rte-dropdown-item
|
|
130
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: DropdownMenuComponent, isStandalone: true, selector: "rte-dropdown-menu", inputs: { items: { classPropertyName: "items", publicName: "items", isSignal: true, isRequired: false, transformFunction: null }, menuId: { classPropertyName: "menuId", publicName: "menuId", isSignal: true, isRequired: false, transformFunction: null }, isOpen: { classPropertyName: "isOpen", publicName: "isOpen", isSignal: true, isRequired: false, transformFunction: null }, width: { classPropertyName: "width", publicName: "width", isSignal: true, isRequired: false, transformFunction: null }, headerTemplate: { classPropertyName: "headerTemplate", publicName: "headerTemplate", isSignal: true, isRequired: false, transformFunction: null }, footerTemplate: { classPropertyName: "footerTemplate", publicName: "footerTemplate", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { itemEvent: "itemEvent", closingMenu: "closingMenu" }, host: { listeners: { "keydown": "onKeyDown($event)" }, properties: { "attr.data-menu-id": "menuId()" } }, queries: [{ propertyName: "headerDirective", first: true, predicate: DropdownMenuHeaderDirective, descendants: true, isSignal: true }, { propertyName: "footerDirective", first: true, predicate: DropdownMenuFooterDirective, descendants: true, isSignal: true }], viewQueries: [{ propertyName: "headerContentRef", first: true, predicate: ["headerContent"], descendants: true, isSignal: true }, { propertyName: "footerContentRef", first: true, predicate: ["footerContent"], descendants: true, isSignal: true }, { propertyName: "itemComponents", predicate: DropdownItemComponent, descendants: true, isSignal: true }], ngImport: i0, template: "<div class=\"rte-dropdown-menu {{ isOpen() ? 'open' : 'closed' }}\" [ngStyle]=\"menuStyle()\">\n @if (hasHeaderContent()) {\n <div class=\"rte-dropdown-menu-header\">\n @if (headerTemplate() || headerDirective()?.templateRef) {\n <ng-container [ngTemplateOutlet]=\"(headerTemplate() || headerDirective()?.templateRef) ?? null\"></ng-container>\n } @else {\n <div #headerContent>\n <ng-content select=\"[dropdown-menu-header]\"></ng-content>\n </div>\n }\n <rte-divider />\n </div>\n }\n <div class=\"rte-dropdown-menu-content\">\n <ul class=\"rte-dropdown-items\" role=\"menu\" [attr.aria-activedescendant]=\"menuId()\">\n @for (item of items(); track item.label; let i = $index) {\n <rte-dropdown-item\n [item]=\"item\"\n [menuId]=\"menuId()\"\n (itemEvent)=\"handleItemEvent($event)\"\n (submenuRequest)=\"handleSubmenuRequest($event)\"\n />\n }\n </ul>\n </div>\n @if (hasFooterContent()) {\n <div class=\"rte-dropdown-menu-footer\">\n <rte-divider />\n @if (footerTemplate() || footerDirective()?.templateRef) {\n <ng-container [ngTemplateOutlet]=\"(footerTemplate() || footerDirective()?.templateRef) ?? null\"></ng-container>\n } @else {\n <div #footerContent>\n <ng-content select=\"[dropdown-menu-footer]\"></ng-content>\n </div>\n }\n </div>\n }\n</div>\n", styles: [":host{display:none;position:absolute}.rte-dropdown-menu{opacity:0;box-shadow:0 8px 16px 0 var(--elevation-shadow-key),0 0 2px 0 var(--elevation-shadow-ambient);background:linear-gradient(0deg,var(--elevation-surface-shadow-4) 0%,var(--elevation-surface-shadow-4) 100%);background-color:var(--background-default);position:absolute;top:0;left:0;display:flex;min-width:112px;max-width:280px;padding:8px 0;flex-direction:column;align-items:flex-start;gap:0px;border-radius:4px;transition:opacity .1s ease-in-out;pointer-events:auto;overflow-x:hidden;overflow-y:auto}.rte-dropdown-menu-header,.rte-dropdown-menu-content,.rte-dropdown-menu-footer{width:100%}.rte-dropdown-menu.open{opacity:1}.rte-dropdown-menu .rte-dropdown-items{width:100%;padding:0;margin:0}.rte-dropdown-menu .rte-dropdown-items li{list-style:none}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: DropdownItemComponent, selector: "rte-dropdown-item", inputs: ["item", "menuId"], outputs: ["itemEvent", "submenuRequest"] }, { kind: "component", type: DividerComponent, selector: "rte-divider", inputs: ["orientation", "thickness", "appearance", "endPoint"] }] }); }
|
|
66
131
|
}
|
|
67
132
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DropdownMenuComponent, decorators: [{
|
|
68
133
|
type: Component,
|
|
69
|
-
args: [{ selector: "rte-dropdown-menu", imports: [CommonModule, DropdownItemComponent, DividerComponent], standalone: true, host: { "[attr.data-menu-id]": "menuId()" }, template: "<div class=\"rte-dropdown-menu {{ isOpen() ? 'open' : 'closed' }}\" [ngStyle]=\"menuStyle()\">\n @if (hasHeaderContent()) {\n <div class=\"rte-dropdown-menu-header\">\n @if (headerTemplate() || headerDirective()?.templateRef) {\n <ng-container [ngTemplateOutlet]=\"(headerTemplate() || headerDirective()?.templateRef) ?? null\"></ng-container>\n } @else {\n <div #headerContent>\n <ng-content select=\"[dropdown-menu-header]\"></ng-content>\n </div>\n }\n <rte-divider />\n </div>\n }\n <div class=\"rte-dropdown-menu-content\">\n <ul class=\"rte-dropdown-items\" role=\"menu\" [attr.aria-activedescendant]=\"menuId()\">\n @for (item of items(); track item.label; let i = $index) {\n <rte-dropdown-item
|
|
134
|
+
args: [{ selector: "rte-dropdown-menu", imports: [CommonModule, DropdownItemComponent, DividerComponent], standalone: true, host: { "[attr.data-menu-id]": "menuId()" }, template: "<div class=\"rte-dropdown-menu {{ isOpen() ? 'open' : 'closed' }}\" [ngStyle]=\"menuStyle()\">\n @if (hasHeaderContent()) {\n <div class=\"rte-dropdown-menu-header\">\n @if (headerTemplate() || headerDirective()?.templateRef) {\n <ng-container [ngTemplateOutlet]=\"(headerTemplate() || headerDirective()?.templateRef) ?? null\"></ng-container>\n } @else {\n <div #headerContent>\n <ng-content select=\"[dropdown-menu-header]\"></ng-content>\n </div>\n }\n <rte-divider />\n </div>\n }\n <div class=\"rte-dropdown-menu-content\">\n <ul class=\"rte-dropdown-items\" role=\"menu\" [attr.aria-activedescendant]=\"menuId()\">\n @for (item of items(); track item.label; let i = $index) {\n <rte-dropdown-item\n [item]=\"item\"\n [menuId]=\"menuId()\"\n (itemEvent)=\"handleItemEvent($event)\"\n (submenuRequest)=\"handleSubmenuRequest($event)\"\n />\n }\n </ul>\n </div>\n @if (hasFooterContent()) {\n <div class=\"rte-dropdown-menu-footer\">\n <rte-divider />\n @if (footerTemplate() || footerDirective()?.templateRef) {\n <ng-container [ngTemplateOutlet]=\"(footerTemplate() || footerDirective()?.templateRef) ?? null\"></ng-container>\n } @else {\n <div #footerContent>\n <ng-content select=\"[dropdown-menu-footer]\"></ng-content>\n </div>\n }\n </div>\n }\n</div>\n", styles: [":host{display:none;position:absolute}.rte-dropdown-menu{opacity:0;box-shadow:0 8px 16px 0 var(--elevation-shadow-key),0 0 2px 0 var(--elevation-shadow-ambient);background:linear-gradient(0deg,var(--elevation-surface-shadow-4) 0%,var(--elevation-surface-shadow-4) 100%);background-color:var(--background-default);position:absolute;top:0;left:0;display:flex;min-width:112px;max-width:280px;padding:8px 0;flex-direction:column;align-items:flex-start;gap:0px;border-radius:4px;transition:opacity .1s ease-in-out;pointer-events:auto;overflow-x:hidden;overflow-y:auto}.rte-dropdown-menu-header,.rte-dropdown-menu-content,.rte-dropdown-menu-footer{width:100%}.rte-dropdown-menu.open{opacity:1}.rte-dropdown-menu .rte-dropdown-items{width:100%;padding:0;margin:0}.rte-dropdown-menu .rte-dropdown-items li{list-style:none}\n"] }]
|
|
70
135
|
}], propDecorators: { onKeyDown: [{
|
|
71
136
|
type: HostListener,
|
|
72
137
|
args: ["keydown", ["$event"]]
|
|
73
138
|
}] } });
|
|
74
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dropdown-menu.component.js","sourceRoot":"","sources":["../../../../../../../projects/ds-rte-lib/src/lib/components/dropdown/dropdown-menu/dropdown-menu.component.ts","../../../../../../../projects/ds-rte-lib/src/lib/components/dropdown/dropdown-menu/dropdown-menu.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from \"@angular/common\";\nimport {\n  Component,\n  computed,\n  contentChild,\n  ElementRef,\n  HostListener,\n  inject,\n  input,\n  output,\n  TemplateRef,\n  viewChild,\n} from \"@angular/core\";\nimport {\n  ARROW_DOWN_KEY,\n  ARROW_LEFT_KEY,\n  ARROW_RIGHT_KEY,\n  ARROW_UP_KEY,\n  ESCAPE_KEY,\n  TAB_KEY,\n} from \"@design-system-rte/core/constants/keyboard/keyboard.constants\";\n\nimport { DropdownService } from \"../../../services/dropdown.service\";\nimport { DividerComponent } from \"../../divider/divider.component\";\nimport { DropdownItemComponent, DropdownItemConfig } from \"../dropdown-item/dropdown-item.component\";\n\nimport { DropdownMenuFooterDirective } from \"./dropdown-menu-footer.directive\";\nimport { DropdownMenuHeaderDirective } from \"./dropdown-menu-header.directive\";\n\n@Component({\n  selector: \"rte-dropdown-menu\",\n  imports: [CommonModule, DropdownItemComponent, DividerComponent],\n  standalone: true,\n  templateUrl: \"./dropdown-menu.component.html\",\n  styleUrl: \"./dropdown-menu.component.scss\",\n  host: { \"[attr.data-menu-id]\": \"menuId()\" },\n})\nexport class DropdownMenuComponent {\n  private readonly elementRef = inject(ElementRef);\n  private readonly dropdownService = inject(DropdownService);\n\n  readonly items = input<DropdownItemConfig[]>([]);\n  readonly menuId = input<string>();\n\n  readonly itemEvent = output<{ event: Event; id: string }>();\n\n  readonly widthStyle = computed(() => (this.width() !== undefined ? `${this.width()}px` : undefined));\n  readonly isOpen = input<boolean>(false);\n  readonly width = input<string | null>(null);\n  readonly closingMenu = output<void>();\n\n  readonly menuStyle = computed(() => {\n    return this.width() ? { width: this.width() + \"px\", \"max-width\": this.width() + \"px\" } : {};\n  });\n\n  readonly headerDirective = contentChild(DropdownMenuHeaderDirective);\n  readonly footerDirective = contentChild(DropdownMenuFooterDirective);\n\n  readonly headerTemplate = input<TemplateRef<HTMLElement> | undefined>(undefined);\n  readonly footerTemplate = input<TemplateRef<HTMLElement> | undefined>(undefined);\n\n  readonly headerContentRef = viewChild<ElementRef<HTMLElement>>(\"headerContent\");\n  readonly footerContentRef = viewChild<ElementRef<HTMLElement>>(\"footerContent\");\n\n  readonly hasHeaderContent = computed(() => {\n    const hasTemplate = !!this.headerTemplate();\n    const hasProjectedContent = !!this.headerContentRef()?.nativeElement?.children.length;\n    return hasTemplate || hasProjectedContent;\n  });\n\n  readonly hasFooterContent = computed(() => {\n    const hasTemplate = !!this.footerTemplate();\n    const hasProjectedContent = !!this.footerContentRef()?.nativeElement?.children.length;\n    return hasTemplate || hasProjectedContent;\n  });\n\n  getChildMenuId(itemIndex: number): string {\n    return `${this.menuId()}:${itemIndex + 1}`;\n  }\n\n  handleItemEvent(itemEvent: { event: Event; id: string }): void {\n    this.itemEvent.emit(itemEvent);\n  }\n\n  @HostListener(\"keydown\", [\"$event\"])\n  onKeyDown(event: KeyboardEvent): void {\n    if (!event.target || !this.menuId()) {\n      return;\n    }\n\n    if ([ARROW_UP_KEY, ARROW_DOWN_KEY, ARROW_LEFT_KEY, ARROW_RIGHT_KEY, TAB_KEY].includes(event.key)) {\n      event.preventDefault();\n    }\n\n    if (event.key === ESCAPE_KEY) {\n      this.closingMenu.emit();\n    }\n\n    const menuId = this.menuId() as string;\n    this.dropdownService.handleKeyboardInput(event.key, {\n      menuElement: this.elementRef,\n      menuId,\n    });\n  }\n}\n","<div class=\"rte-dropdown-menu {{ isOpen() ? 'open' : 'closed' }}\" [ngStyle]=\"menuStyle()\">\n  @if (hasHeaderContent()) {\n    <div class=\"rte-dropdown-menu-header\">\n      @if (headerTemplate() || headerDirective()?.templateRef) {\n        <ng-container [ngTemplateOutlet]=\"(headerTemplate() || headerDirective()?.templateRef) ?? null\"></ng-container>\n      } @else {\n        <div #headerContent>\n          <ng-content select=\"[dropdown-menu-header]\"></ng-content>\n        </div>\n      }\n      <rte-divider />\n    </div>\n  }\n  <div class=\"rte-dropdown-menu-content\">\n    <ul class=\"rte-dropdown-items\" role=\"menu\" [attr.aria-activedescendant]=\"menuId()\">\n      @for (item of items(); track item.label; let i = $index) {\n        <rte-dropdown-item [item]=\"item\" (itemEvent)=\"handleItemEvent($event)\" />\n      }\n    </ul>\n  </div>\n  @if (hasFooterContent()) {\n    <div class=\"rte-dropdown-menu-footer\">\n      <rte-divider />\n      @if (footerTemplate() || footerDirective()?.templateRef) {\n        <ng-container [ngTemplateOutlet]=\"(footerTemplate() || footerDirective()?.templateRef) ?? null\"></ng-container>\n      } @else {\n        <div #footerContent>\n          <ng-content select=\"[dropdown-menu-footer]\"></ng-content>\n        </div>\n      }\n    </div>\n  }\n</div>\n"]}
|
|
139
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dropdown-menu.component.js","sourceRoot":"","sources":["../../../../../../../projects/ds-rte-lib/src/lib/components/dropdown/dropdown-menu/dropdown-menu.component.ts","../../../../../../../projects/ds-rte-lib/src/lib/components/dropdown/dropdown-menu/dropdown-menu.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from \"@angular/common\";\nimport {\n  Component,\n  computed,\n  contentChild,\n  ElementRef,\n  HostListener,\n  inject,\n  input,\n  OnDestroy,\n  output,\n  TemplateRef,\n  ViewContainerRef,\n  viewChild,\n  viewChildren,\n} from \"@angular/core\";\nimport { DropdownManager } from \"@design-system-rte/core/components/dropdown/DropdownManager\";\nimport {\n  getAutoAlignment,\n  getAutoPlacementDropdown,\n  getCoordinates,\n} from \"@design-system-rte/core/components/utils/auto-placement\";\nimport {\n  ARROW_DOWN_KEY,\n  ARROW_UP_KEY,\n  ESCAPE_KEY,\n  TAB_KEY,\n} from \"@design-system-rte/core/constants/keyboard/keyboard.constants\";\nimport { logError, logWarn } from \"@design-system-rte/core/utils/log-handlers\";\n\nimport { DropdownService } from \"../../../services/dropdown.service\";\nimport { OverlayService } from \"../../../services/overlay.service\";\nimport { DividerComponent } from \"../../divider/divider.component\";\nimport { DropdownItemComponent } from \"../dropdown-item/dropdown-item.component\";\nimport { DropdownItemConfig, SubmenuRequestEvent } from \"../dropdown.types\";\nimport { focusParentDropdownFirstElement } from \"../dropdown.utils\";\n\nimport { DropdownMenuFooterDirective } from \"./dropdown-menu-footer.directive\";\nimport { DropdownMenuHeaderDirective } from \"./dropdown-menu-header.directive\";\n\n@Component({\n  selector: \"rte-dropdown-menu\",\n  imports: [CommonModule, DropdownItemComponent, DividerComponent],\n  standalone: true,\n  templateUrl: \"./dropdown-menu.component.html\",\n  styleUrl: \"./dropdown-menu.component.scss\",\n  host: { \"[attr.data-menu-id]\": \"menuId()\" },\n})\nexport class DropdownMenuComponent implements OnDestroy {\n  private static readonly SUB_MENU_OFFSET = 4;\n  private readonly elementRef = inject(ElementRef);\n  private pendingPositioningFrameId: number | null = null;\n  private readonly dropdownService = inject(DropdownService);\n  private readonly overlayService = inject(OverlayService);\n  private readonly viewContainerRef = inject(ViewContainerRef);\n\n  readonly items = input<DropdownItemConfig[]>([]);\n  readonly menuId = input<string>();\n\n  readonly itemEvent = output<{ event: Event; id: string; item?: DropdownItemConfig }>();\n\n  readonly widthStyle = computed(() => (this.width() !== undefined ? `${this.width()}px` : undefined));\n  readonly isOpen = input<boolean>(false);\n  readonly width = input<string | null>(null);\n  readonly closingMenu = output<void>();\n\n  readonly menuStyle = computed(() => {\n    return this.width() ? { width: this.width() + \"px\", \"max-width\": this.width() + \"px\" } : {};\n  });\n\n  readonly headerDirective = contentChild(DropdownMenuHeaderDirective);\n  readonly footerDirective = contentChild(DropdownMenuFooterDirective);\n\n  readonly headerTemplate = input<TemplateRef<HTMLElement> | undefined>(undefined);\n  readonly footerTemplate = input<TemplateRef<HTMLElement> | undefined>(undefined);\n\n  readonly headerContentRef = viewChild<ElementRef<HTMLElement>>(\"headerContent\");\n  readonly footerContentRef = viewChild<ElementRef<HTMLElement>>(\"footerContent\");\n  readonly itemComponents = viewChildren(DropdownItemComponent);\n\n  readonly hasHeaderContent = computed(() => {\n    const hasTemplate = !!this.headerTemplate();\n    const hasProjectedContent = !!this.headerContentRef()?.nativeElement?.children.length;\n    return hasTemplate || hasProjectedContent;\n  });\n\n  readonly hasFooterContent = computed(() => {\n    const hasTemplate = !!this.footerTemplate();\n    const hasProjectedContent = !!this.footerContentRef()?.nativeElement?.children.length;\n    return hasTemplate || hasProjectedContent;\n  });\n\n  getChildMenuId(itemIndex: number): string {\n    return `${this.menuId()}:${itemIndex + 1}`;\n  }\n\n  handleItemEvent(itemEvent: { event: Event; id: string; item?: DropdownItemConfig }): void {\n    this.itemEvent.emit(itemEvent);\n  }\n\n  handleSubmenuRequest(event: SubmenuRequestEvent): void {\n    const { children, childId, triggerElement, onCreated } = event;\n\n    const componentRef = this.overlayService.create(DropdownMenuComponent, this.viewContainerRef);\n    componentRef.setInput(\"items\", children);\n    componentRef.setInput(\"menuId\", childId);\n    componentRef.setInput(\"isOpen\", true);\n\n    const hostElement = componentRef.location.nativeElement as HTMLElement;\n\n    this.cancelPendingPositioningFrame();\n    this.pendingPositioningFrameId = requestAnimationFrame(() => {\n      this.pendingPositioningFrameId = null;\n      try {\n        const subMenuHost = componentRef.location?.nativeElement as HTMLElement | null;\n        const menuElement = subMenuHost?.querySelector(\".rte-dropdown-menu\") as HTMLElement | null;\n\n        if (!triggerElement || !menuElement) {\n          logWarn(\"DropdownMenuComponent\", \"Unable to position submenu: required DOM elements not found.\");\n          return;\n        }\n\n        const position = getAutoPlacementDropdown({\n          hostElement: triggerElement,\n          castedElement: menuElement,\n          defaultPosition: \"right\",\n          offset: DropdownMenuComponent.SUB_MENU_OFFSET,\n          hasDropdownParent: true,\n        });\n        const alignment = getAutoAlignment(triggerElement, menuElement, position);\n        const coords = getCoordinates(\n          position,\n          triggerElement,\n          menuElement,\n          DropdownMenuComponent.SUB_MENU_OFFSET,\n          alignment,\n        );\n\n        hostElement.style.display = \"block\";\n        hostElement.style.position = \"absolute\";\n        hostElement.style.top = `${coords.top}px`;\n        hostElement.style.left = `${coords.left}px`;\n        hostElement.style.opacity = \"1\";\n      } catch (error) {\n        logError(\"DropdownMenuComponent\", \"Error while positioning submenu:\", error);\n      }\n    });\n\n    onCreated({ componentRef, hostElement });\n  }\n\n  ngOnDestroy(): void {\n    this.cancelPendingPositioningFrame();\n  }\n\n  private cancelPendingPositioningFrame(): void {\n    if (this.pendingPositioningFrameId !== null) {\n      cancelAnimationFrame(this.pendingPositioningFrameId);\n      this.pendingPositioningFrameId = null;\n    }\n  }\n\n  @HostListener(\"keydown\", [\"$event\"])\n  onKeyDown(event: KeyboardEvent): void {\n    if (!event.target || !this.menuId()) {\n      return;\n    }\n\n    if ([ARROW_UP_KEY, ARROW_DOWN_KEY, TAB_KEY].includes(event.key)) {\n      event.preventDefault();\n    }\n\n    if (event.key === ESCAPE_KEY) {\n      this.closingMenu.emit();\n    }\n\n    const menuId = this.menuId() as string;\n\n    if (event.key === TAB_KEY && event.shiftKey) {\n      const parentMenuId = DropdownManager.getParentDropdownId(menuId);\n      if (parentMenuId) {\n        focusParentDropdownFirstElement(menuId);\n        this.closingMenu.emit();\n        return;\n      }\n    }\n\n    this.dropdownService.handleKeyboardInput(event.key, {\n      menuElement: this.elementRef,\n      menuId,\n    });\n  }\n}\n","<div class=\"rte-dropdown-menu {{ isOpen() ? 'open' : 'closed' }}\" [ngStyle]=\"menuStyle()\">\n  @if (hasHeaderContent()) {\n    <div class=\"rte-dropdown-menu-header\">\n      @if (headerTemplate() || headerDirective()?.templateRef) {\n        <ng-container [ngTemplateOutlet]=\"(headerTemplate() || headerDirective()?.templateRef) ?? null\"></ng-container>\n      } @else {\n        <div #headerContent>\n          <ng-content select=\"[dropdown-menu-header]\"></ng-content>\n        </div>\n      }\n      <rte-divider />\n    </div>\n  }\n  <div class=\"rte-dropdown-menu-content\">\n    <ul class=\"rte-dropdown-items\" role=\"menu\" [attr.aria-activedescendant]=\"menuId()\">\n      @for (item of items(); track item.label; let i = $index) {\n        <rte-dropdown-item\n          [item]=\"item\"\n          [menuId]=\"menuId()\"\n          (itemEvent)=\"handleItemEvent($event)\"\n          (submenuRequest)=\"handleSubmenuRequest($event)\"\n        />\n      }\n    </ul>\n  </div>\n  @if (hasFooterContent()) {\n    <div class=\"rte-dropdown-menu-footer\">\n      <rte-divider />\n      @if (footerTemplate() || footerDirective()?.templateRef) {\n        <ng-container [ngTemplateOutlet]=\"(footerTemplate() || footerDirective()?.templateRef) ?? null\"></ng-container>\n      } @else {\n        <div #footerContent>\n          <ng-content select=\"[dropdown-menu-footer]\"></ng-content>\n        </div>\n      }\n    </div>\n  }\n</div>\n"]}
|