@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.
Files changed (30) hide show
  1. package/esm2022/lib/components/dropdown/dropdown-item/dropdown-item.component.mjs +157 -14
  2. package/esm2022/lib/components/dropdown/dropdown-menu/dropdown-menu.component.mjs +71 -6
  3. package/esm2022/lib/components/dropdown/dropdown.directive.mjs +48 -39
  4. package/esm2022/lib/components/dropdown/dropdown.types.mjs +2 -0
  5. package/esm2022/lib/components/dropdown/index.mjs +1 -3
  6. package/esm2022/lib/components/icon/icon-map.mjs +112 -112
  7. package/esm2022/lib/components/modal/modal.component.mjs +3 -3
  8. package/esm2022/lib/components/segmented-control/segmented-control.component.mjs +5 -3
  9. package/esm2022/lib/components/side-nav/side-nav.component.mjs +1 -1
  10. package/esm2022/lib/components/stepper/stepper.component.mjs +3 -3
  11. package/esm2022/lib/components/tab/tab-item/tab-item.component.mjs +3 -3
  12. package/esm2022/lib/components/tab/tab.component.mjs +1 -1
  13. package/esm2022/lib/components/tag/tag.component.mjs +4 -3
  14. package/esm2022/public-api.mjs +9 -1
  15. package/fesm2022/design-system-rte-angular.mjs +431 -219
  16. package/fesm2022/design-system-rte-angular.mjs.map +1 -1
  17. package/lib/components/assistive-text/assistive-text.component.d.ts +1 -1
  18. package/lib/components/chip/chip.component.d.ts +1 -1
  19. package/lib/components/dropdown/dropdown-item/dropdown-item.component.d.ts +28 -16
  20. package/lib/components/dropdown/dropdown-menu/dropdown-menu.component.d.ts +14 -3
  21. package/lib/components/dropdown/dropdown.directive.d.ts +8 -4
  22. package/lib/components/dropdown/dropdown.types.d.ts +30 -0
  23. package/lib/components/dropdown/index.d.ts +1 -0
  24. package/lib/components/segmented-control/segmented-control.component.d.ts +3 -1
  25. package/lib/components/side-nav/side-nav.component.d.ts +2 -2
  26. package/lib/components/tab/tab.component.d.ts +1 -1
  27. package/lib/components/tag/tag.component.d.ts +2 -1
  28. package/lib/components/text-input/base-text-input/base-text-input.component.d.ts +1 -1
  29. package/package.json +2 -2
  30. 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
- if (this.item()?.link) {
40
- const link = event.target.closest("li")?.querySelector("a");
41
- link?.click();
42
- }
43
- else {
44
- this.itemEvent.emit({
45
- event,
46
- id: this.item()?.id || this.item()?.label || "",
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 { ARROW_DOWN_KEY, ARROW_LEFT_KEY, ARROW_RIGHT_KEY, ARROW_UP_KEY, ESCAPE_KEY, TAB_KEY, } from "@design-system-rte/core/constants/keyboard/keyboard.constants";
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, ARROW_LEFT_KEY, ARROW_RIGHT_KEY, TAB_KEY].includes(event.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 [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", 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"] }, { kind: "component", type: DividerComponent, selector: "rte-divider", inputs: ["orientation", "thickness", "appearance", "endPoint"] }] }); }
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 [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", 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"] }]
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"]}