@agorapulse/ui-components 18.1.2 → 18.1.3
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/action-dropdown/action-dropdown-trigger.directive.d.ts +17 -0
- package/action-dropdown/action-dropdown.component.d.ts +85 -0
- package/{popover-menu → action-dropdown}/index.d.ts +1 -1
- package/action-dropdown/public_api.d.ts +2 -0
- package/agorapulse-ui-components-18.1.3.tgz +0 -0
- package/esm2022/action-dropdown/action-dropdown-trigger.directive.mjs +54 -0
- package/esm2022/action-dropdown/action-dropdown.component.mjs +270 -0
- package/esm2022/action-dropdown/agorapulse-ui-components-action-dropdown.mjs +5 -0
- package/esm2022/action-dropdown/public_api.mjs +3 -0
- package/esm2022/nav-selector/nav-selector-leaf/nav-selector-leaf.component.mjs +16 -6
- package/fesm2022/agorapulse-ui-components-action-dropdown.mjs +329 -0
- package/fesm2022/agorapulse-ui-components-action-dropdown.mjs.map +1 -0
- package/fesm2022/agorapulse-ui-components-nav-selector.mjs +16 -15
- package/fesm2022/agorapulse-ui-components-nav-selector.mjs.map +1 -1
- package/nav-selector/nav-selector-leaf/nav-selector-leaf.component.d.ts +4 -2
- package/package.json +7 -7
- package/agorapulse-ui-components-18.1.2.tgz +0 -0
- package/esm2022/nav-selector/nav-selector-leaf-action/nav-selector-leaf-action.component.mjs +0 -13
- package/esm2022/popover-menu/agorapulse-ui-components-popover-menu.mjs +0 -5
- package/esm2022/popover-menu/popover-menu-trigger.directive.mjs +0 -58
- package/esm2022/popover-menu/popover-menu.component.mjs +0 -207
- package/esm2022/popover-menu/public_api.mjs +0 -3
- package/fesm2022/agorapulse-ui-components-popover-menu.mjs +0 -270
- package/fesm2022/agorapulse-ui-components-popover-menu.mjs.map +0 -1
- package/nav-selector/nav-selector-leaf-action/nav-selector-leaf-action.component.d.ts +0 -7
- package/popover-menu/popover-menu-trigger.directive.d.ts +0 -15
- package/popover-menu/popover-menu.component.d.ts +0 -50
- package/popover-menu/public_api.d.ts +0 -2
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import { ActionDropdownComponent } from './action-dropdown.component';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
/**
|
|
4
|
+
* Directive that turns any element into a trigger for an ActionDropdown component.
|
|
5
|
+
* Handles click and keyboard interactions to open/close the dropdown.
|
|
6
|
+
*/
|
|
7
|
+
export declare class ActionDropdownTriggerDirective {
|
|
8
|
+
private readonly elementRef;
|
|
9
|
+
/** Reference to the ActionDropdown component that this trigger controls */
|
|
10
|
+
apActionDropdownTrigger: import("@angular/core").InputSignal<ActionDropdownComponent>;
|
|
11
|
+
/** Handles click events to toggle the dropdown */
|
|
12
|
+
onClick(event: MouseEvent): void;
|
|
13
|
+
/** Handles keyboard events for accessibility (Enter, Space, Arrow Down, Escape) */
|
|
14
|
+
onKeyDown(event: KeyboardEvent): void;
|
|
15
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<ActionDropdownTriggerDirective, never>;
|
|
16
|
+
static ɵdir: i0.ɵɵDirectiveDeclaration<ActionDropdownTriggerDirective, "[apActionDropdownTrigger]", never, { "apActionDropdownTrigger": { "alias": "apActionDropdownTrigger"; "required": true; "isSignal": true; }; }, {}, never, never, true, never>;
|
|
17
|
+
}
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
import { agorapulseSymbol, SymbolColor } from '@agorapulse/ui-symbol';
|
|
2
|
+
import { TemplateRef } from '@angular/core';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
export interface ActionDropdownItem {
|
|
5
|
+
/** Unique identifier for the item */
|
|
6
|
+
name?: string;
|
|
7
|
+
/** Display text for the item */
|
|
8
|
+
label?: string;
|
|
9
|
+
/** Secondary text displayed below the label */
|
|
10
|
+
description?: string;
|
|
11
|
+
/** Text displayed in a badge next to the item */
|
|
12
|
+
badgeLabel?: string;
|
|
13
|
+
/** Whether to show a feature lock icon */
|
|
14
|
+
featureLockEnabled?: boolean;
|
|
15
|
+
/** Whether to apply red styling for destructive actions */
|
|
16
|
+
redModeEnabled?: boolean;
|
|
17
|
+
/** Tooltip text displayed when hovering over the item */
|
|
18
|
+
itemTooltipText?: string;
|
|
19
|
+
/** Icon symbol identifier displayed at the start of the item */
|
|
20
|
+
startSymbolId?: agorapulseSymbol;
|
|
21
|
+
/** Color theme for the start symbol */
|
|
22
|
+
startSymbolColor?: SymbolColor;
|
|
23
|
+
/** Tooltip text for the start symbol */
|
|
24
|
+
startSymbolTooltipText?: string;
|
|
25
|
+
/** Icon symbol identifier displayed at the end of the item */
|
|
26
|
+
endSymbolId?: agorapulseSymbol;
|
|
27
|
+
/** Tooltip text for the end symbol */
|
|
28
|
+
endSymbolTooltipText?: string;
|
|
29
|
+
/** Whether the item is disabled and non-interactive */
|
|
30
|
+
disabled?: boolean;
|
|
31
|
+
/** Whether to render this item as a visual divider */
|
|
32
|
+
divider?: boolean;
|
|
33
|
+
}
|
|
34
|
+
/**
|
|
35
|
+
* A dropdown component that displays a list of actionable items with support for icons,
|
|
36
|
+
* badges, tooltips, and keyboard navigation.
|
|
37
|
+
*/
|
|
38
|
+
export declare class ActionDropdownComponent {
|
|
39
|
+
private readonly elementRef;
|
|
40
|
+
private readonly overlay;
|
|
41
|
+
private readonly positionBuilder;
|
|
42
|
+
private readonly viewContainerRef;
|
|
43
|
+
actionDropdownTemplate: import("@angular/core").Signal<TemplateRef<unknown> | undefined>;
|
|
44
|
+
trigger: import("@angular/core").Signal<TemplateRef<unknown> | undefined>;
|
|
45
|
+
/** Array of items to display in the dropdown menu */
|
|
46
|
+
items: import("@angular/core").InputSignal<ActionDropdownItem[]>;
|
|
47
|
+
/** Whether the dropdown is disabled and cannot be opened */
|
|
48
|
+
disabled: import("@angular/core").InputSignal<boolean>;
|
|
49
|
+
/** Whether to enable large mode styling for the dropdown */
|
|
50
|
+
largeModeEnabled: import("@angular/core").InputSignal<boolean>;
|
|
51
|
+
/** Whether to show a backdrop that closes the dropdown when clicked */
|
|
52
|
+
showBackdrop: import("@angular/core").InputSignal<boolean>;
|
|
53
|
+
/** Custom width for the dropdown menu in pixels */
|
|
54
|
+
customWidth: import("@angular/core").InputSignal<number | null>;
|
|
55
|
+
/** Default position for the dropdown relative to the trigger element */
|
|
56
|
+
defaultPosition: import("@angular/core").InputSignal<"left" | "right">;
|
|
57
|
+
/** Emits when a dropdown item is clicked */
|
|
58
|
+
itemClick: import("@angular/core").OutputEmitterRef<ActionDropdownItem>;
|
|
59
|
+
/** Emits when the dropdown menu is opened */
|
|
60
|
+
opened: import("@angular/core").OutputEmitterRef<void>;
|
|
61
|
+
/** Emits when the dropdown menu is closed */
|
|
62
|
+
closed: import("@angular/core").OutputEmitterRef<void>;
|
|
63
|
+
private overlayRef;
|
|
64
|
+
private portal;
|
|
65
|
+
protected readonly isOpen: import("@angular/core").WritableSignal<boolean>;
|
|
66
|
+
protected readonly focusedIndex: import("@angular/core").WritableSignal<number>;
|
|
67
|
+
protected readonly itemsWithoutDividers: import("@angular/core").Signal<ActionDropdownItem[]>;
|
|
68
|
+
constructor();
|
|
69
|
+
/** Opens the dropdown menu at the specified trigger element or component's element */
|
|
70
|
+
open(triggerElement?: HTMLElement): void;
|
|
71
|
+
/** Closes the dropdown menu and cleans up overlay resources */
|
|
72
|
+
close(): void;
|
|
73
|
+
/** Toggles the dropdown menu open or closed state */
|
|
74
|
+
toggle(triggerElement?: HTMLElement): void;
|
|
75
|
+
/** Handles item click events, emitting the selected item and closing the dropdown */
|
|
76
|
+
onItemClick(item: ActionDropdownItem): void;
|
|
77
|
+
/** Handles keyboard navigation within the dropdown menu */
|
|
78
|
+
onKeyDown(event: KeyboardEvent): void;
|
|
79
|
+
/** Creates positioning strategy for the overlay based on trigger element and default position */
|
|
80
|
+
private createPositionStrategy;
|
|
81
|
+
/** Sets up keyboard event handling for the opened dropdown */
|
|
82
|
+
private setupKeyboardNavigation;
|
|
83
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<ActionDropdownComponent, never>;
|
|
84
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<ActionDropdownComponent, "ap-action-dropdown", never, { "items": { "alias": "items"; "required": false; "isSignal": true; }; "disabled": { "alias": "disabled"; "required": false; "isSignal": true; }; "largeModeEnabled": { "alias": "largeModeEnabled"; "required": false; "isSignal": true; }; "showBackdrop": { "alias": "showBackdrop"; "required": false; "isSignal": true; }; "customWidth": { "alias": "customWidth"; "required": false; "isSignal": true; }; "defaultPosition": { "alias": "defaultPosition"; "required": false; "isSignal": true; }; }, { "itemClick": "itemClick"; "opened": "opened"; "closed": "closed"; }, ["trigger"], never, true, never>;
|
|
85
|
+
}
|
|
Binary file
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
import { Directive, ElementRef, HostListener, inject, input } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
/**
|
|
4
|
+
* Directive that turns any element into a trigger for an ActionDropdown component.
|
|
5
|
+
* Handles click and keyboard interactions to open/close the dropdown.
|
|
6
|
+
*/
|
|
7
|
+
export class ActionDropdownTriggerDirective {
|
|
8
|
+
elementRef = inject((ElementRef));
|
|
9
|
+
/** Reference to the ActionDropdown component that this trigger controls */
|
|
10
|
+
apActionDropdownTrigger = input.required();
|
|
11
|
+
/** Handles click events to toggle the dropdown */
|
|
12
|
+
onClick(event) {
|
|
13
|
+
event.preventDefault();
|
|
14
|
+
event.stopPropagation();
|
|
15
|
+
const actionDropdown = this.apActionDropdownTrigger();
|
|
16
|
+
if (actionDropdown) {
|
|
17
|
+
actionDropdown.toggle(this.elementRef.nativeElement);
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
/** Handles keyboard events for accessibility (Enter, Space, Arrow Down, Escape) */
|
|
21
|
+
onKeyDown(event) {
|
|
22
|
+
const actionDropdown = this.apActionDropdownTrigger();
|
|
23
|
+
if (actionDropdown) {
|
|
24
|
+
switch (event.key) {
|
|
25
|
+
case 'Enter':
|
|
26
|
+
case ' ':
|
|
27
|
+
case 'ArrowDown':
|
|
28
|
+
event.preventDefault();
|
|
29
|
+
actionDropdown.open(this.elementRef.nativeElement);
|
|
30
|
+
break;
|
|
31
|
+
case 'Escape':
|
|
32
|
+
event.preventDefault();
|
|
33
|
+
actionDropdown.close();
|
|
34
|
+
break;
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: ActionDropdownTriggerDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
|
|
39
|
+
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.2.9", type: ActionDropdownTriggerDirective, isStandalone: true, selector: "[apActionDropdownTrigger]", inputs: { apActionDropdownTrigger: { classPropertyName: "apActionDropdownTrigger", publicName: "apActionDropdownTrigger", isSignal: true, isRequired: true, transformFunction: null } }, host: { listeners: { "click": "onClick($event)", "keydown": "onKeyDown($event)" } }, ngImport: i0 });
|
|
40
|
+
}
|
|
41
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: ActionDropdownTriggerDirective, decorators: [{
|
|
42
|
+
type: Directive,
|
|
43
|
+
args: [{
|
|
44
|
+
selector: '[apActionDropdownTrigger]',
|
|
45
|
+
standalone: true,
|
|
46
|
+
}]
|
|
47
|
+
}], propDecorators: { onClick: [{
|
|
48
|
+
type: HostListener,
|
|
49
|
+
args: ['click', ['$event']]
|
|
50
|
+
}], onKeyDown: [{
|
|
51
|
+
type: HostListener,
|
|
52
|
+
args: ['keydown', ['$event']]
|
|
53
|
+
}] } });
|
|
54
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWN0aW9uLWRyb3Bkb3duLXRyaWdnZXIuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vbGlicy91aS1jb21wb25lbnRzL2FjdGlvbi1kcm9wZG93bi9zcmMvYWN0aW9uLWRyb3Bkb3duLXRyaWdnZXIuZGlyZWN0aXZlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsVUFBVSxFQUFFLFlBQVksRUFBRSxNQUFNLEVBQUUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDOztBQUduRjs7O0dBR0c7QUFLSCxNQUFNLE9BQU8sOEJBQThCO0lBQ3RCLFVBQVUsR0FBRyxNQUFNLENBQUMsQ0FBQSxVQUF1QixDQUFBLENBQUMsQ0FBQztJQUU5RCwyRUFBMkU7SUFDM0UsdUJBQXVCLEdBQUcsS0FBSyxDQUFDLFFBQVEsRUFBMkIsQ0FBQztJQUVwRSxrREFBa0Q7SUFFbEQsT0FBTyxDQUFDLEtBQWlCO1FBQ3JCLEtBQUssQ0FBQyxjQUFjLEVBQUUsQ0FBQztRQUN2QixLQUFLLENBQUMsZUFBZSxFQUFFLENBQUM7UUFFeEIsTUFBTSxjQUFjLEdBQUcsSUFBSSxDQUFDLHVCQUF1QixFQUFFLENBQUM7UUFDdEQsSUFBSSxjQUFjLEVBQUUsQ0FBQztZQUNqQixjQUFjLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxVQUFVLENBQUMsYUFBYSxDQUFDLENBQUM7UUFDekQsQ0FBQztJQUNMLENBQUM7SUFFRCxtRkFBbUY7SUFFbkYsU0FBUyxDQUFDLEtBQW9CO1FBQzFCLE1BQU0sY0FBYyxHQUFHLElBQUksQ0FBQyx1QkFBdUIsRUFBRSxDQUFDO1FBQ3RELElBQUksY0FBYyxFQUFFLENBQUM7WUFDakIsUUFBUSxLQUFLLENBQUMsR0FBRyxFQUFFLENBQUM7Z0JBQ2hCLEtBQUssT0FBTyxDQUFDO2dCQUNiLEtBQUssR0FBRyxDQUFDO2dCQUNULEtBQUssV0FBVztvQkFDWixLQUFLLENBQUMsY0FBYyxFQUFFLENBQUM7b0JBQ3ZCLGNBQWMsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxhQUFhLENBQUMsQ0FBQztvQkFDbkQsTUFBTTtnQkFFVixLQUFLLFFBQVE7b0JBQ1QsS0FBSyxDQUFDLGNBQWMsRUFBRSxDQUFDO29CQUN2QixjQUFjLENBQUMsS0FBSyxFQUFFLENBQUM7b0JBQ3ZCLE1BQU07WUFDZCxDQUFDO1FBQ0wsQ0FBQztJQUNMLENBQUM7dUdBckNRLDhCQUE4QjsyRkFBOUIsOEJBQThCOzsyRkFBOUIsOEJBQThCO2tCQUoxQyxTQUFTO21CQUFDO29CQUNQLFFBQVEsRUFBRSwyQkFBMkI7b0JBQ3JDLFVBQVUsRUFBRSxJQUFJO2lCQUNuQjs4QkFTRyxPQUFPO3NCQUROLFlBQVk7dUJBQUMsT0FBTyxFQUFFLENBQUMsUUFBUSxDQUFDO2dCQWFqQyxTQUFTO3NCQURSLFlBQVk7dUJBQUMsU0FBUyxFQUFFLENBQUMsUUFBUSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRGlyZWN0aXZlLCBFbGVtZW50UmVmLCBIb3N0TGlzdGVuZXIsIGluamVjdCwgaW5wdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEFjdGlvbkRyb3Bkb3duQ29tcG9uZW50IH0gZnJvbSAnLi9hY3Rpb24tZHJvcGRvd24uY29tcG9uZW50JztcblxuLyoqXG4gKiBEaXJlY3RpdmUgdGhhdCB0dXJucyBhbnkgZWxlbWVudCBpbnRvIGEgdHJpZ2dlciBmb3IgYW4gQWN0aW9uRHJvcGRvd24gY29tcG9uZW50LlxuICogSGFuZGxlcyBjbGljayBhbmQga2V5Ym9hcmQgaW50ZXJhY3Rpb25zIHRvIG9wZW4vY2xvc2UgdGhlIGRyb3Bkb3duLlxuICovXG5ARGlyZWN0aXZlKHtcbiAgICBzZWxlY3RvcjogJ1thcEFjdGlvbkRyb3Bkb3duVHJpZ2dlcl0nLFxuICAgIHN0YW5kYWxvbmU6IHRydWUsXG59KVxuZXhwb3J0IGNsYXNzIEFjdGlvbkRyb3Bkb3duVHJpZ2dlckRpcmVjdGl2ZSB7XG4gICAgcHJpdmF0ZSByZWFkb25seSBlbGVtZW50UmVmID0gaW5qZWN0KEVsZW1lbnRSZWY8SFRNTEVsZW1lbnQ+KTtcblxuICAgIC8qKiBSZWZlcmVuY2UgdG8gdGhlIEFjdGlvbkRyb3Bkb3duIGNvbXBvbmVudCB0aGF0IHRoaXMgdHJpZ2dlciBjb250cm9scyAqL1xuICAgIGFwQWN0aW9uRHJvcGRvd25UcmlnZ2VyID0gaW5wdXQucmVxdWlyZWQ8QWN0aW9uRHJvcGRvd25Db21wb25lbnQ+KCk7XG5cbiAgICAvKiogSGFuZGxlcyBjbGljayBldmVudHMgdG8gdG9nZ2xlIHRoZSBkcm9wZG93biAqL1xuICAgIEBIb3N0TGlzdGVuZXIoJ2NsaWNrJywgWyckZXZlbnQnXSlcbiAgICBvbkNsaWNrKGV2ZW50OiBNb3VzZUV2ZW50KTogdm9pZCB7XG4gICAgICAgIGV2ZW50LnByZXZlbnREZWZhdWx0KCk7XG4gICAgICAgIGV2ZW50LnN0b3BQcm9wYWdhdGlvbigpO1xuXG4gICAgICAgIGNvbnN0IGFjdGlvbkRyb3Bkb3duID0gdGhpcy5hcEFjdGlvbkRyb3Bkb3duVHJpZ2dlcigpO1xuICAgICAgICBpZiAoYWN0aW9uRHJvcGRvd24pIHtcbiAgICAgICAgICAgIGFjdGlvbkRyb3Bkb3duLnRvZ2dsZSh0aGlzLmVsZW1lbnRSZWYubmF0aXZlRWxlbWVudCk7XG4gICAgICAgIH1cbiAgICB9XG5cbiAgICAvKiogSGFuZGxlcyBrZXlib2FyZCBldmVudHMgZm9yIGFjY2Vzc2liaWxpdHkgKEVudGVyLCBTcGFjZSwgQXJyb3cgRG93biwgRXNjYXBlKSAqL1xuICAgIEBIb3N0TGlzdGVuZXIoJ2tleWRvd24nLCBbJyRldmVudCddKVxuICAgIG9uS2V5RG93bihldmVudDogS2V5Ym9hcmRFdmVudCk6IHZvaWQge1xuICAgICAgICBjb25zdCBhY3Rpb25Ecm9wZG93biA9IHRoaXMuYXBBY3Rpb25Ecm9wZG93blRyaWdnZXIoKTtcbiAgICAgICAgaWYgKGFjdGlvbkRyb3Bkb3duKSB7XG4gICAgICAgICAgICBzd2l0Y2ggKGV2ZW50LmtleSkge1xuICAgICAgICAgICAgICAgIGNhc2UgJ0VudGVyJzpcbiAgICAgICAgICAgICAgICBjYXNlICcgJzpcbiAgICAgICAgICAgICAgICBjYXNlICdBcnJvd0Rvd24nOlxuICAgICAgICAgICAgICAgICAgICBldmVudC5wcmV2ZW50RGVmYXVsdCgpO1xuICAgICAgICAgICAgICAgICAgICBhY3Rpb25Ecm9wZG93bi5vcGVuKHRoaXMuZWxlbWVudFJlZi5uYXRpdmVFbGVtZW50KTtcbiAgICAgICAgICAgICAgICAgICAgYnJlYWs7XG5cbiAgICAgICAgICAgICAgICBjYXNlICdFc2NhcGUnOlxuICAgICAgICAgICAgICAgICAgICBldmVudC5wcmV2ZW50RGVmYXVsdCgpO1xuICAgICAgICAgICAgICAgICAgICBhY3Rpb25Ecm9wZG93bi5jbG9zZSgpO1xuICAgICAgICAgICAgICAgICAgICBicmVhaztcbiAgICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgIH1cbn1cbiJdfQ==
|
|
@@ -0,0 +1,270 @@
|
|
|
1
|
+
import { BadgeComponent } from '@agorapulse/ui-components/badge';
|
|
2
|
+
import { TooltipDirective } from '@agorapulse/ui-components/tooltip';
|
|
3
|
+
import { apFeatureLock, apInfo, SymbolComponent, withSymbols } from '@agorapulse/ui-symbol';
|
|
4
|
+
import { Overlay, OverlayPositionBuilder } from '@angular/cdk/overlay';
|
|
5
|
+
import { TemplatePortal } from '@angular/cdk/portal';
|
|
6
|
+
import { CommonModule } from '@angular/common';
|
|
7
|
+
import { ChangeDetectionStrategy, Component, computed, contentChild, effect, ElementRef, inject, input, output, signal, viewChild, ViewContainerRef, } from '@angular/core';
|
|
8
|
+
import { take } from 'rxjs/operators';
|
|
9
|
+
import * as i0 from "@angular/core";
|
|
10
|
+
/**
|
|
11
|
+
* A dropdown component that displays a list of actionable items with support for icons,
|
|
12
|
+
* badges, tooltips, and keyboard navigation.
|
|
13
|
+
*/
|
|
14
|
+
export class ActionDropdownComponent {
|
|
15
|
+
elementRef = inject(ElementRef);
|
|
16
|
+
overlay = inject(Overlay);
|
|
17
|
+
positionBuilder = inject(OverlayPositionBuilder);
|
|
18
|
+
viewContainerRef = inject(ViewContainerRef);
|
|
19
|
+
actionDropdownTemplate = viewChild('actionDropdownTemplate');
|
|
20
|
+
trigger = contentChild('trigger');
|
|
21
|
+
/** Array of items to display in the dropdown menu */
|
|
22
|
+
items = input([]);
|
|
23
|
+
/** Whether the dropdown is disabled and cannot be opened */
|
|
24
|
+
disabled = input(false);
|
|
25
|
+
/** Whether to enable large mode styling for the dropdown */
|
|
26
|
+
largeModeEnabled = input(false);
|
|
27
|
+
/** Whether to show a backdrop that closes the dropdown when clicked */
|
|
28
|
+
showBackdrop = input(true);
|
|
29
|
+
/** Custom width for the dropdown menu in pixels */
|
|
30
|
+
customWidth = input(null);
|
|
31
|
+
/** Default position for the dropdown relative to the trigger element */
|
|
32
|
+
defaultPosition = input('right');
|
|
33
|
+
/** Emits when a dropdown item is clicked */
|
|
34
|
+
itemClick = output();
|
|
35
|
+
/** Emits when the dropdown menu is opened */
|
|
36
|
+
opened = output();
|
|
37
|
+
/** Emits when the dropdown menu is closed */
|
|
38
|
+
closed = output();
|
|
39
|
+
overlayRef = null;
|
|
40
|
+
portal = null;
|
|
41
|
+
isOpen = signal(false);
|
|
42
|
+
focusedIndex = signal(-1);
|
|
43
|
+
itemsWithoutDividers = computed(() => this.items().filter(item => !item.divider));
|
|
44
|
+
constructor() {
|
|
45
|
+
// Set up keyboard navigation when dropdown opens
|
|
46
|
+
effect(() => {
|
|
47
|
+
if (this.isOpen()) {
|
|
48
|
+
this.setupKeyboardNavigation();
|
|
49
|
+
}
|
|
50
|
+
});
|
|
51
|
+
}
|
|
52
|
+
/** Opens the dropdown menu at the specified trigger element or component's element */
|
|
53
|
+
open(triggerElement) {
|
|
54
|
+
const actionDropdownTemplate = this.actionDropdownTemplate();
|
|
55
|
+
if (this.disabled() || this.isOpen() || !actionDropdownTemplate) {
|
|
56
|
+
return;
|
|
57
|
+
}
|
|
58
|
+
const target = triggerElement || this.elementRef.nativeElement;
|
|
59
|
+
if (this.overlayRef) {
|
|
60
|
+
this.overlayRef.dispose();
|
|
61
|
+
}
|
|
62
|
+
const positionStrategy = this.createPositionStrategy(target);
|
|
63
|
+
this.overlayRef = this.overlay.create({
|
|
64
|
+
positionStrategy,
|
|
65
|
+
hasBackdrop: this.showBackdrop(),
|
|
66
|
+
backdropClass: '',
|
|
67
|
+
panelClass: '',
|
|
68
|
+
scrollStrategy: this.overlay.scrollStrategies.reposition(),
|
|
69
|
+
});
|
|
70
|
+
this.portal = new TemplatePortal(actionDropdownTemplate, this.viewContainerRef);
|
|
71
|
+
this.overlayRef.attach(this.portal);
|
|
72
|
+
this.isOpen.set(true);
|
|
73
|
+
this.focusedIndex.set(-1);
|
|
74
|
+
this.opened.emit();
|
|
75
|
+
this.overlayRef
|
|
76
|
+
.backdropClick()
|
|
77
|
+
.pipe(take(1))
|
|
78
|
+
.subscribe(() => {
|
|
79
|
+
this.close();
|
|
80
|
+
});
|
|
81
|
+
this.overlayRef
|
|
82
|
+
.keydownEvents()
|
|
83
|
+
.pipe(take(1))
|
|
84
|
+
.subscribe((event) => {
|
|
85
|
+
if (event.key === 'Escape') {
|
|
86
|
+
this.close();
|
|
87
|
+
}
|
|
88
|
+
});
|
|
89
|
+
}
|
|
90
|
+
/** Closes the dropdown menu and cleans up overlay resources */
|
|
91
|
+
close() {
|
|
92
|
+
if (!this.isOpen()) {
|
|
93
|
+
return;
|
|
94
|
+
}
|
|
95
|
+
if (this.overlayRef) {
|
|
96
|
+
this.overlayRef.dispose();
|
|
97
|
+
this.overlayRef = null;
|
|
98
|
+
}
|
|
99
|
+
this.portal = null;
|
|
100
|
+
this.isOpen.set(false);
|
|
101
|
+
this.focusedIndex.set(-1);
|
|
102
|
+
this.closed.emit();
|
|
103
|
+
}
|
|
104
|
+
/** Toggles the dropdown menu open or closed state */
|
|
105
|
+
toggle(triggerElement) {
|
|
106
|
+
if (this.isOpen()) {
|
|
107
|
+
this.close();
|
|
108
|
+
}
|
|
109
|
+
else {
|
|
110
|
+
this.open(triggerElement);
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
/** Handles item click events, emitting the selected item and closing the dropdown */
|
|
114
|
+
onItemClick(item) {
|
|
115
|
+
if (item.disabled) {
|
|
116
|
+
return;
|
|
117
|
+
}
|
|
118
|
+
this.itemClick.emit(item);
|
|
119
|
+
this.close();
|
|
120
|
+
}
|
|
121
|
+
/** Handles keyboard navigation within the dropdown menu */
|
|
122
|
+
onKeyDown(event) {
|
|
123
|
+
const items = this.itemsWithoutDividers();
|
|
124
|
+
const currentIndex = this.focusedIndex();
|
|
125
|
+
let nextIndex;
|
|
126
|
+
let prevIndex;
|
|
127
|
+
switch (event.key) {
|
|
128
|
+
case 'ArrowDown':
|
|
129
|
+
event.preventDefault();
|
|
130
|
+
nextIndex = currentIndex < items.length - 1 ? currentIndex + 1 : 0;
|
|
131
|
+
this.focusedIndex.set(nextIndex);
|
|
132
|
+
break;
|
|
133
|
+
case 'ArrowUp':
|
|
134
|
+
event.preventDefault();
|
|
135
|
+
prevIndex = currentIndex > 0 ? currentIndex - 1 : items.length - 1;
|
|
136
|
+
this.focusedIndex.set(prevIndex);
|
|
137
|
+
break;
|
|
138
|
+
case 'Enter':
|
|
139
|
+
case ' ':
|
|
140
|
+
event.preventDefault();
|
|
141
|
+
if (currentIndex >= 0 && currentIndex < items.length) {
|
|
142
|
+
this.onItemClick(items[currentIndex]);
|
|
143
|
+
}
|
|
144
|
+
break;
|
|
145
|
+
case 'Escape':
|
|
146
|
+
event.preventDefault();
|
|
147
|
+
this.close();
|
|
148
|
+
break;
|
|
149
|
+
case 'Tab':
|
|
150
|
+
event.preventDefault();
|
|
151
|
+
if (event.shiftKey) {
|
|
152
|
+
// Shift+Tab - move to previous item
|
|
153
|
+
prevIndex = currentIndex > 0 ? currentIndex - 1 : items.length - 1;
|
|
154
|
+
this.focusedIndex.set(prevIndex);
|
|
155
|
+
}
|
|
156
|
+
else {
|
|
157
|
+
// Tab - move to next item
|
|
158
|
+
nextIndex = currentIndex < items.length - 1 ? currentIndex + 1 : 0;
|
|
159
|
+
this.focusedIndex.set(nextIndex);
|
|
160
|
+
}
|
|
161
|
+
break;
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
/** Creates positioning strategy for the overlay based on trigger element and default position */
|
|
165
|
+
createPositionStrategy(target) {
|
|
166
|
+
const defaultPosition = this.defaultPosition();
|
|
167
|
+
const gap = 4;
|
|
168
|
+
const positions = defaultPosition === 'right'
|
|
169
|
+
? [
|
|
170
|
+
// Bottom-right (default) - left border aligns with left side of trigger
|
|
171
|
+
{
|
|
172
|
+
originX: 'start',
|
|
173
|
+
originY: 'bottom',
|
|
174
|
+
overlayX: 'start',
|
|
175
|
+
overlayY: 'top',
|
|
176
|
+
offsetX: 0,
|
|
177
|
+
offsetY: gap,
|
|
178
|
+
},
|
|
179
|
+
// Bottom-left (if not enough space on right) - right border aligns with right side of trigger
|
|
180
|
+
{
|
|
181
|
+
originX: 'end',
|
|
182
|
+
originY: 'bottom',
|
|
183
|
+
overlayX: 'end',
|
|
184
|
+
overlayY: 'top',
|
|
185
|
+
offsetX: 0,
|
|
186
|
+
offsetY: gap,
|
|
187
|
+
},
|
|
188
|
+
// Top-right (if not enough space below) - left border aligns with left side of trigger
|
|
189
|
+
{
|
|
190
|
+
originX: 'start',
|
|
191
|
+
originY: 'top',
|
|
192
|
+
overlayX: 'start',
|
|
193
|
+
overlayY: 'bottom',
|
|
194
|
+
offsetX: 0,
|
|
195
|
+
offsetY: -gap,
|
|
196
|
+
},
|
|
197
|
+
// Top-left (if not enough space below and on right) - right border aligns with right side of trigger
|
|
198
|
+
{
|
|
199
|
+
originX: 'end',
|
|
200
|
+
originY: 'top',
|
|
201
|
+
overlayX: 'end',
|
|
202
|
+
overlayY: 'bottom',
|
|
203
|
+
offsetX: 0,
|
|
204
|
+
offsetY: -gap,
|
|
205
|
+
},
|
|
206
|
+
]
|
|
207
|
+
: [
|
|
208
|
+
// Bottom-left (default) - right border aligns with right side of trigger
|
|
209
|
+
{
|
|
210
|
+
originX: 'end',
|
|
211
|
+
originY: 'bottom',
|
|
212
|
+
overlayX: 'end',
|
|
213
|
+
overlayY: 'top',
|
|
214
|
+
offsetX: 0,
|
|
215
|
+
offsetY: gap,
|
|
216
|
+
},
|
|
217
|
+
// Bottom-right (if not enough space on left) - left border aligns with left side of trigger
|
|
218
|
+
{
|
|
219
|
+
originX: 'start',
|
|
220
|
+
originY: 'bottom',
|
|
221
|
+
overlayX: 'start',
|
|
222
|
+
overlayY: 'top',
|
|
223
|
+
offsetX: 0,
|
|
224
|
+
offsetY: gap,
|
|
225
|
+
},
|
|
226
|
+
// Top-left (if not enough space below) - right border aligns with right side of trigger
|
|
227
|
+
{
|
|
228
|
+
originX: 'end',
|
|
229
|
+
originY: 'top',
|
|
230
|
+
overlayX: 'end',
|
|
231
|
+
overlayY: 'bottom',
|
|
232
|
+
offsetX: 0,
|
|
233
|
+
offsetY: -gap,
|
|
234
|
+
},
|
|
235
|
+
// Top-right (if not enough space below and on left) - left border aligns with left side of trigger
|
|
236
|
+
{
|
|
237
|
+
originX: 'start',
|
|
238
|
+
originY: 'top',
|
|
239
|
+
overlayX: 'start',
|
|
240
|
+
overlayY: 'bottom',
|
|
241
|
+
offsetX: 0,
|
|
242
|
+
offsetY: -gap,
|
|
243
|
+
},
|
|
244
|
+
];
|
|
245
|
+
return this.positionBuilder
|
|
246
|
+
.flexibleConnectedTo(target)
|
|
247
|
+
.withPositions(positions)
|
|
248
|
+
.withFlexibleDimensions(true)
|
|
249
|
+
.withPush(true)
|
|
250
|
+
.withGrowAfterOpen(true)
|
|
251
|
+
.withViewportMargin(8);
|
|
252
|
+
}
|
|
253
|
+
/** Sets up keyboard event handling for the opened dropdown */
|
|
254
|
+
setupKeyboardNavigation() {
|
|
255
|
+
if (!this.overlayRef) {
|
|
256
|
+
return;
|
|
257
|
+
}
|
|
258
|
+
const keydownEvents = this.overlayRef.keydownEvents();
|
|
259
|
+
keydownEvents.subscribe((event) => {
|
|
260
|
+
this.onKeyDown(event);
|
|
261
|
+
});
|
|
262
|
+
}
|
|
263
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: ActionDropdownComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
264
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.9", type: ActionDropdownComponent, isStandalone: true, selector: "ap-action-dropdown", inputs: { items: { classPropertyName: "items", publicName: "items", isSignal: true, isRequired: false, transformFunction: null }, disabled: { classPropertyName: "disabled", publicName: "disabled", isSignal: true, isRequired: false, transformFunction: null }, largeModeEnabled: { classPropertyName: "largeModeEnabled", publicName: "largeModeEnabled", isSignal: true, isRequired: false, transformFunction: null }, showBackdrop: { classPropertyName: "showBackdrop", publicName: "showBackdrop", isSignal: true, isRequired: false, transformFunction: null }, customWidth: { classPropertyName: "customWidth", publicName: "customWidth", isSignal: true, isRequired: false, transformFunction: null }, defaultPosition: { classPropertyName: "defaultPosition", publicName: "defaultPosition", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { itemClick: "itemClick", opened: "opened", closed: "closed" }, providers: [withSymbols(apInfo, apFeatureLock)], queries: [{ propertyName: "trigger", first: true, predicate: ["trigger"], descendants: true, isSignal: true }], viewQueries: [{ propertyName: "actionDropdownTemplate", first: true, predicate: ["actionDropdownTemplate"], descendants: true, isSignal: true }], ngImport: i0, template: "<ng-template #actionDropdownTemplate>\n <!-- Action dropdown -->\n <div\n class=\"ap-action-dropdown__content\"\n role=\"menu\"\n tabindex=\"-1\"\n [attr.aria-label]=\"'Action dropdown'\"\n [class.ap-action-dropdown__content--default]=\"!largeModeEnabled() && !customWidth()\"\n [class.ap-action-dropdown__content--large]=\"largeModeEnabled()\"\n [style.width]=\"customWidth() ? customWidth() + 'px' : undefined\"\n (keydown)=\"onKeyDown($event)\">\n <!-- Action dropdown items -->\n @for (item of items(); track item.label) {\n <!-- Divider -->\n @if (item.divider) {\n <div\n class=\"ap-action-dropdown__divider\"\n role=\"separator\"></div>\n } @else {\n <!-- Action dropdown item -->\n <button\n type=\"button\"\n role=\"menuitem\"\n class=\"ap-action-dropdown__item\"\n [class.ap-action-dropdown__item--has-description]=\"item.description\"\n [class.ap-action-dropdown__item--disabled]=\"item.disabled\"\n [class.ap-action-dropdown__item--focused]=\"focusedIndex() === $index\"\n [class.ap-action-dropdown__item--red-mode]=\"item.redModeEnabled\"\n [class.ap-action-dropdown__item--feature-lock]=\"item.featureLockEnabled\"\n [attr.aria-disabled]=\"item.disabled\"\n [disabled]=\"item.disabled\"\n [apTooltip]=\"item.itemTooltipText\"\n (click)=\"onItemClick(item)\">\n <!-- Start icon -->\n @if (item.startSymbolId) {\n <ap-symbol\n class=\"ap-action-dropdown__item-start-icon\"\n size=\"sm\"\n [color]=\"item.startSymbolColor ? item.startSymbolColor : item.redModeEnabled ? 'red' : 'basic-grey'\"\n [symbolId]=\"item.startSymbolId\"\n [apTooltip]=\"item.startSymbolTooltipText\" />\n }\n <!-- Label -->\n <div class=\"ap-action-dropdown__item-text-container\">\n <div class=\"ap-action-dropdown__item-label-container\">\n <div\n class=\"ap-action-dropdown__item-label\"\n [class.ap-action-dropdown__item-label--bold]=\"item.description\">\n {{ item.label }}\n </div>\n <!-- Badge -->\n @if (item.badgeLabel) {\n <ap-badge color=\"blue\">\n {{ item.badgeLabel }}\n </ap-badge>\n }\n </div>\n <!-- Description -->\n <div class=\"ap-action-dropdown__item-description\">{{ item.description }}</div>\n </div>\n <!-- End icon -->\n @if (item.endSymbolId) {\n <ap-symbol\n class=\"ap-action-dropdown__item-end-icon\"\n size=\"sm\"\n color=\"#858FA1\"\n [symbolId]=\"item.endSymbolId\"\n [apTooltip]=\"item.endSymbolTooltipText\" />\n }\n <!-- Feature lock icon -->\n @if (item.featureLockEnabled) {\n <ap-symbol\n class=\"ap-action-dropdown__item-end-icon\"\n size=\"sm\"\n color=\"purple\"\n symbolId=\"feature-lock\" />\n }\n </button>\n }\n }\n </div>\n</ng-template>\n", styles: [".ap-action-dropdown__content{background-color:var(--comp-action-dropdown-background-color);border:1px solid var(--comp-action-dropdown-border-color);border-radius:var(--comp-action-dropdown-border-radius);box-shadow:var(--comp-action-dropdown-box-shadow);padding:var(--comp-action-dropdown-padding);outline:none;overflow:hidden;z-index:1000}.ap-action-dropdown__content.ap-action-dropdown__content--default{width:250px}.ap-action-dropdown__content.ap-action-dropdown__content--large{width:340px}.ap-action-dropdown__content:focus{outline:2px solid var(--comp-action-dropdown-item-background-color-focused);outline-offset:2px}.ap-action-dropdown__item{display:flex;align-items:center;width:100%;height:40px;padding:var(--comp-action-dropdown-item-padding);border:none;background:transparent;color:var(--comp-action-dropdown-item-text-color);cursor:pointer;font-family:Averta;font-size:var(--comp-action-dropdown-item-text-size);line-height:var(--comp-action-dropdown-item-text-line-height);text-align:left;transition:background-color .2s ease,color .2s ease}.ap-action-dropdown__item:hover{background-color:var(--comp-action-dropdown-item-background-color-hover)}.ap-action-dropdown__item:focus{outline:none;background-color:var(--comp-action-dropdown-item-background-color-focused)}.ap-action-dropdown__item.ap-action-dropdown__item--has-description{height:auto;min-height:50px}.ap-action-dropdown__item.ap-action-dropdown__item--focused{background-color:var(--comp-action-dropdown-item-background-color-focused)}.ap-action-dropdown__item.ap-action-dropdown__item--disabled{cursor:not-allowed;opacity:.4}.ap-action-dropdown__item.ap-action-dropdown__item--red-mode{color:var(--comp-action-dropdown-item-text-color-red-mode)}.ap-action-dropdown__item.ap-action-dropdown__item--red-mode:hover{background-color:var(--comp-action-dropdown-item-background-color-red-mode-hover)}.ap-action-dropdown__item.ap-action-dropdown__item--red-mode:focus{background-color:var(--comp-action-dropdown-item-background-color-red-mode-focused)}.ap-action-dropdown__item.ap-action-dropdown__item--red-mode.ap-action-dropdown__item--focused{background-color:var(--comp-action-dropdown-item-background-color-red-mode-focused)}.ap-action-dropdown__item.ap-action-dropdown__item--feature-lock:hover{background-color:var(--comp-action-dropdown-item-background-color-feature-lock-hover)}.ap-action-dropdown__item.ap-action-dropdown__item--feature-lock:focus{background-color:var(--comp-action-dropdown-item-background-color-feature-lock-focused)}.ap-action-dropdown__item.ap-action-dropdown__item--feature-lock.ap-action-dropdown__item--focused{background-color:var(--comp-action-dropdown-item-background-color-feature-lock-focused)}.ap-action-dropdown__item.ap-action-dropdown__item--disabled:hover{background-color:transparent}.ap-action-dropdown__item .ap-action-dropdown__item-start-icon{margin-right:var(--comp-action-dropdown-item-icon-spacing)}.ap-action-dropdown__item .ap-action-dropdown__item-end-icon{margin-left:var(--comp-action-dropdown-item-icon-spacing)}.ap-action-dropdown__item .ap-action-dropdown__item-text-container{display:flex;flex-direction:column;flex:1;min-width:0}.ap-action-dropdown__item .ap-action-dropdown__item-text-container .ap-action-dropdown__item-label-container{display:flex;align-items:center;gap:var(--comp-action-dropdown-item-label-spacing);white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.ap-action-dropdown__item .ap-action-dropdown__item-text-container .ap-action-dropdown__item-label-container .ap-action-dropdown__item-label{white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.ap-action-dropdown__item .ap-action-dropdown__item-text-container .ap-action-dropdown__item-label-container .ap-action-dropdown__item-label.ap-action-dropdown__item-label--bold{font-weight:var(--ref-font-weight-bold)}.ap-action-dropdown__item .ap-action-dropdown__item-text-container .ap-action-dropdown__item-description{color:var(--comp-action-dropdown-item-description-color);font-size:var(--comp-action-dropdown-item-description-size);line-height:var(--comp-action-dropdown-item-description-line-height)}.ap-action-dropdown__divider{height:1px;background-color:var(--comp-action-dropdown-divider-color);margin:var(--comp-action-dropdown-divider-margin)}\n"], dependencies: [{ kind: "component", type: BadgeComponent, selector: "ap-badge", inputs: ["color"] }, { kind: "ngmodule", type: CommonModule }, { kind: "component", type: SymbolComponent, selector: "ap-symbol", inputs: ["symbolId", "color", "size"], outputs: ["sizeChange"] }, { kind: "directive", type: TooltipDirective, selector: "[apTooltip]", inputs: ["apTooltip", "apTooltipPosition", "apTooltipShowDelay", "apTooltipHideDelay", "apTooltipDuration", "apTooltipDisabled", "apTooltipTruncatedTextOnly", "apTooltipTemplateContext", "apTooltipVirtualScrollElement"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
265
|
+
}
|
|
266
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: ActionDropdownComponent, decorators: [{
|
|
267
|
+
type: Component,
|
|
268
|
+
args: [{ changeDetection: ChangeDetectionStrategy.OnPush, selector: 'ap-action-dropdown', standalone: true, imports: [BadgeComponent, CommonModule, SymbolComponent, TooltipDirective], providers: [withSymbols(apInfo, apFeatureLock)], template: "<ng-template #actionDropdownTemplate>\n <!-- Action dropdown -->\n <div\n class=\"ap-action-dropdown__content\"\n role=\"menu\"\n tabindex=\"-1\"\n [attr.aria-label]=\"'Action dropdown'\"\n [class.ap-action-dropdown__content--default]=\"!largeModeEnabled() && !customWidth()\"\n [class.ap-action-dropdown__content--large]=\"largeModeEnabled()\"\n [style.width]=\"customWidth() ? customWidth() + 'px' : undefined\"\n (keydown)=\"onKeyDown($event)\">\n <!-- Action dropdown items -->\n @for (item of items(); track item.label) {\n <!-- Divider -->\n @if (item.divider) {\n <div\n class=\"ap-action-dropdown__divider\"\n role=\"separator\"></div>\n } @else {\n <!-- Action dropdown item -->\n <button\n type=\"button\"\n role=\"menuitem\"\n class=\"ap-action-dropdown__item\"\n [class.ap-action-dropdown__item--has-description]=\"item.description\"\n [class.ap-action-dropdown__item--disabled]=\"item.disabled\"\n [class.ap-action-dropdown__item--focused]=\"focusedIndex() === $index\"\n [class.ap-action-dropdown__item--red-mode]=\"item.redModeEnabled\"\n [class.ap-action-dropdown__item--feature-lock]=\"item.featureLockEnabled\"\n [attr.aria-disabled]=\"item.disabled\"\n [disabled]=\"item.disabled\"\n [apTooltip]=\"item.itemTooltipText\"\n (click)=\"onItemClick(item)\">\n <!-- Start icon -->\n @if (item.startSymbolId) {\n <ap-symbol\n class=\"ap-action-dropdown__item-start-icon\"\n size=\"sm\"\n [color]=\"item.startSymbolColor ? item.startSymbolColor : item.redModeEnabled ? 'red' : 'basic-grey'\"\n [symbolId]=\"item.startSymbolId\"\n [apTooltip]=\"item.startSymbolTooltipText\" />\n }\n <!-- Label -->\n <div class=\"ap-action-dropdown__item-text-container\">\n <div class=\"ap-action-dropdown__item-label-container\">\n <div\n class=\"ap-action-dropdown__item-label\"\n [class.ap-action-dropdown__item-label--bold]=\"item.description\">\n {{ item.label }}\n </div>\n <!-- Badge -->\n @if (item.badgeLabel) {\n <ap-badge color=\"blue\">\n {{ item.badgeLabel }}\n </ap-badge>\n }\n </div>\n <!-- Description -->\n <div class=\"ap-action-dropdown__item-description\">{{ item.description }}</div>\n </div>\n <!-- End icon -->\n @if (item.endSymbolId) {\n <ap-symbol\n class=\"ap-action-dropdown__item-end-icon\"\n size=\"sm\"\n color=\"#858FA1\"\n [symbolId]=\"item.endSymbolId\"\n [apTooltip]=\"item.endSymbolTooltipText\" />\n }\n <!-- Feature lock icon -->\n @if (item.featureLockEnabled) {\n <ap-symbol\n class=\"ap-action-dropdown__item-end-icon\"\n size=\"sm\"\n color=\"purple\"\n symbolId=\"feature-lock\" />\n }\n </button>\n }\n }\n </div>\n</ng-template>\n", styles: [".ap-action-dropdown__content{background-color:var(--comp-action-dropdown-background-color);border:1px solid var(--comp-action-dropdown-border-color);border-radius:var(--comp-action-dropdown-border-radius);box-shadow:var(--comp-action-dropdown-box-shadow);padding:var(--comp-action-dropdown-padding);outline:none;overflow:hidden;z-index:1000}.ap-action-dropdown__content.ap-action-dropdown__content--default{width:250px}.ap-action-dropdown__content.ap-action-dropdown__content--large{width:340px}.ap-action-dropdown__content:focus{outline:2px solid var(--comp-action-dropdown-item-background-color-focused);outline-offset:2px}.ap-action-dropdown__item{display:flex;align-items:center;width:100%;height:40px;padding:var(--comp-action-dropdown-item-padding);border:none;background:transparent;color:var(--comp-action-dropdown-item-text-color);cursor:pointer;font-family:Averta;font-size:var(--comp-action-dropdown-item-text-size);line-height:var(--comp-action-dropdown-item-text-line-height);text-align:left;transition:background-color .2s ease,color .2s ease}.ap-action-dropdown__item:hover{background-color:var(--comp-action-dropdown-item-background-color-hover)}.ap-action-dropdown__item:focus{outline:none;background-color:var(--comp-action-dropdown-item-background-color-focused)}.ap-action-dropdown__item.ap-action-dropdown__item--has-description{height:auto;min-height:50px}.ap-action-dropdown__item.ap-action-dropdown__item--focused{background-color:var(--comp-action-dropdown-item-background-color-focused)}.ap-action-dropdown__item.ap-action-dropdown__item--disabled{cursor:not-allowed;opacity:.4}.ap-action-dropdown__item.ap-action-dropdown__item--red-mode{color:var(--comp-action-dropdown-item-text-color-red-mode)}.ap-action-dropdown__item.ap-action-dropdown__item--red-mode:hover{background-color:var(--comp-action-dropdown-item-background-color-red-mode-hover)}.ap-action-dropdown__item.ap-action-dropdown__item--red-mode:focus{background-color:var(--comp-action-dropdown-item-background-color-red-mode-focused)}.ap-action-dropdown__item.ap-action-dropdown__item--red-mode.ap-action-dropdown__item--focused{background-color:var(--comp-action-dropdown-item-background-color-red-mode-focused)}.ap-action-dropdown__item.ap-action-dropdown__item--feature-lock:hover{background-color:var(--comp-action-dropdown-item-background-color-feature-lock-hover)}.ap-action-dropdown__item.ap-action-dropdown__item--feature-lock:focus{background-color:var(--comp-action-dropdown-item-background-color-feature-lock-focused)}.ap-action-dropdown__item.ap-action-dropdown__item--feature-lock.ap-action-dropdown__item--focused{background-color:var(--comp-action-dropdown-item-background-color-feature-lock-focused)}.ap-action-dropdown__item.ap-action-dropdown__item--disabled:hover{background-color:transparent}.ap-action-dropdown__item .ap-action-dropdown__item-start-icon{margin-right:var(--comp-action-dropdown-item-icon-spacing)}.ap-action-dropdown__item .ap-action-dropdown__item-end-icon{margin-left:var(--comp-action-dropdown-item-icon-spacing)}.ap-action-dropdown__item .ap-action-dropdown__item-text-container{display:flex;flex-direction:column;flex:1;min-width:0}.ap-action-dropdown__item .ap-action-dropdown__item-text-container .ap-action-dropdown__item-label-container{display:flex;align-items:center;gap:var(--comp-action-dropdown-item-label-spacing);white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.ap-action-dropdown__item .ap-action-dropdown__item-text-container .ap-action-dropdown__item-label-container .ap-action-dropdown__item-label{white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.ap-action-dropdown__item .ap-action-dropdown__item-text-container .ap-action-dropdown__item-label-container .ap-action-dropdown__item-label.ap-action-dropdown__item-label--bold{font-weight:var(--ref-font-weight-bold)}.ap-action-dropdown__item .ap-action-dropdown__item-text-container .ap-action-dropdown__item-description{color:var(--comp-action-dropdown-item-description-color);font-size:var(--comp-action-dropdown-item-description-size);line-height:var(--comp-action-dropdown-item-description-line-height)}.ap-action-dropdown__divider{height:1px;background-color:var(--comp-action-dropdown-divider-color);margin:var(--comp-action-dropdown-divider-margin)}\n"] }]
|
|
269
|
+
}], ctorParameters: () => [] });
|
|
270
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"action-dropdown.component.js","sourceRoot":"","sources":["../../../../libs/ui-components/action-dropdown/src/action-dropdown.component.ts","../../../../libs/ui-components/action-dropdown/src/action-dropdown.component.html"],"names":[],"mappings":"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","sourcesContent":["import { BadgeComponent } from '@agorapulse/ui-components/badge';\nimport { TooltipDirective } from '@agorapulse/ui-components/tooltip';\nimport { agorapulseSymbol, apFeatureLock, apInfo, SymbolColor, SymbolComponent, withSymbols } from '@agorapulse/ui-symbol';\nimport { ConnectedPosition, FlexibleConnectedPositionStrategy, Overlay, OverlayPositionBuilder, OverlayRef } from '@angular/cdk/overlay';\nimport { TemplatePortal } from '@angular/cdk/portal';\nimport { CommonModule } from '@angular/common';\nimport {\n    ChangeDetectionStrategy,\n    Component,\n    computed,\n    contentChild,\n    effect,\n    ElementRef,\n    inject,\n    input,\n    output,\n    signal,\n    TemplateRef,\n    viewChild,\n    ViewContainerRef,\n} from '@angular/core';\nimport { take } from 'rxjs/operators';\n\nexport interface ActionDropdownItem {\n    /** Unique identifier for the item */\n    name?: string;\n    /** Display text for the item */\n    label?: string;\n    /** Secondary text displayed below the label */\n    description?: string;\n    /** Text displayed in a badge next to the item */\n    badgeLabel?: string;\n    /** Whether to show a feature lock icon */\n    featureLockEnabled?: boolean;\n    /** Whether to apply red styling for destructive actions */\n    redModeEnabled?: boolean;\n    /** Tooltip text displayed when hovering over the item */\n    itemTooltipText?: string;\n    /** Icon symbol identifier displayed at the start of the item */\n    startSymbolId?: agorapulseSymbol;\n    /** Color theme for the start symbol */\n    startSymbolColor?: SymbolColor;\n    /** Tooltip text for the start symbol */\n    startSymbolTooltipText?: string;\n    /** Icon symbol identifier displayed at the end of the item */\n    endSymbolId?: agorapulseSymbol;\n    /** Tooltip text for the end symbol */\n    endSymbolTooltipText?: string;\n    /** Whether the item is disabled and non-interactive */\n    disabled?: boolean;\n    /** Whether to render this item as a visual divider */\n    divider?: boolean;\n}\n\n/**\n * A dropdown component that displays a list of actionable items with support for icons,\n * badges, tooltips, and keyboard navigation.\n */\n@Component({\n    changeDetection: ChangeDetectionStrategy.OnPush,\n    selector: 'ap-action-dropdown',\n    templateUrl: './action-dropdown.component.html',\n    styleUrls: ['./action-dropdown.component.scss'],\n    standalone: true,\n    imports: [BadgeComponent, CommonModule, SymbolComponent, TooltipDirective],\n    providers: [withSymbols(apInfo, apFeatureLock)],\n})\nexport class ActionDropdownComponent {\n    private readonly elementRef = inject(ElementRef);\n    private readonly overlay = inject(Overlay);\n    private readonly positionBuilder = inject(OverlayPositionBuilder);\n    private readonly viewContainerRef = inject(ViewContainerRef);\n\n    actionDropdownTemplate = viewChild<TemplateRef<unknown>>('actionDropdownTemplate');\n    trigger = contentChild<TemplateRef<unknown>>('trigger');\n\n    /** Array of items to display in the dropdown menu */\n    items = input<ActionDropdownItem[]>([]);\n    /** Whether the dropdown is disabled and cannot be opened */\n    disabled = input(false);\n    /** Whether to enable large mode styling for the dropdown */\n    largeModeEnabled = input(false);\n    /** Whether to show a backdrop that closes the dropdown when clicked */\n    showBackdrop = input(true);\n    /** Custom width for the dropdown menu in pixels */\n    customWidth = input<number | null>(null);\n    /** Default position for the dropdown relative to the trigger element */\n    defaultPosition = input<'right' | 'left'>('right');\n\n    /** Emits when a dropdown item is clicked */\n    itemClick = output<ActionDropdownItem>();\n    /** Emits when the dropdown menu is opened */\n    opened = output<void>();\n    /** Emits when the dropdown menu is closed */\n    closed = output<void>();\n\n    private overlayRef: OverlayRef | null = null;\n    private portal: TemplatePortal<unknown> | null = null;\n\n    protected readonly isOpen = signal(false);\n    protected readonly focusedIndex = signal(-1);\n    protected readonly itemsWithoutDividers = computed(() => this.items().filter(item => !item.divider));\n\n    constructor() {\n        // Set up keyboard navigation when dropdown opens\n        effect(() => {\n            if (this.isOpen()) {\n                this.setupKeyboardNavigation();\n            }\n        });\n    }\n\n    /** Opens the dropdown menu at the specified trigger element or component's element */\n    open(triggerElement?: HTMLElement): void {\n        const actionDropdownTemplate = this.actionDropdownTemplate();\n\n        if (this.disabled() || this.isOpen() || !actionDropdownTemplate) {\n            return;\n        }\n\n        const target = triggerElement || this.elementRef.nativeElement;\n\n        if (this.overlayRef) {\n            this.overlayRef.dispose();\n        }\n\n        const positionStrategy = this.createPositionStrategy(target);\n\n        this.overlayRef = this.overlay.create({\n            positionStrategy,\n            hasBackdrop: this.showBackdrop(),\n            backdropClass: '',\n            panelClass: '',\n            scrollStrategy: this.overlay.scrollStrategies.reposition(),\n        });\n\n        this.portal = new TemplatePortal(actionDropdownTemplate, this.viewContainerRef);\n        this.overlayRef.attach(this.portal);\n\n        this.isOpen.set(true);\n        this.focusedIndex.set(-1);\n        this.opened.emit();\n\n        this.overlayRef\n            .backdropClick()\n            .pipe(take(1))\n            .subscribe(() => {\n                this.close();\n            });\n\n        this.overlayRef\n            .keydownEvents()\n            .pipe(take(1))\n            .subscribe((event: KeyboardEvent) => {\n                if (event.key === 'Escape') {\n                    this.close();\n                }\n            });\n    }\n\n    /** Closes the dropdown menu and cleans up overlay resources */\n    close(): void {\n        if (!this.isOpen()) {\n            return;\n        }\n\n        if (this.overlayRef) {\n            this.overlayRef.dispose();\n            this.overlayRef = null;\n        }\n\n        this.portal = null;\n        this.isOpen.set(false);\n        this.focusedIndex.set(-1);\n        this.closed.emit();\n    }\n\n    /** Toggles the dropdown menu open or closed state */\n    toggle(triggerElement?: HTMLElement): void {\n        if (this.isOpen()) {\n            this.close();\n        } else {\n            this.open(triggerElement);\n        }\n    }\n\n    /** Handles item click events, emitting the selected item and closing the dropdown */\n    onItemClick(item: ActionDropdownItem): void {\n        if (item.disabled) {\n            return;\n        }\n\n        this.itemClick.emit(item);\n\n        this.close();\n    }\n\n    /** Handles keyboard navigation within the dropdown menu */\n    onKeyDown(event: KeyboardEvent): void {\n        const items = this.itemsWithoutDividers();\n        const currentIndex = this.focusedIndex();\n        let nextIndex: number;\n        let prevIndex: number;\n\n        switch (event.key) {\n            case 'ArrowDown':\n                event.preventDefault();\n                nextIndex = currentIndex < items.length - 1 ? currentIndex + 1 : 0;\n                this.focusedIndex.set(nextIndex);\n                break;\n\n            case 'ArrowUp':\n                event.preventDefault();\n                prevIndex = currentIndex > 0 ? currentIndex - 1 : items.length - 1;\n                this.focusedIndex.set(prevIndex);\n                break;\n\n            case 'Enter':\n            case ' ':\n                event.preventDefault();\n                if (currentIndex >= 0 && currentIndex < items.length) {\n                    this.onItemClick(items[currentIndex]);\n                }\n                break;\n\n            case 'Escape':\n                event.preventDefault();\n                this.close();\n                break;\n\n            case 'Tab':\n                event.preventDefault();\n                if (event.shiftKey) {\n                    // Shift+Tab - move to previous item\n                    prevIndex = currentIndex > 0 ? currentIndex - 1 : items.length - 1;\n                    this.focusedIndex.set(prevIndex);\n                } else {\n                    // Tab - move to next item\n                    nextIndex = currentIndex < items.length - 1 ? currentIndex + 1 : 0;\n                    this.focusedIndex.set(nextIndex);\n                }\n                break;\n        }\n    }\n\n    /** Creates positioning strategy for the overlay based on trigger element and default position */\n    private createPositionStrategy(target: HTMLElement): FlexibleConnectedPositionStrategy {\n        const defaultPosition = this.defaultPosition();\n        const gap = 4;\n\n        const positions: ConnectedPosition[] =\n            defaultPosition === 'right'\n                ? [\n                      // Bottom-right (default) - left border aligns with left side of trigger\n                      {\n                          originX: 'start',\n                          originY: 'bottom',\n                          overlayX: 'start',\n                          overlayY: 'top',\n                          offsetX: 0,\n                          offsetY: gap,\n                      },\n                      // Bottom-left (if not enough space on right) - right border aligns with right side of trigger\n                      {\n                          originX: 'end',\n                          originY: 'bottom',\n                          overlayX: 'end',\n                          overlayY: 'top',\n                          offsetX: 0,\n                          offsetY: gap,\n                      },\n                      // Top-right (if not enough space below) - left border aligns with left side of trigger\n                      {\n                          originX: 'start',\n                          originY: 'top',\n                          overlayX: 'start',\n                          overlayY: 'bottom',\n                          offsetX: 0,\n                          offsetY: -gap,\n                      },\n                      // Top-left (if not enough space below and on right) - right border aligns with right side of trigger\n                      {\n                          originX: 'end',\n                          originY: 'top',\n                          overlayX: 'end',\n                          overlayY: 'bottom',\n                          offsetX: 0,\n                          offsetY: -gap,\n                      },\n                  ]\n                : [\n                      // Bottom-left (default) - right border aligns with right side of trigger\n                      {\n                          originX: 'end',\n                          originY: 'bottom',\n                          overlayX: 'end',\n                          overlayY: 'top',\n                          offsetX: 0,\n                          offsetY: gap,\n                      },\n                      // Bottom-right (if not enough space on left) - left border aligns with left side of trigger\n                      {\n                          originX: 'start',\n                          originY: 'bottom',\n                          overlayX: 'start',\n                          overlayY: 'top',\n                          offsetX: 0,\n                          offsetY: gap,\n                      },\n                      // Top-left (if not enough space below) - right border aligns with right side of trigger\n                      {\n                          originX: 'end',\n                          originY: 'top',\n                          overlayX: 'end',\n                          overlayY: 'bottom',\n                          offsetX: 0,\n                          offsetY: -gap,\n                      },\n                      // Top-right (if not enough space below and on left) - left border aligns with left side of trigger\n                      {\n                          originX: 'start',\n                          originY: 'top',\n                          overlayX: 'start',\n                          overlayY: 'bottom',\n                          offsetX: 0,\n                          offsetY: -gap,\n                      },\n                  ];\n\n        return this.positionBuilder\n            .flexibleConnectedTo(target)\n            .withPositions(positions)\n            .withFlexibleDimensions(true)\n            .withPush(true)\n            .withGrowAfterOpen(true)\n            .withViewportMargin(8);\n    }\n\n    /** Sets up keyboard event handling for the opened dropdown */\n    private setupKeyboardNavigation(): void {\n        if (!this.overlayRef) {\n            return;\n        }\n\n        const keydownEvents = this.overlayRef.keydownEvents();\n\n        keydownEvents.subscribe((event: KeyboardEvent) => {\n            this.onKeyDown(event);\n        });\n    }\n}\n","<ng-template #actionDropdownTemplate>\n    <!-- Action dropdown -->\n    <div\n        class=\"ap-action-dropdown__content\"\n        role=\"menu\"\n        tabindex=\"-1\"\n        [attr.aria-label]=\"'Action dropdown'\"\n        [class.ap-action-dropdown__content--default]=\"!largeModeEnabled() && !customWidth()\"\n        [class.ap-action-dropdown__content--large]=\"largeModeEnabled()\"\n        [style.width]=\"customWidth() ? customWidth() + 'px' : undefined\"\n        (keydown)=\"onKeyDown($event)\">\n        <!-- Action dropdown items -->\n        @for (item of items(); track item.label) {\n            <!-- Divider -->\n            @if (item.divider) {\n                <div\n                    class=\"ap-action-dropdown__divider\"\n                    role=\"separator\"></div>\n            } @else {\n                <!-- Action dropdown item -->\n                <button\n                    type=\"button\"\n                    role=\"menuitem\"\n                    class=\"ap-action-dropdown__item\"\n                    [class.ap-action-dropdown__item--has-description]=\"item.description\"\n                    [class.ap-action-dropdown__item--disabled]=\"item.disabled\"\n                    [class.ap-action-dropdown__item--focused]=\"focusedIndex() === $index\"\n                    [class.ap-action-dropdown__item--red-mode]=\"item.redModeEnabled\"\n                    [class.ap-action-dropdown__item--feature-lock]=\"item.featureLockEnabled\"\n                    [attr.aria-disabled]=\"item.disabled\"\n                    [disabled]=\"item.disabled\"\n                    [apTooltip]=\"item.itemTooltipText\"\n                    (click)=\"onItemClick(item)\">\n                    <!-- Start icon -->\n                    @if (item.startSymbolId) {\n                        <ap-symbol\n                            class=\"ap-action-dropdown__item-start-icon\"\n                            size=\"sm\"\n                            [color]=\"item.startSymbolColor ? item.startSymbolColor : item.redModeEnabled ? 'red' : 'basic-grey'\"\n                            [symbolId]=\"item.startSymbolId\"\n                            [apTooltip]=\"item.startSymbolTooltipText\" />\n                    }\n                    <!-- Label -->\n                    <div class=\"ap-action-dropdown__item-text-container\">\n                        <div class=\"ap-action-dropdown__item-label-container\">\n                            <div\n                                class=\"ap-action-dropdown__item-label\"\n                                [class.ap-action-dropdown__item-label--bold]=\"item.description\">\n                                {{ item.label }}\n                            </div>\n                            <!-- Badge -->\n                            @if (item.badgeLabel) {\n                                <ap-badge color=\"blue\">\n                                    {{ item.badgeLabel }}\n                                </ap-badge>\n                            }\n                        </div>\n                        <!-- Description -->\n                        <div class=\"ap-action-dropdown__item-description\">{{ item.description }}</div>\n                    </div>\n                    <!-- End icon -->\n                    @if (item.endSymbolId) {\n                        <ap-symbol\n                            class=\"ap-action-dropdown__item-end-icon\"\n                            size=\"sm\"\n                            color=\"#858FA1\"\n                            [symbolId]=\"item.endSymbolId\"\n                            [apTooltip]=\"item.endSymbolTooltipText\" />\n                    }\n                    <!-- Feature lock icon -->\n                    @if (item.featureLockEnabled) {\n                        <ap-symbol\n                            class=\"ap-action-dropdown__item-end-icon\"\n                            size=\"sm\"\n                            color=\"purple\"\n                            symbolId=\"feature-lock\" />\n                    }\n                </button>\n            }\n        }\n    </div>\n</ng-template>\n"]}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Generated bundle index. Do not edit.
|
|
3
|
+
*/
|
|
4
|
+
export * from './public_api';
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWdvcmFwdWxzZS11aS1jb21wb25lbnRzLWFjdGlvbi1kcm9wZG93bi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL2xpYnMvdWktY29tcG9uZW50cy9hY3Rpb24tZHJvcGRvd24vc3JjL2Fnb3JhcHVsc2UtdWktY29tcG9uZW50cy1hY3Rpb24tZHJvcGRvd24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFFSCxjQUFjLGNBQWMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogR2VuZXJhdGVkIGJ1bmRsZSBpbmRleC4gRG8gbm90IGVkaXQuXG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9wdWJsaWNfYXBpJztcbiJdfQ==
|
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
export { ActionDropdownTriggerDirective } from './action-dropdown-trigger.directive';
|
|
2
|
+
export { ActionDropdownComponent } from './action-dropdown.component';
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljX2FwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL2xpYnMvdWktY29tcG9uZW50cy9hY3Rpb24tZHJvcGRvd24vc3JjL3B1YmxpY19hcGkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLDhCQUE4QixFQUFFLE1BQU0scUNBQXFDLENBQUM7QUFDckYsT0FBTyxFQUFFLHVCQUF1QixFQUFzQixNQUFNLDZCQUE2QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgQWN0aW9uRHJvcGRvd25UcmlnZ2VyRGlyZWN0aXZlIH0gZnJvbSAnLi9hY3Rpb24tZHJvcGRvd24tdHJpZ2dlci5kaXJlY3RpdmUnO1xuZXhwb3J0IHsgQWN0aW9uRHJvcGRvd25Db21wb25lbnQsIEFjdGlvbkRyb3Bkb3duSXRlbSB9IGZyb20gJy4vYWN0aW9uLWRyb3Bkb3duLmNvbXBvbmVudCc7XG4iXX0=
|