@odx/angular 12.19.2 → 12.21.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +12 -0
- package/cdk/scrollable/README.md +3 -0
- package/cdk/scrollable/index.d.ts +1 -0
- package/cdk/scrollable/lib/scrollable.service.d.ts +53 -0
- package/components/autocomplete/lib/autocomplete.component.d.ts +7 -2
- package/components/tab-bar/lib/components/tab-bar-item/tab-bar-item.component.d.ts +4 -5
- package/components/tab-bar/lib/tab-bar.component.d.ts +22 -44
- package/esm2022/cdk/scrollable/index.mjs +2 -0
- package/esm2022/cdk/scrollable/lib/scrollable.service.mjs +108 -0
- package/esm2022/cdk/scrollable/odx-angular-cdk-scrollable.mjs +5 -0
- package/esm2022/components/autocomplete/lib/autocomplete.component.mjs +17 -16
- package/esm2022/components/tab-bar/lib/components/tab-bar-item/tab-bar-item.component.mjs +10 -17
- package/esm2022/components/tab-bar/lib/tab-bar.component.mjs +75 -114
- package/fesm2022/odx-angular-cdk-scrollable.mjs +115 -0
- package/fesm2022/odx-angular-cdk-scrollable.mjs.map +1 -0
- package/fesm2022/odx-angular-components-autocomplete.mjs +16 -15
- package/fesm2022/odx-angular-components-autocomplete.mjs.map +1 -1
- package/fesm2022/odx-angular-components-tab-bar.mjs +82 -128
- package/fesm2022/odx-angular-components-tab-bar.mjs.map +1 -1
- package/package.json +13 -7
package/CHANGELOG.md
CHANGED
|
@@ -1,5 +1,17 @@
|
|
|
1
1
|
# @odx/angular
|
|
2
2
|
|
|
3
|
+
## 12.21.0
|
|
4
|
+
|
|
5
|
+
### Minor Changes
|
|
6
|
+
|
|
7
|
+
- 37cb498: Adding text to display when no options are available in the autocomplete
|
|
8
|
+
|
|
9
|
+
## 12.20.0
|
|
10
|
+
|
|
11
|
+
### Minor Changes
|
|
12
|
+
|
|
13
|
+
- 0bff895: Added scroll for tab bar, z-index fixed for action buttons click event handling
|
|
14
|
+
|
|
3
15
|
## 12.19.2
|
|
4
16
|
|
|
5
17
|
### Patch Changes
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export * from './lib/scrollable.service';
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
import { ElementRef, OnDestroy } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
export interface ScrollableItem {
|
|
4
|
+
element: ElementRef<HTMLElement>;
|
|
5
|
+
}
|
|
6
|
+
/**
|
|
7
|
+
* Service to manage scrollable items within a container, providing functionality to observe item visibility
|
|
8
|
+
* and navigate through them.
|
|
9
|
+
*/
|
|
10
|
+
export declare class ScrollableService implements OnDestroy {
|
|
11
|
+
private intersectionObserver?;
|
|
12
|
+
private scrollContainer?;
|
|
13
|
+
/**
|
|
14
|
+
* References to the next and previous scrollable items.
|
|
15
|
+
*
|
|
16
|
+
* @type {HTMLElement | null}
|
|
17
|
+
*/
|
|
18
|
+
readonly nextItem: import("@angular/core").WritableSignal<HTMLElement | null>;
|
|
19
|
+
/**
|
|
20
|
+
* References to the previous scrollable item.
|
|
21
|
+
*
|
|
22
|
+
* @type {HTMLElement | null}
|
|
23
|
+
*/
|
|
24
|
+
readonly prevItem: import("@angular/core").WritableSignal<HTMLElement | null>;
|
|
25
|
+
/** List of currently visible items within the scroll container.
|
|
26
|
+
*
|
|
27
|
+
* @type {HTMLElement[]}
|
|
28
|
+
*/
|
|
29
|
+
readonly visibleItems: import("@angular/core").WritableSignal<HTMLElement[]>;
|
|
30
|
+
ngOnDestroy(): void;
|
|
31
|
+
/**
|
|
32
|
+
* Starts observing the scrollable items within the specified container.
|
|
33
|
+
*
|
|
34
|
+
* @param {ElementRef<HTMLElement>} scrollContainer - The container element that holds the scrollable items.
|
|
35
|
+
* @param {readonly ScrollableItem[]} items - The list of scrollable items to observe.
|
|
36
|
+
*/
|
|
37
|
+
observe(scrollContainer: ElementRef<HTMLElement>, items: readonly ScrollableItem[]): void;
|
|
38
|
+
/**
|
|
39
|
+
* Scrolls the container in the specified direction.
|
|
40
|
+
*
|
|
41
|
+
* @param {'next' | 'prev'} direction - The direction to scroll ('next' for right, 'prev' for left).
|
|
42
|
+
*/
|
|
43
|
+
scroll(direction?: 1 | -1): void;
|
|
44
|
+
/**
|
|
45
|
+
* Scrolls the active item into view within the scroll container.
|
|
46
|
+
*
|
|
47
|
+
* @param {HTMLElement | null} activeItem - The currently active item to scroll into view.
|
|
48
|
+
*/
|
|
49
|
+
scrollActiveItemIntoView(activeItem: HTMLElement | null): void;
|
|
50
|
+
private observeVisibility;
|
|
51
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<ScrollableService, never>;
|
|
52
|
+
static ɵprov: i0.ɵɵInjectableDeclaration<ScrollableService>;
|
|
53
|
+
}
|
|
@@ -24,6 +24,11 @@ export declare class AutocompleteComponent<T = unknown> extends AutocompleteCont
|
|
|
24
24
|
* @type {QueryList<AutocompleteOptionComponent<T>> | undefined}
|
|
25
25
|
*/
|
|
26
26
|
options?: QueryList<AutocompleteOptionComponent<T>>;
|
|
27
|
+
/** Text to display when no options are found.
|
|
28
|
+
*
|
|
29
|
+
* @type {string | null}
|
|
30
|
+
*/
|
|
31
|
+
notFoundText: import("@angular/core").InputSignal<string | null>;
|
|
27
32
|
/**
|
|
28
33
|
* The search field input control.
|
|
29
34
|
*
|
|
@@ -50,11 +55,11 @@ export declare class AutocompleteComponent<T = unknown> extends AutocompleteCont
|
|
|
50
55
|
protected clickOutside(): void;
|
|
51
56
|
protected updateDropdownState(): void;
|
|
52
57
|
protected handleClickEvent(): void;
|
|
53
|
-
protected handleKeyboardEvent(event: KeyboardEvent): void
|
|
58
|
+
protected handleKeyboardEvent(event: KeyboardEvent): Promise<void>;
|
|
54
59
|
protected activateSelectedOption(): void;
|
|
55
60
|
protected onDropdownClosed(): void;
|
|
56
61
|
private updateSearchField;
|
|
57
62
|
private blurSelectSearchField;
|
|
58
63
|
static ɵfac: i0.ɵɵFactoryDeclaration<AutocompleteComponent<any>, never>;
|
|
59
|
-
static ɵcmp: i0.ɵɵComponentDeclaration<AutocompleteComponent<any>, "odx-autocomplete", never, {}, {}, ["searchField", "options"], ["input[odxAutocompleteControl]", "*"], true, [{ directive: typeof i1.ClickOutsideDirective; inputs: {}; outputs: {}; }]>;
|
|
64
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<AutocompleteComponent<any>, "odx-autocomplete", never, { "notFoundText": { "alias": "notFoundText"; "required": false; "isSignal": true; }; }, {}, ["searchField", "options"], ["input[odxAutocompleteControl]", "*"], true, [{ directive: typeof i1.ClickOutsideDirective; inputs: {}; outputs: {}; }]>;
|
|
60
65
|
}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { Highlightable } from '@angular/cdk/a11y';
|
|
2
|
-
import { ElementRef,
|
|
2
|
+
import { ElementRef, OnInit } from '@angular/core';
|
|
3
3
|
import * as i0 from "@angular/core";
|
|
4
4
|
/**
|
|
5
5
|
* Represents an individual tab item in a tab bar interface, providing functionalities
|
|
@@ -27,13 +27,13 @@ export declare class TabBarItemComponent implements Highlightable, OnInit {
|
|
|
27
27
|
* @type {boolean}
|
|
28
28
|
* @default false
|
|
29
29
|
*/
|
|
30
|
-
closable: boolean
|
|
30
|
+
closable: import("@angular/core").InputSignalWithTransform<boolean, unknown>;
|
|
31
31
|
/**
|
|
32
32
|
* Event emitter that fires when the tab is closed. It emits the instance of the tab item component.
|
|
33
33
|
*
|
|
34
34
|
* @emits {TabBarItemComponent}
|
|
35
35
|
*/
|
|
36
|
-
tabClose:
|
|
36
|
+
tabClose: import("@angular/core").OutputEmitterRef<TabBarItemComponent>;
|
|
37
37
|
ngOnInit(): void;
|
|
38
38
|
/**
|
|
39
39
|
* Sets the tab as active and applies corresponding styles. This method is part of the
|
|
@@ -48,6 +48,5 @@ export declare class TabBarItemComponent implements Highlightable, OnInit {
|
|
|
48
48
|
protected onClick(): void;
|
|
49
49
|
protected handleClose(event: Event): void;
|
|
50
50
|
static ɵfac: i0.ɵɵFactoryDeclaration<TabBarItemComponent, never>;
|
|
51
|
-
static ɵcmp: i0.ɵɵComponentDeclaration<TabBarItemComponent, "odx-tab-bar-item", never, { "closable": { "alias": "closable"; "required": false; }; }, { "tabClose": "tabClose"; }, never, ["odx-icon", "*"], true, never>;
|
|
52
|
-
static ngAcceptInputType_closable: unknown;
|
|
51
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<TabBarItemComponent, "odx-tab-bar-item", never, { "closable": { "alias": "closable"; "required": false; "isSignal": true; }; }, { "tabClose": "tabClose"; }, never, ["odx-icon", "*"], true, never>;
|
|
53
52
|
}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { ActiveDescendantKeyManager } from '@angular/cdk/a11y';
|
|
2
|
-
import {
|
|
2
|
+
import { ElementRef } from '@angular/core';
|
|
3
3
|
import { TabBarItemComponent } from './components';
|
|
4
4
|
import { TabChangeEvent } from './models';
|
|
5
5
|
import * as i0 from "@angular/core";
|
|
@@ -13,30 +13,18 @@ import * as i0 from "@angular/core";
|
|
|
13
13
|
* making it accessible and user-friendly. Additionally, it handles overflow with navigational arrows,
|
|
14
14
|
* allowing for a smooth user experience even with a large number of tabs.
|
|
15
15
|
*/
|
|
16
|
-
export declare class TabBarComponent
|
|
17
|
-
private readonly changeDetector;
|
|
16
|
+
export declare class TabBarComponent {
|
|
18
17
|
private readonly takeUntilDestroyed;
|
|
19
|
-
private readonly
|
|
20
|
-
protected
|
|
21
|
-
protected
|
|
22
|
-
protected
|
|
23
|
-
protected
|
|
24
|
-
protected
|
|
18
|
+
private readonly scrollableService;
|
|
19
|
+
protected readonly nextTab: import("@angular/core").WritableSignal<HTMLElement | null>;
|
|
20
|
+
protected readonly prevTab: import("@angular/core").WritableSignal<HTMLElement | null>;
|
|
21
|
+
protected readonly tabs: import("@angular/core").Signal<readonly TabBarItemComponent[]>;
|
|
22
|
+
protected readonly activeItem: import("@angular/core").Signal<TabBarItemComponent | null>;
|
|
23
|
+
protected readonly keyManager: ActiveDescendantKeyManager<TabBarItemComponent>;
|
|
24
|
+
protected readonly scrollContainerElement: import("@angular/core").Signal<ElementRef<HTMLElement>>;
|
|
25
25
|
readonly element: ElementRef<HTMLElement>;
|
|
26
26
|
/**
|
|
27
|
-
*
|
|
28
|
-
*
|
|
29
|
-
* @type {boolean}
|
|
30
|
-
*/
|
|
31
|
-
overflowLeft: boolean;
|
|
32
|
-
/**
|
|
33
|
-
* Indicates whether the tab bar has overflowed to the right.
|
|
34
|
-
*
|
|
35
|
-
* @type {boolean}
|
|
36
|
-
*/
|
|
37
|
-
overflowRight: boolean;
|
|
38
|
-
/**
|
|
39
|
-
* Gets or sets the index of the currently selected tab. Use this property to programmatically
|
|
27
|
+
* Index of the currently selected tab. Use this property to programmatically
|
|
40
28
|
* change the active tab. It responds to changes by updating the active state of the corresponding
|
|
41
29
|
* tab item and adjusting the view if necessary.
|
|
42
30
|
*
|
|
@@ -47,7 +35,7 @@ export declare class TabBarComponent implements AfterViewInit {
|
|
|
47
35
|
* <odx-tab-bar [selectedIndex]="1"></odx-tab-bar>
|
|
48
36
|
* ```
|
|
49
37
|
*/
|
|
50
|
-
|
|
38
|
+
selectedIndex: import("@angular/core").ModelSignal<number>;
|
|
51
39
|
/**
|
|
52
40
|
* An EventEmitter that emits whenever a new tab is selected. It provides the `TabChangeEvent`
|
|
53
41
|
* which contains the index of the newly selected tab and the instance of the corresponding
|
|
@@ -65,35 +53,25 @@ export declare class TabBarComponent implements AfterViewInit {
|
|
|
65
53
|
* }
|
|
66
54
|
* ```
|
|
67
55
|
*/
|
|
68
|
-
selectedTabChanged:
|
|
69
|
-
|
|
56
|
+
selectedTabChanged: import("@angular/core").OutputEmitterRef<TabChangeEvent>;
|
|
57
|
+
constructor();
|
|
70
58
|
/**
|
|
71
|
-
* Activates the specified tab item.
|
|
59
|
+
* Activates the specified tab item, making it the currently selected tab.
|
|
60
|
+
* If the item is already active, no action is taken.
|
|
72
61
|
*
|
|
73
|
-
* @param {TabBarItemComponent | null} item - The tab item to activate.
|
|
74
|
-
*
|
|
75
|
-
* @example
|
|
76
|
-
* ```ts
|
|
77
|
-
* // Activate the first tab item
|
|
78
|
-
* tabBar.activateItem(tabItems[0]);
|
|
79
|
-
*
|
|
80
|
-
* // Clear the active item
|
|
81
|
-
* tabBar.activateItem(null);
|
|
82
|
-
* ```
|
|
62
|
+
* @param {TabBarItemComponent | null} item - The tab item to activate.
|
|
83
63
|
*/
|
|
84
64
|
activateItem(item?: TabBarItemComponent | null): void;
|
|
85
65
|
/**
|
|
86
|
-
*
|
|
87
|
-
* If the
|
|
66
|
+
* Fires when the specified tab item is removed from the tab bar.
|
|
67
|
+
* If the item is currently active, the previous tab is activated.
|
|
88
68
|
*
|
|
89
|
-
* @param {TabBarItemComponent} item - The item to
|
|
69
|
+
* @param {TabBarItemComponent} item - The tab item to remove.
|
|
90
70
|
*/
|
|
91
71
|
removeItem(item: TabBarItemComponent): void;
|
|
92
72
|
protected onKeydown(event: KeyboardEvent): void;
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
private activeItemChange;
|
|
73
|
+
protected onFocused(): void;
|
|
74
|
+
protected scrollTabs(direction?: 1 | -1): void;
|
|
96
75
|
static ɵfac: i0.ɵɵFactoryDeclaration<TabBarComponent, never>;
|
|
97
|
-
static ɵcmp: i0.ɵɵComponentDeclaration<TabBarComponent, "odx-tab-bar", never, { "selectedIndex": { "alias": "selectedIndex"; "required": false; }; }, { "selectedTabChanged": "selectedTabChanged"; }, ["tabs"], ["*"], true, never>;
|
|
98
|
-
static ngAcceptInputType_selectedIndex: unknown;
|
|
76
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<TabBarComponent, "odx-tab-bar", never, { "selectedIndex": { "alias": "selectedIndex"; "required": false; "isSignal": true; }; }, { "selectedIndex": "selectedIndexChange"; "selectedTabChanged": "selectedTabChanged"; }, ["tabs"], ["*"], true, never>;
|
|
99
77
|
}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export * from './lib/scrollable.service';
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9saWJzL2FuZ3VsYXIvY2RrL3Njcm9sbGFibGUvc3JjL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2xpYi9zY3JvbGxhYmxlLnNlcnZpY2UnO1xuIl19
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
import { Injectable, signal } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
/**
|
|
4
|
+
* Service to manage scrollable items within a container, providing functionality to observe item visibility
|
|
5
|
+
* and navigate through them.
|
|
6
|
+
*/
|
|
7
|
+
export class ScrollableService {
|
|
8
|
+
constructor() {
|
|
9
|
+
/**
|
|
10
|
+
* References to the next and previous scrollable items.
|
|
11
|
+
*
|
|
12
|
+
* @type {HTMLElement | null}
|
|
13
|
+
*/
|
|
14
|
+
this.nextItem = signal(null);
|
|
15
|
+
/**
|
|
16
|
+
* References to the previous scrollable item.
|
|
17
|
+
*
|
|
18
|
+
* @type {HTMLElement | null}
|
|
19
|
+
*/
|
|
20
|
+
this.prevItem = signal(null);
|
|
21
|
+
/** List of currently visible items within the scroll container.
|
|
22
|
+
*
|
|
23
|
+
* @type {HTMLElement[]}
|
|
24
|
+
*/
|
|
25
|
+
this.visibleItems = signal([]);
|
|
26
|
+
}
|
|
27
|
+
ngOnDestroy() {
|
|
28
|
+
this.intersectionObserver?.disconnect();
|
|
29
|
+
}
|
|
30
|
+
/**
|
|
31
|
+
* Starts observing the scrollable items within the specified container.
|
|
32
|
+
*
|
|
33
|
+
* @param {ElementRef<HTMLElement>} scrollContainer - The container element that holds the scrollable items.
|
|
34
|
+
* @param {readonly ScrollableItem[]} items - The list of scrollable items to observe.
|
|
35
|
+
*/
|
|
36
|
+
observe(scrollContainer, items) {
|
|
37
|
+
this.scrollContainer = scrollContainer;
|
|
38
|
+
if (this.intersectionObserver) {
|
|
39
|
+
this.intersectionObserver.disconnect();
|
|
40
|
+
}
|
|
41
|
+
const allItems = items.map((item) => item.element.nativeElement);
|
|
42
|
+
if (!this.scrollContainer || allItems.length === 0) {
|
|
43
|
+
this.nextItem.set(null);
|
|
44
|
+
this.prevItem.set(null);
|
|
45
|
+
return;
|
|
46
|
+
}
|
|
47
|
+
this.intersectionObserver = this.observeVisibility(allItems, (visible) => {
|
|
48
|
+
let prevItem = null;
|
|
49
|
+
let nextItem = null;
|
|
50
|
+
if (visible.length > 0 && allItems.length > visible.length) {
|
|
51
|
+
const firstVisibleIndex = allItems.indexOf(visible[0]);
|
|
52
|
+
if (firstVisibleIndex > 0) {
|
|
53
|
+
prevItem = allItems[firstVisibleIndex - 1];
|
|
54
|
+
}
|
|
55
|
+
const lastVisibleIndex = allItems.indexOf(visible[visible.length - 1]);
|
|
56
|
+
if (lastVisibleIndex < allItems.length - 1) {
|
|
57
|
+
nextItem = allItems[lastVisibleIndex + 1];
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
this.nextItem.set(nextItem);
|
|
61
|
+
this.prevItem.set(prevItem);
|
|
62
|
+
});
|
|
63
|
+
}
|
|
64
|
+
/**
|
|
65
|
+
* Scrolls the container in the specified direction.
|
|
66
|
+
*
|
|
67
|
+
* @param {'next' | 'prev'} direction - The direction to scroll ('next' for right, 'prev' for left).
|
|
68
|
+
*/
|
|
69
|
+
scroll(direction = 1) {
|
|
70
|
+
const container = this.scrollContainer?.nativeElement;
|
|
71
|
+
if (!container)
|
|
72
|
+
return;
|
|
73
|
+
container.scrollBy({ left: (container.clientWidth * direction) / 2 });
|
|
74
|
+
}
|
|
75
|
+
/**
|
|
76
|
+
* Scrolls the active item into view within the scroll container.
|
|
77
|
+
*
|
|
78
|
+
* @param {HTMLElement | null} activeItem - The currently active item to scroll into view.
|
|
79
|
+
*/
|
|
80
|
+
scrollActiveItemIntoView(activeItem) {
|
|
81
|
+
if (activeItem) {
|
|
82
|
+
activeItem.scrollIntoView({ behavior: 'smooth', block: 'nearest', inline: 'center' });
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
observeVisibility(items, callback) {
|
|
86
|
+
const visibleMap = new Map();
|
|
87
|
+
const observer = new IntersectionObserver((entries) => {
|
|
88
|
+
for (const entry of entries) {
|
|
89
|
+
const isVisible = entry.isIntersecting && entry.intersectionRatio >= 0.75;
|
|
90
|
+
visibleMap.set(entry.target, isVisible);
|
|
91
|
+
}
|
|
92
|
+
const visible = items.filter((item) => visibleMap.get(item));
|
|
93
|
+
this.visibleItems.set(visible);
|
|
94
|
+
callback(visible);
|
|
95
|
+
}, {
|
|
96
|
+
root: this.scrollContainer.nativeElement,
|
|
97
|
+
threshold: 1,
|
|
98
|
+
});
|
|
99
|
+
items.forEach((item) => observer.observe(item));
|
|
100
|
+
return observer;
|
|
101
|
+
}
|
|
102
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScrollableService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
103
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScrollableService }); }
|
|
104
|
+
}
|
|
105
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScrollableService, decorators: [{
|
|
106
|
+
type: Injectable
|
|
107
|
+
}] });
|
|
108
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scrollable.service.js","sourceRoot":"","sources":["../../../../../../../libs/angular/cdk/scrollable/src/lib/scrollable.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { ElementRef, Injectable, OnDestroy, signal } from '@angular/core';\n\nexport interface ScrollableItem {\n  element: ElementRef<HTMLElement>;\n}\n\n/**\n * Service to manage scrollable items within a container, providing functionality to observe item visibility\n * and navigate through them.\n */\n@Injectable()\nexport class ScrollableService implements OnDestroy {\n  private intersectionObserver?: IntersectionObserver;\n  private scrollContainer?: ElementRef<HTMLElement>;\n\n  /**\n   * References to the next and previous scrollable items.\n   *\n   * @type {HTMLElement | null}\n   */\n  public readonly nextItem = signal<HTMLElement | null>(null);\n\n  /**\n   * References to the previous scrollable item.\n   *\n   * @type {HTMLElement | null}\n   */\n  public readonly prevItem = signal<HTMLElement | null>(null);\n\n  /** List of currently visible items within the scroll container.\n   *\n   * @type {HTMLElement[]}\n   */\n  public readonly visibleItems = signal<HTMLElement[]>([]);\n\n  public ngOnDestroy(): void {\n    this.intersectionObserver?.disconnect();\n  }\n\n  /**\n   * Starts observing the scrollable items within the specified container.\n   *\n   * @param {ElementRef<HTMLElement>} scrollContainer - The container element that holds the scrollable items.\n   * @param {readonly ScrollableItem[]} items - The list of scrollable items to observe.\n   */\n  public observe(scrollContainer: ElementRef<HTMLElement>, items: readonly ScrollableItem[]): void {\n    this.scrollContainer = scrollContainer;\n    if (this.intersectionObserver) {\n      this.intersectionObserver.disconnect();\n    }\n\n    const allItems = items.map((item) => item.element.nativeElement);\n    if (!this.scrollContainer || allItems.length === 0) {\n      this.nextItem.set(null);\n      this.prevItem.set(null);\n      return;\n    }\n\n    this.intersectionObserver = this.observeVisibility(allItems, (visible) => {\n      let prevItem: HTMLElement | null = null;\n      let nextItem: HTMLElement | null = null;\n\n      if (visible.length > 0 && allItems.length > visible.length) {\n        const firstVisibleIndex = allItems.indexOf(visible[0]);\n        if (firstVisibleIndex > 0) {\n          prevItem = allItems[firstVisibleIndex - 1];\n        }\n\n        const lastVisibleIndex = allItems.indexOf(visible[visible.length - 1]);\n        if (lastVisibleIndex < allItems.length - 1) {\n          nextItem = allItems[lastVisibleIndex + 1];\n        }\n      }\n\n      this.nextItem.set(nextItem);\n      this.prevItem.set(prevItem);\n    });\n  }\n\n  /**\n   * Scrolls the container in the specified direction.\n   *\n   * @param {'next' | 'prev'} direction - The direction to scroll ('next' for right, 'prev' for left).\n   */\n  public scroll(direction: 1 | -1 = 1): void {\n    const container = this.scrollContainer?.nativeElement;\n    if (!container) return;\n\n    container.scrollBy({ left: (container.clientWidth * direction) / 2 });\n  }\n\n  /**\n   * Scrolls the active item into view within the scroll container.\n   *\n   * @param {HTMLElement | null} activeItem - The currently active item to scroll into view.\n   */\n  public scrollActiveItemIntoView(activeItem: HTMLElement | null): void {\n    if (activeItem) {\n      activeItem.scrollIntoView({ behavior: 'smooth', block: 'nearest', inline: 'center' });\n    }\n  }\n\n  private observeVisibility(items: HTMLElement[], callback: (visible: HTMLElement[]) => void): IntersectionObserver {\n    const visibleMap = new Map<HTMLElement, boolean>();\n    const observer = new IntersectionObserver(\n      (entries) => {\n        for (const entry of entries) {\n          const isVisible = entry.isIntersecting && entry.intersectionRatio >= 0.75;\n          visibleMap.set(entry.target as HTMLElement, isVisible);\n        }\n        const visible = items.filter((item) => visibleMap.get(item));\n        this.visibleItems.set(visible);\n        callback(visible);\n      },\n      {\n        root: this.scrollContainer!.nativeElement,\n        threshold: 1,\n      },\n    );\n\n    items.forEach((item) => observer.observe(item));\n    return observer;\n  }\n}\n"]}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Generated bundle index. Do not edit.
|
|
3
|
+
*/
|
|
4
|
+
export * from './index';
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoib2R4LWFuZ3VsYXItY2RrLXNjcm9sbGFibGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9saWJzL2FuZ3VsYXIvY2RrL3Njcm9sbGFibGUvc3JjL29keC1hbmd1bGFyLWNkay1zY3JvbGxhYmxlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxTQUFTLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEdlbmVyYXRlZCBidW5kbGUgaW5kZXguIERvIG5vdCBlZGl0LlxuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vaW5kZXgnO1xuIl19
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { __decorate } from "tslib";
|
|
2
|
-
import { ChangeDetectionStrategy, Component, ContentChild, ContentChildren, forwardRef, HostListener, inject, QueryList, ViewEncapsulation, } from '@angular/core';
|
|
2
|
+
import { ChangeDetectionStrategy, Component, ContentChild, ContentChildren, forwardRef, HostListener, inject, input, QueryList, ViewEncapsulation, } from '@angular/core';
|
|
3
3
|
import { ClickOutsideDirective } from '@odx/angular';
|
|
4
4
|
import { AutocompleteControl, ODX_SEARCH_FILTER_HOST } from '@odx/angular/cdk/autocomplete-control';
|
|
5
5
|
import { DropdownDirective } from '@odx/angular/components/dropdown';
|
|
@@ -27,6 +27,11 @@ let AutocompleteComponent = class AutocompleteComponent extends AutocompleteCont
|
|
|
27
27
|
this.defaultActiveOptionIndex = 0;
|
|
28
28
|
this.patchValueFlag = false;
|
|
29
29
|
this.clickOutsideDirective = inject(ClickOutsideDirective, { host: true });
|
|
30
|
+
/** Text to display when no options are found.
|
|
31
|
+
*
|
|
32
|
+
* @type {string | null}
|
|
33
|
+
*/
|
|
34
|
+
this.notFoundText = input(null);
|
|
30
35
|
}
|
|
31
36
|
ngAfterViewInit() {
|
|
32
37
|
if (!this.options)
|
|
@@ -83,11 +88,12 @@ let AutocompleteComponent = class AutocompleteComponent extends AutocompleteCont
|
|
|
83
88
|
this.patchValueFlag = false;
|
|
84
89
|
return;
|
|
85
90
|
}
|
|
86
|
-
if (this.isOpen &&
|
|
87
|
-
this.closeDropdown();
|
|
88
|
-
}
|
|
89
|
-
else if (!this.isOpen && this.hasOptions) {
|
|
91
|
+
if (!this.isOpen && (this.hasOptions || !!this.notFoundText())) {
|
|
90
92
|
this.openDropdown();
|
|
93
|
+
return;
|
|
94
|
+
}
|
|
95
|
+
if (this.isOpen && !this.hasOptions && !this.notFoundText()) {
|
|
96
|
+
this.closeDropdown();
|
|
91
97
|
}
|
|
92
98
|
}
|
|
93
99
|
handleClickEvent() {
|
|
@@ -102,7 +108,7 @@ let AutocompleteComponent = class AutocompleteComponent extends AutocompleteCont
|
|
|
102
108
|
this.openDropdown();
|
|
103
109
|
}
|
|
104
110
|
}
|
|
105
|
-
handleKeyboardEvent(event) {
|
|
111
|
+
async handleKeyboardEvent(event) {
|
|
106
112
|
if (this.isReadonly || this.isDisabled)
|
|
107
113
|
return;
|
|
108
114
|
if (event.key === 'Escape') {
|
|
@@ -125,11 +131,6 @@ let AutocompleteComponent = class AutocompleteComponent extends AutocompleteCont
|
|
|
125
131
|
if (!this.isOpen && event.key === 'Tab') {
|
|
126
132
|
return;
|
|
127
133
|
}
|
|
128
|
-
deferFn(() => {
|
|
129
|
-
if (!this.isOpen && this.hasOptions) {
|
|
130
|
-
this.openDropdown();
|
|
131
|
-
}
|
|
132
|
-
});
|
|
133
134
|
this.keyManager?.onKeydown(event);
|
|
134
135
|
}
|
|
135
136
|
activateSelectedOption() {
|
|
@@ -149,7 +150,7 @@ let AutocompleteComponent = class AutocompleteComponent extends AutocompleteCont
|
|
|
149
150
|
this.searchField?.blur();
|
|
150
151
|
}
|
|
151
152
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AutocompleteComponent, deps: null, target: i0.ɵɵFactoryTarget.Component }); }
|
|
152
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: AutocompleteComponent, isStandalone: true, selector: "odx-autocomplete", host: { listeners: { "click": "handleClickEvent()", "keydown": "handleKeyboardEvent($event)" } }, providers: [
|
|
153
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: AutocompleteComponent, isStandalone: true, selector: "odx-autocomplete", inputs: { notFoundText: { classPropertyName: "notFoundText", publicName: "notFoundText", isSignal: true, isRequired: false, transformFunction: null } }, host: { listeners: { "click": "handleClickEvent()", "keydown": "handleKeyboardEvent($event)" } }, providers: [
|
|
153
154
|
{
|
|
154
155
|
provide: AUTOCOMPLETE_CONTROL,
|
|
155
156
|
useExisting: forwardRef(() => AutocompleteComponent),
|
|
@@ -158,7 +159,7 @@ let AutocompleteComponent = class AutocompleteComponent extends AutocompleteCont
|
|
|
158
159
|
provide: ODX_SEARCH_FILTER_HOST,
|
|
159
160
|
useExisting: AUTOCOMPLETE_CONTROL,
|
|
160
161
|
},
|
|
161
|
-
], queries: [{ propertyName: "searchField", first: true, predicate: AutocompleteInputControlDirective, descendants: true }, { propertyName: "options", predicate: AutocompleteOptionComponent, descendants: true }], usesInheritance: true, hostDirectives: [{ directive: i1.ClickOutsideDirective }], ngImport: i0, template: "<div\n aria-haspopup=\"listbox\"\n class=\"odx-autocomplete__trigger\"\n [odxDropdown]=\"dropdownContent\"\n [odxDropdownDisabled]=\"isDisabled || isReadonly\"\n [odxDropdownOptions]=\"{ matchReferenceWidth: true, offset: 4, outerPadding: 10, position: 'bottom-start' }\"\n [odxDropdownReferenceElement]=\"dropdownReferenceElement\"\n [odxDropdownShowLoader]=\"isLoading\"\n [odxDropdownOpenTrigger]=\"[]\"\n [odxDropdownClickOutsideActive]=\"false\"\n (odxDropdownBeforeOpen)=\"onDropdownOpen()\"\n (odxDropdownAfterOpen)=\"onDropdownOpened()\"\n (odxDropdownBeforeClose)=\"onDropdownClose()\"\n (odxDropdownAfterClose)=\"onDropdownClosed()\"\n>\n <ng-content select=\"input[odxAutocompleteControl]\" />\n</div>\n<ng-template #dropdownContent>\n <div class=\"odx-dropdown__option-list\" role=\"listbox\">\n @if (hasOptions) {\n <ng-content />\n }\n </div>\n</ng-template>\n", dependencies: [{ kind: "directive", type: DropdownDirective, selector: "[odxDropdown]", inputs: ["odxDropdown", "odxDropdownDisabled", "odxDropdownShowLoader", "odxDropdownClickOutsideActive", "odxDropdownOptions", "odxDropdownReferenceElement", "odxDropdownTriggerElement", "odxDropdownHost", "odxDropdownOpenTrigger", "odxDropdownCloseTrigger"], outputs: ["odxDropdownBeforeOpen", "odxDropdownAfterOpen", "odxDropdownBeforeClose", "odxDropdownAfterClose"], exportAs: ["odxDropdown"] }, { kind: "ngmodule", type: LoadingSpinnerModule }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
|
|
162
|
+
], queries: [{ propertyName: "searchField", first: true, predicate: AutocompleteInputControlDirective, descendants: true }, { propertyName: "options", predicate: AutocompleteOptionComponent, descendants: true }], usesInheritance: true, hostDirectives: [{ directive: i1.ClickOutsideDirective }], ngImport: i0, template: "<div\n aria-haspopup=\"listbox\"\n class=\"odx-autocomplete__trigger\"\n [odxDropdown]=\"dropdownContent\"\n [odxDropdownDisabled]=\"isDisabled || isReadonly\"\n [odxDropdownOptions]=\"{ matchReferenceWidth: true, offset: 4, outerPadding: 10, position: 'bottom-start' }\"\n [odxDropdownReferenceElement]=\"dropdownReferenceElement\"\n [odxDropdownShowLoader]=\"isLoading\"\n [odxDropdownOpenTrigger]=\"[]\"\n [odxDropdownClickOutsideActive]=\"false\"\n (odxDropdownBeforeOpen)=\"onDropdownOpen()\"\n (odxDropdownAfterOpen)=\"onDropdownOpened()\"\n (odxDropdownBeforeClose)=\"onDropdownClose()\"\n (odxDropdownAfterClose)=\"onDropdownClosed()\"\n>\n <ng-content select=\"input[odxAutocompleteControl]\" />\n</div>\n<ng-template #dropdownContent>\n <div class=\"odx-dropdown__option-list\" role=\"listbox\">\n @if (hasOptions) {\n <ng-content />\n } @else if (!!notFoundText()) {\n <odx-autocomplete-option class=\"is-disabled\">{{ notFoundText() }}</odx-autocomplete-option>\n }\n </div>\n</ng-template>\n", dependencies: [{ kind: "directive", type: DropdownDirective, selector: "[odxDropdown]", inputs: ["odxDropdown", "odxDropdownDisabled", "odxDropdownShowLoader", "odxDropdownClickOutsideActive", "odxDropdownOptions", "odxDropdownReferenceElement", "odxDropdownTriggerElement", "odxDropdownHost", "odxDropdownOpenTrigger", "odxDropdownCloseTrigger"], outputs: ["odxDropdownBeforeOpen", "odxDropdownAfterOpen", "odxDropdownBeforeClose", "odxDropdownAfterClose"], exportAs: ["odxDropdown"] }, { kind: "ngmodule", type: LoadingSpinnerModule }, { kind: "component", type: AutocompleteOptionComponent, selector: "odx-autocomplete-option" }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
|
|
162
163
|
};
|
|
163
164
|
AutocompleteComponent = __decorate([
|
|
164
165
|
CSSComponent('autocomplete')
|
|
@@ -166,7 +167,7 @@ AutocompleteComponent = __decorate([
|
|
|
166
167
|
export { AutocompleteComponent };
|
|
167
168
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AutocompleteComponent, decorators: [{
|
|
168
169
|
type: Component,
|
|
169
|
-
args: [{ standalone: true, selector: 'odx-autocomplete', imports: [DropdownDirective, LoadingSpinnerModule], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, providers: [
|
|
170
|
+
args: [{ standalone: true, selector: 'odx-autocomplete', imports: [DropdownDirective, LoadingSpinnerModule, AutocompleteOptionComponent], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, providers: [
|
|
170
171
|
{
|
|
171
172
|
provide: AUTOCOMPLETE_CONTROL,
|
|
172
173
|
useExisting: forwardRef(() => AutocompleteComponent),
|
|
@@ -175,7 +176,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
175
176
|
provide: ODX_SEARCH_FILTER_HOST,
|
|
176
177
|
useExisting: AUTOCOMPLETE_CONTROL,
|
|
177
178
|
},
|
|
178
|
-
], hostDirectives: [ClickOutsideDirective], template: "<div\n aria-haspopup=\"listbox\"\n class=\"odx-autocomplete__trigger\"\n [odxDropdown]=\"dropdownContent\"\n [odxDropdownDisabled]=\"isDisabled || isReadonly\"\n [odxDropdownOptions]=\"{ matchReferenceWidth: true, offset: 4, outerPadding: 10, position: 'bottom-start' }\"\n [odxDropdownReferenceElement]=\"dropdownReferenceElement\"\n [odxDropdownShowLoader]=\"isLoading\"\n [odxDropdownOpenTrigger]=\"[]\"\n [odxDropdownClickOutsideActive]=\"false\"\n (odxDropdownBeforeOpen)=\"onDropdownOpen()\"\n (odxDropdownAfterOpen)=\"onDropdownOpened()\"\n (odxDropdownBeforeClose)=\"onDropdownClose()\"\n (odxDropdownAfterClose)=\"onDropdownClosed()\"\n>\n <ng-content select=\"input[odxAutocompleteControl]\" />\n</div>\n<ng-template #dropdownContent>\n <div class=\"odx-dropdown__option-list\" role=\"listbox\">\n @if (hasOptions) {\n <ng-content />\n }\n </div>\n</ng-template>\n" }]
|
|
179
|
+
], hostDirectives: [ClickOutsideDirective], template: "<div\n aria-haspopup=\"listbox\"\n class=\"odx-autocomplete__trigger\"\n [odxDropdown]=\"dropdownContent\"\n [odxDropdownDisabled]=\"isDisabled || isReadonly\"\n [odxDropdownOptions]=\"{ matchReferenceWidth: true, offset: 4, outerPadding: 10, position: 'bottom-start' }\"\n [odxDropdownReferenceElement]=\"dropdownReferenceElement\"\n [odxDropdownShowLoader]=\"isLoading\"\n [odxDropdownOpenTrigger]=\"[]\"\n [odxDropdownClickOutsideActive]=\"false\"\n (odxDropdownBeforeOpen)=\"onDropdownOpen()\"\n (odxDropdownAfterOpen)=\"onDropdownOpened()\"\n (odxDropdownBeforeClose)=\"onDropdownClose()\"\n (odxDropdownAfterClose)=\"onDropdownClosed()\"\n>\n <ng-content select=\"input[odxAutocompleteControl]\" />\n</div>\n<ng-template #dropdownContent>\n <div class=\"odx-dropdown__option-list\" role=\"listbox\">\n @if (hasOptions) {\n <ng-content />\n } @else if (!!notFoundText()) {\n <odx-autocomplete-option class=\"is-disabled\">{{ notFoundText() }}</odx-autocomplete-option>\n }\n </div>\n</ng-template>\n" }]
|
|
179
180
|
}], propDecorators: { options: [{
|
|
180
181
|
type: ContentChildren,
|
|
181
182
|
args: [AutocompleteOptionComponent, { descendants: true, emitDistinctChangesOnly: true }]
|
|
@@ -189,4 +190,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
189
190
|
type: HostListener,
|
|
190
191
|
args: ['keydown', ['$event']]
|
|
191
192
|
}] } });
|
|
192
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"autocomplete.component.js","sourceRoot":"","sources":["../../../../../../../libs/angular/components/autocomplete/src/lib/autocomplete.component.ts","../../../../../../../libs/angular/components/autocomplete/src/lib/autocomplete.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  Component,\n  ContentChild,\n  ContentChildren,\n  forwardRef,\n  HostListener,\n  inject,\n  QueryList,\n  ViewEncapsulation,\n} from '@angular/core';\nimport { ClickOutsideDirective } from '@odx/angular';\nimport { AutocompleteControl, ODX_SEARCH_FILTER_HOST } from '@odx/angular/cdk/autocomplete-control';\nimport { DropdownDirective } from '@odx/angular/components/dropdown';\nimport { LoadingSpinnerModule } from '@odx/angular/components/loading-spinner';\nimport { CSSComponent } from '@odx/angular/internal';\nimport { deferFn } from '@odx/angular/utils';\nimport { filter, tap } from 'rxjs';\nimport { AUTOCOMPLETE_CONTROL } from './autocomplete.tokens';\nimport { AutocompleteOptionComponent } from './components';\nimport { AutocompleteInputControlDirective } from './directives';\n\n/**\n * Represents an autocomplete component that provides user interface for a dropdown list to select from as users type.\n * It extends `AutocompleteControl`, integrating custom logic for handling keyboard and mouse events,\n * managing the dropdown state, and updating the associated search field.\n *\n * @see {AutocompleteControl}\n *\n * @template T - The type of the value handled by the autocomplete.\n */\n@CSSComponent('autocomplete')\n@Component({\n  standalone: true,\n  selector: 'odx-autocomplete',\n  imports: [DropdownDirective, LoadingSpinnerModule],\n  templateUrl: './autocomplete.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  encapsulation: ViewEncapsulation.None,\n  providers: [\n    {\n      provide: AUTOCOMPLETE_CONTROL,\n      useExisting: forwardRef(() => AutocompleteComponent),\n    },\n    {\n      provide: ODX_SEARCH_FILTER_HOST,\n      useExisting: AUTOCOMPLETE_CONTROL,\n    },\n  ],\n  hostDirectives: [ClickOutsideDirective],\n})\nexport class AutocompleteComponent<T = unknown> extends AutocompleteControl<T | null> implements AfterViewInit {\n  private readonly defaultActiveOptionIndex = 0;\n  private patchValueFlag = false;\n  protected readonly clickOutsideDirective = inject(ClickOutsideDirective, { host: true });\n\n  /**\n   * The list of autocomplete options.\n   *\n   * @type {QueryList<AutocompleteOptionComponent<T>> | undefined}\n   */\n  @ContentChildren(AutocompleteOptionComponent, { descendants: true, emitDistinctChangesOnly: true })\n  public options?: QueryList<AutocompleteOptionComponent<T>>;\n\n  /**\n   * The search field input control.\n   *\n   * @type {AutocompleteInputControlDirective | undefined}\n   */\n  @ContentChild(AutocompleteInputControlDirective)\n  public searchField?: AutocompleteInputControlDirective;\n\n  public override ngAfterViewInit(): void {\n    if (!this.options) return;\n    if (this.value) this.updateSearchField(this.value);\n    this.initKeyManager(this.options.toArray());\n    this.handleQueryListOption(this.options);\n    this.handleSearchFieldChanges();\n    this.handleClickOutside();\n  }\n\n  /**\n   * Selects an option, updates the value, updates the search field display,\n   * emits the selected value, and closes the dropdown.\n   *\n   * @param {AutocompleteOptionComponent<T> | null} option - The option component to select. If the option is undefined or its value is undefined,\n   * no action is taken.\n   */\n  public selectOption(option?: AutocompleteOptionComponent<T> | null): void {\n    if (!option?.value) return;\n\n    this.updateValue(option.value);\n    this.updateSearchField(option.value);\n\n    this.optionSelected.emit(option.value);\n\n    this.closeDropdown();\n  }\n\n  /**\n   * Resets the search field to its initial state and updates the value of the autocomplete\n   * to an empty string or initial value.\n   */\n  public resetSearchField(): void {\n    this.updateValue('' as T);\n    this.searchField?.reset();\n  }\n\n  protected handleQueryListOption(options: QueryList<AutocompleteOptionComponent<T>>): void {\n    options.changes\n      .pipe(\n        tap(() => deferFn(() => this.updateDropdownState())),\n        filter(() => this.isOpen),\n        this.takeUntilDestroyed(),\n      )\n      .subscribe(() => {\n        deferFn(() => this.activateSelectedOption());\n      });\n  }\n\n  protected handleSearchFieldChanges(): void {\n    this.searchField?.valueChange$.pipe(this.takeUntilDestroyed()).subscribe(() => this.triggerControllerChange());\n  }\n\n  protected handleClickOutside(): void {\n    this.clickOutsideDirective.odxClickOutside.pipe(this.takeUntilDestroyed()).subscribe(() => this.clickOutside());\n  }\n\n  protected clickOutside(): void {\n    this.closeDropdown();\n    this.blurSelectSearchField();\n  }\n\n  protected updateDropdownState(): void {\n    if (this.patchValueFlag) {\n      this.patchValueFlag = false;\n      return;\n    }\n\n    if (this.isOpen && !this.hasOptions) {\n      this.closeDropdown();\n    } else if (!this.isOpen && this.hasOptions) {\n      this.openDropdown();\n    }\n  }\n\n  @HostListener('click')\n  protected handleClickEvent() {\n    if (this.isLoading && this.isOpen) {\n      this.closeDropdown();\n      this.blurSelectSearchField();\n      return;\n    }\n\n    if (this.isReadonly || this.isDisabled) return;\n\n    if (!this.isOpen && this.hasOptions) {\n      this.openDropdown();\n    }\n  }\n\n  @HostListener('keydown', ['$event'])\n  protected handleKeyboardEvent(event: KeyboardEvent) {\n    if (this.isReadonly || this.isDisabled) return;\n\n    if (event.key === 'Escape') {\n      this.resetSearchField();\n      this.blurSelectSearchField();\n      return;\n    }\n\n    if (this.isOpen && this.hasOptions) {\n      if (event.key === 'Enter' || event.key === 'Tab') {\n        event.preventDefault();\n        event.stopImmediatePropagation();\n        this.selectOption(this.keyManager?.activeItem as AutocompleteOptionComponent<T> | undefined);\n        return;\n      }\n    }\n\n    if (event.key === 'Enter') {\n      this.optionSelected.emit((this.value ?? '') as T);\n      return;\n    }\n\n    if (!this.isOpen && event.key === 'Tab') {\n      return;\n    }\n\n    deferFn(() => {\n      if (!this.isOpen && this.hasOptions) {\n        this.openDropdown();\n      }\n    });\n\n    this.keyManager?.onKeydown(event);\n  }\n\n  protected activateSelectedOption(): void {\n    this.keyManager?.setActiveItem(this.defaultActiveOptionIndex);\n  }\n\n  protected override onDropdownClosed(): void {\n    super.onDropdownClosed();\n    !this.searchField?.nativeElementValue && this.blurSelectSearchField();\n  }\n\n  private updateSearchField(value: T): void {\n    if (!this.searchField) return;\n    this.patchValueFlag = true;\n    this.searchField.nativeElementValue = this.stringify?.(value) ?? (value as string);\n  }\n\n  private blurSelectSearchField(): void {\n    this.searchField?.blur();\n  }\n}\n","<div\n  aria-haspopup=\"listbox\"\n  class=\"odx-autocomplete__trigger\"\n  [odxDropdown]=\"dropdownContent\"\n  [odxDropdownDisabled]=\"isDisabled || isReadonly\"\n  [odxDropdownOptions]=\"{ matchReferenceWidth: true, offset: 4, outerPadding: 10, position: 'bottom-start' }\"\n  [odxDropdownReferenceElement]=\"dropdownReferenceElement\"\n  [odxDropdownShowLoader]=\"isLoading\"\n  [odxDropdownOpenTrigger]=\"[]\"\n  [odxDropdownClickOutsideActive]=\"false\"\n  (odxDropdownBeforeOpen)=\"onDropdownOpen()\"\n  (odxDropdownAfterOpen)=\"onDropdownOpened()\"\n  (odxDropdownBeforeClose)=\"onDropdownClose()\"\n  (odxDropdownAfterClose)=\"onDropdownClosed()\"\n>\n  <ng-content select=\"input[odxAutocompleteControl]\" />\n</div>\n<ng-template #dropdownContent>\n  <div class=\"odx-dropdown__option-list\" role=\"listbox\">\n    @if (hasOptions) {\n      <ng-content />\n    }\n  </div>\n</ng-template>\n"]}
|
|
193
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"autocomplete.component.js","sourceRoot":"","sources":["../../../../../../../libs/angular/components/autocomplete/src/lib/autocomplete.component.ts","../../../../../../../libs/angular/components/autocomplete/src/lib/autocomplete.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  Component,\n  ContentChild,\n  ContentChildren,\n  forwardRef,\n  HostListener,\n  inject,\n  input,\n  QueryList,\n  ViewEncapsulation,\n} from '@angular/core';\nimport { ClickOutsideDirective } from '@odx/angular';\nimport { AutocompleteControl, ODX_SEARCH_FILTER_HOST } from '@odx/angular/cdk/autocomplete-control';\nimport { DropdownDirective } from '@odx/angular/components/dropdown';\nimport { LoadingSpinnerModule } from '@odx/angular/components/loading-spinner';\nimport { CSSComponent } from '@odx/angular/internal';\nimport { deferFn } from '@odx/angular/utils';\nimport { filter, tap } from 'rxjs';\nimport { AUTOCOMPLETE_CONTROL } from './autocomplete.tokens';\nimport { AutocompleteOptionComponent } from './components';\nimport { AutocompleteInputControlDirective } from './directives';\n\n/**\n * Represents an autocomplete component that provides user interface for a dropdown list to select from as users type.\n * It extends `AutocompleteControl`, integrating custom logic for handling keyboard and mouse events,\n * managing the dropdown state, and updating the associated search field.\n *\n * @see {AutocompleteControl}\n *\n * @template T - The type of the value handled by the autocomplete.\n */\n@CSSComponent('autocomplete')\n@Component({\n  standalone: true,\n  selector: 'odx-autocomplete',\n  imports: [DropdownDirective, LoadingSpinnerModule, AutocompleteOptionComponent],\n  templateUrl: './autocomplete.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  encapsulation: ViewEncapsulation.None,\n  providers: [\n    {\n      provide: AUTOCOMPLETE_CONTROL,\n      useExisting: forwardRef(() => AutocompleteComponent),\n    },\n    {\n      provide: ODX_SEARCH_FILTER_HOST,\n      useExisting: AUTOCOMPLETE_CONTROL,\n    },\n  ],\n  hostDirectives: [ClickOutsideDirective],\n})\nexport class AutocompleteComponent<T = unknown> extends AutocompleteControl<T | null> implements AfterViewInit {\n  private readonly defaultActiveOptionIndex = 0;\n  private patchValueFlag = false;\n  protected readonly clickOutsideDirective = inject(ClickOutsideDirective, { host: true });\n\n  /**\n   * The list of autocomplete options.\n   *\n   * @type {QueryList<AutocompleteOptionComponent<T>> | undefined}\n   */\n  @ContentChildren(AutocompleteOptionComponent, { descendants: true, emitDistinctChangesOnly: true })\n  public options?: QueryList<AutocompleteOptionComponent<T>>;\n\n  /** Text to display when no options are found.\n   *\n   * @type {string | null}\n   */\n  public notFoundText = input<string | null>(null);\n\n  /**\n   * The search field input control.\n   *\n   * @type {AutocompleteInputControlDirective | undefined}\n   */\n  @ContentChild(AutocompleteInputControlDirective)\n  public searchField?: AutocompleteInputControlDirective;\n\n  public override ngAfterViewInit(): void {\n    if (!this.options) return;\n    if (this.value) this.updateSearchField(this.value);\n    this.initKeyManager(this.options.toArray());\n    this.handleQueryListOption(this.options);\n    this.handleSearchFieldChanges();\n    this.handleClickOutside();\n  }\n\n  /**\n   * Selects an option, updates the value, updates the search field display,\n   * emits the selected value, and closes the dropdown.\n   *\n   * @param {AutocompleteOptionComponent<T> | null} option - The option component to select. If the option is undefined or its value is undefined,\n   * no action is taken.\n   */\n  public selectOption(option?: AutocompleteOptionComponent<T> | null): void {\n    if (!option?.value) return;\n\n    this.updateValue(option.value);\n    this.updateSearchField(option.value);\n\n    this.optionSelected.emit(option.value);\n\n    this.closeDropdown();\n  }\n\n  /**\n   * Resets the search field to its initial state and updates the value of the autocomplete\n   * to an empty string or initial value.\n   */\n  public resetSearchField(): void {\n    this.updateValue('' as T);\n    this.searchField?.reset();\n  }\n\n  protected handleQueryListOption(options: QueryList<AutocompleteOptionComponent<T>>): void {\n    options.changes\n      .pipe(\n        tap(() => deferFn(() => this.updateDropdownState())),\n        filter(() => this.isOpen),\n        this.takeUntilDestroyed(),\n      )\n      .subscribe(() => {\n        deferFn(() => this.activateSelectedOption());\n      });\n  }\n\n  protected handleSearchFieldChanges(): void {\n    this.searchField?.valueChange$.pipe(this.takeUntilDestroyed()).subscribe(() => this.triggerControllerChange());\n  }\n\n  protected handleClickOutside(): void {\n    this.clickOutsideDirective.odxClickOutside.pipe(this.takeUntilDestroyed()).subscribe(() => this.clickOutside());\n  }\n\n  protected clickOutside(): void {\n    this.closeDropdown();\n    this.blurSelectSearchField();\n  }\n\n  protected updateDropdownState(): void {\n    if (this.patchValueFlag) {\n      this.patchValueFlag = false;\n      return;\n    }\n    if (!this.isOpen && (this.hasOptions || !!this.notFoundText())) {\n      this.openDropdown();\n      return;\n    }\n    if (this.isOpen && !this.hasOptions && !this.notFoundText()) {\n      this.closeDropdown();\n    }\n  }\n\n  @HostListener('click')\n  protected handleClickEvent() {\n    if (this.isLoading && this.isOpen) {\n      this.closeDropdown();\n      this.blurSelectSearchField();\n      return;\n    }\n\n    if (this.isReadonly || this.isDisabled) return;\n\n    if (!this.isOpen && this.hasOptions) {\n      this.openDropdown();\n    }\n  }\n\n  @HostListener('keydown', ['$event'])\n  protected async handleKeyboardEvent(event: KeyboardEvent) {\n    if (this.isReadonly || this.isDisabled) return;\n\n    if (event.key === 'Escape') {\n      this.resetSearchField();\n      this.blurSelectSearchField();\n      return;\n    }\n\n    if (this.isOpen && this.hasOptions) {\n      if (event.key === 'Enter' || event.key === 'Tab') {\n        event.preventDefault();\n        event.stopImmediatePropagation();\n        this.selectOption(this.keyManager?.activeItem as AutocompleteOptionComponent<T> | undefined);\n        return;\n      }\n    }\n\n    if (event.key === 'Enter') {\n      this.optionSelected.emit((this.value ?? '') as T);\n      return;\n    }\n\n    if (!this.isOpen && event.key === 'Tab') {\n      return;\n    }\n\n    this.keyManager?.onKeydown(event);\n  }\n\n  protected activateSelectedOption(): void {\n    this.keyManager?.setActiveItem(this.defaultActiveOptionIndex);\n  }\n\n  protected override onDropdownClosed(): void {\n    super.onDropdownClosed();\n    !this.searchField?.nativeElementValue && this.blurSelectSearchField();\n  }\n\n  private updateSearchField(value: T): void {\n    if (!this.searchField) return;\n    this.patchValueFlag = true;\n    this.searchField.nativeElementValue = this.stringify?.(value) ?? (value as string);\n  }\n\n  private blurSelectSearchField(): void {\n    this.searchField?.blur();\n  }\n}\n","<div\n  aria-haspopup=\"listbox\"\n  class=\"odx-autocomplete__trigger\"\n  [odxDropdown]=\"dropdownContent\"\n  [odxDropdownDisabled]=\"isDisabled || isReadonly\"\n  [odxDropdownOptions]=\"{ matchReferenceWidth: true, offset: 4, outerPadding: 10, position: 'bottom-start' }\"\n  [odxDropdownReferenceElement]=\"dropdownReferenceElement\"\n  [odxDropdownShowLoader]=\"isLoading\"\n  [odxDropdownOpenTrigger]=\"[]\"\n  [odxDropdownClickOutsideActive]=\"false\"\n  (odxDropdownBeforeOpen)=\"onDropdownOpen()\"\n  (odxDropdownAfterOpen)=\"onDropdownOpened()\"\n  (odxDropdownBeforeClose)=\"onDropdownClose()\"\n  (odxDropdownAfterClose)=\"onDropdownClosed()\"\n>\n  <ng-content select=\"input[odxAutocompleteControl]\" />\n</div>\n<ng-template #dropdownContent>\n  <div class=\"odx-dropdown__option-list\" role=\"listbox\">\n    @if (hasOptions) {\n      <ng-content />\n    } @else if (!!notFoundText()) {\n      <odx-autocomplete-option class=\"is-disabled\">{{ notFoundText() }}</odx-autocomplete-option>\n    }\n  </div>\n</ng-template>\n"]}
|