ng-primitives 0.8.0 → 0.10.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/accordion/accordion-trigger/accordion-trigger.directive.d.ts +1 -2
- package/esm2022/accordion/accordion-trigger/accordion-trigger.directive.mjs +2 -5
- package/esm2022/interactions/focus-visible/focus-visible.directive.mjs +27 -20
- package/esm2022/popover/config/popover.config.mjs +39 -0
- package/esm2022/popover/index.mjs +13 -0
- package/esm2022/popover/ng-primitives-popover.mjs +5 -0
- package/esm2022/popover/popover/popover.directive.mjs +60 -0
- package/esm2022/popover/popover/popover.token.mjs +16 -0
- package/esm2022/popover/popover-trigger/popover-trigger.directive.mjs +222 -0
- package/esm2022/popover/popover-trigger/popover-trigger.token.mjs +23 -0
- package/esm2022/toast/config/toast.config.mjs +38 -0
- package/esm2022/toast/index.mjs +12 -0
- package/esm2022/toast/ng-primitives-toast.mjs +5 -0
- package/esm2022/toast/toast/toast-ref.mjs +96 -0
- package/esm2022/toast/toast/toast.directive.mjs +100 -0
- package/esm2022/toast/toast/toast.token.mjs +16 -0
- package/esm2022/utils/index.mjs +2 -2
- package/esm2022/utils/signals/async.mjs +18 -2
- package/fesm2022/ng-primitives-accordion.mjs +1 -4
- package/fesm2022/ng-primitives-accordion.mjs.map +1 -1
- package/fesm2022/ng-primitives-interactions.mjs +25 -19
- package/fesm2022/ng-primitives-interactions.mjs.map +1 -1
- package/fesm2022/ng-primitives-popover.mjs +366 -0
- package/fesm2022/ng-primitives-popover.mjs.map +1 -0
- package/fesm2022/ng-primitives-toast.mjs +260 -0
- package/fesm2022/ng-primitives-toast.mjs.map +1 -0
- package/fesm2022/ng-primitives-utils.mjs +18 -2
- package/fesm2022/ng-primitives-utils.mjs.map +1 -1
- package/generators.json +20 -0
- package/interactions/focus-visible/focus-visible.directive.d.ts +3 -2
- package/package.json +26 -14
- package/popover/README.md +3 -0
- package/popover/config/popover.config.d.ts +59 -0
- package/popover/index.d.ts +12 -0
- package/popover/popover/popover.directive.d.ts +31 -0
- package/popover/popover/popover.token.d.ts +14 -0
- package/popover/popover-trigger/popover-trigger.directive.d.ts +127 -0
- package/popover/popover-trigger/popover-trigger.token.d.ts +22 -0
- package/schematics/checkbox/compat.d.ts +2 -0
- package/schematics/checkbox/compat.js +13 -0
- package/schematics/checkbox/compat.js.map +1 -0
- package/schematics/checkbox/files/checkbox.component.__style__.template +1 -0
- package/schematics/checkbox/files/checkbox.component.html.template +1 -0
- package/schematics/checkbox/files/checkbox.component.ts.template +69 -0
- package/schematics/checkbox/generator.d.ts +11 -0
- package/schematics/checkbox/generator.js +48 -0
- package/schematics/checkbox/generator.js.map +1 -0
- package/schematics/checkbox/schema.d.ts +16 -0
- package/schematics/checkbox/schema.json +39 -0
- package/schematics/file-upload/compat.d.ts +2 -0
- package/schematics/file-upload/compat.js +13 -0
- package/schematics/file-upload/compat.js.map +1 -0
- package/schematics/file-upload/files/file-upload.component.__style__.template +1 -0
- package/schematics/file-upload/files/file-upload.component.html.template +1 -0
- package/schematics/file-upload/files/file-upload.component.ts.template +28 -0
- package/schematics/file-upload/generator.d.ts +11 -0
- package/schematics/file-upload/generator.js +37 -0
- package/schematics/file-upload/generator.js.map +1 -0
- package/schematics/file-upload/schema.d.ts +16 -0
- package/schematics/file-upload/schema.json +39 -0
- package/toast/README.md +3 -0
- package/toast/config/toast.config.d.ts +48 -0
- package/toast/index.d.ts +11 -0
- package/toast/toast/toast-ref.d.ts +56 -0
- package/toast/toast/toast.directive.d.ts +39 -0
- package/toast/toast/toast.token.d.ts +14 -0
- package/utils/index.d.ts +1 -1
- package/utils/signals/async.d.ts +10 -1
|
@@ -1,5 +1,4 @@
|
|
|
1
1
|
import * as i0 from "@angular/core";
|
|
2
|
-
import * as i1 from "ng-primitives/interactions";
|
|
3
2
|
export declare class NgpAccordionTrigger {
|
|
4
3
|
/**
|
|
5
4
|
* Access the parent accordion.
|
|
@@ -18,5 +17,5 @@ export declare class NgpAccordionTrigger {
|
|
|
18
17
|
*/
|
|
19
18
|
toggle(): void;
|
|
20
19
|
static ɵfac: i0.ɵɵFactoryDeclaration<NgpAccordionTrigger, never>;
|
|
21
|
-
static ɵdir: i0.ɵɵDirectiveDeclaration<NgpAccordionTrigger, "[ngpAccordionTrigger]", ["ngpAccordionTrigger"], { "id": { "alias": "id"; "required": false; "isSignal": true; }; }, {}, never, never, true,
|
|
20
|
+
static ɵdir: i0.ɵɵDirectiveDeclaration<NgpAccordionTrigger, "[ngpAccordionTrigger]", ["ngpAccordionTrigger"], { "id": { "alias": "id"; "required": false; "isSignal": true; }; }, {}, never, never, true, never>;
|
|
22
21
|
}
|
|
@@ -6,13 +6,11 @@
|
|
|
6
6
|
* LICENSE file in the root directory of this source tree.
|
|
7
7
|
*/
|
|
8
8
|
import { Directive, HostListener, input } from '@angular/core';
|
|
9
|
-
import { NgpFocusVisible, NgpHover, NgpPress } from 'ng-primitives/interactions';
|
|
10
9
|
import { uniqueId } from 'ng-primitives/utils';
|
|
11
10
|
import { injectAccordionItem } from '../accordion-item/accordion-item.token';
|
|
12
11
|
import { injectAccordion } from '../accordion/accordion.token';
|
|
13
12
|
import { NgpAccordionTriggerToken } from './accordion-trigger.token';
|
|
14
13
|
import * as i0 from "@angular/core";
|
|
15
|
-
import * as i1 from "ng-primitives/interactions";
|
|
16
14
|
export class NgpAccordionTrigger {
|
|
17
15
|
constructor() {
|
|
18
16
|
/**
|
|
@@ -38,7 +36,7 @@ export class NgpAccordionTrigger {
|
|
|
38
36
|
this.accordion.toggle(this.item.value());
|
|
39
37
|
}
|
|
40
38
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.1", ngImport: i0, type: NgpAccordionTrigger, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
41
|
-
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.1.1", type: NgpAccordionTrigger, isStandalone: true, selector: "[ngpAccordionTrigger]", inputs: { id: { classPropertyName: "id", publicName: "id", isSignal: true, isRequired: false, transformFunction: null } }, host: { listeners: { "click": "toggle()" }, properties: { "id": "id()", "attr.data-orientation": "accordion.orientation()", "attr.data-open": "item.open()", "attr.data-disabled": "item.disabled() || accordion.disabled()", "attr.aria-controls": "item.contentId()", "attr.aria-expanded": "item.open()" } }, providers: [{ provide: NgpAccordionTriggerToken, useExisting: NgpAccordionTrigger }], exportAs: ["ngpAccordionTrigger"],
|
|
39
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.1.1", type: NgpAccordionTrigger, isStandalone: true, selector: "[ngpAccordionTrigger]", inputs: { id: { classPropertyName: "id", publicName: "id", isSignal: true, isRequired: false, transformFunction: null } }, host: { listeners: { "click": "toggle()" }, properties: { "id": "id()", "attr.data-orientation": "accordion.orientation()", "attr.data-open": "item.open()", "attr.data-disabled": "item.disabled() || accordion.disabled()", "attr.aria-controls": "item.contentId()", "attr.aria-expanded": "item.open()" } }, providers: [{ provide: NgpAccordionTriggerToken, useExisting: NgpAccordionTrigger }], exportAs: ["ngpAccordionTrigger"], ngImport: i0 }); }
|
|
42
40
|
}
|
|
43
41
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.1", ngImport: i0, type: NgpAccordionTrigger, decorators: [{
|
|
44
42
|
type: Directive,
|
|
@@ -47,7 +45,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.1", ngImpor
|
|
|
47
45
|
selector: '[ngpAccordionTrigger]',
|
|
48
46
|
exportAs: 'ngpAccordionTrigger',
|
|
49
47
|
providers: [{ provide: NgpAccordionTriggerToken, useExisting: NgpAccordionTrigger }],
|
|
50
|
-
hostDirectives: [NgpHover, NgpPress, NgpFocusVisible],
|
|
51
48
|
host: {
|
|
52
49
|
'[id]': 'id()',
|
|
53
50
|
'[attr.data-orientation]': 'accordion.orientation()',
|
|
@@ -61,4 +58,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.1", ngImpor
|
|
|
61
58
|
type: HostListener,
|
|
62
59
|
args: ['click']
|
|
63
60
|
}] } });
|
|
64
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
61
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWNjb3JkaW9uLXRyaWdnZXIuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvbmctcHJpbWl0aXZlcy9hY2NvcmRpb24vc3JjL2FjY29yZGlvbi10cmlnZ2VyL2FjY29yZGlvbi10cmlnZ2VyLmRpcmVjdGl2ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7O0dBTUc7QUFDSCxPQUFPLEVBQUUsU0FBUyxFQUFFLFlBQVksRUFBRSxLQUFLLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDL0QsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxNQUFNLHdDQUF3QyxDQUFDO0FBQzdFLE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSw4QkFBOEIsQ0FBQztBQUMvRCxPQUFPLEVBQUUsd0JBQXdCLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQzs7QUFnQnJFLE1BQU0sT0FBTyxtQkFBbUI7SUFkaEM7UUFlRTs7V0FFRztRQUNnQixjQUFTLEdBQUcsZUFBZSxFQUFFLENBQUM7UUFFakQ7O1dBRUc7UUFDZ0IsU0FBSSxHQUFHLG1CQUFtQixFQUFFLENBQUM7UUFFaEQ7O1dBRUc7UUFDTSxPQUFFLEdBQUcsS0FBSyxDQUFTLFFBQVEsQ0FBQyx1QkFBdUIsQ0FBQyxDQUFDLENBQUM7S0FhaEU7SUFYQzs7T0FFRztJQUVILE1BQU07UUFDSixJQUFJLElBQUksQ0FBQyxJQUFJLENBQUMsUUFBUSxFQUFFLElBQUksSUFBSSxDQUFDLFNBQVMsQ0FBQyxRQUFRLEVBQUUsRUFBRSxDQUFDO1lBQ3RELE9BQU87UUFDVCxDQUFDO1FBRUQsSUFBSSxDQUFDLFNBQVMsQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxLQUFLLEVBQUUsQ0FBQyxDQUFDO0lBQzNDLENBQUM7OEdBMUJVLG1CQUFtQjtrR0FBbkIsbUJBQW1CLGdmQVZuQixDQUFDLEVBQUUsT0FBTyxFQUFFLHdCQUF3QixFQUFFLFdBQVcsRUFBRSxtQkFBbUIsRUFBRSxDQUFDOzsyRkFVekUsbUJBQW1CO2tCQWQvQixTQUFTO21CQUFDO29CQUNULFVBQVUsRUFBRSxJQUFJO29CQUNoQixRQUFRLEVBQUUsdUJBQXVCO29CQUNqQyxRQUFRLEVBQUUscUJBQXFCO29CQUMvQixTQUFTLEVBQUUsQ0FBQyxFQUFFLE9BQU8sRUFBRSx3QkFBd0IsRUFBRSxXQUFXLHFCQUFxQixFQUFFLENBQUM7b0JBQ3BGLElBQUksRUFBRTt3QkFDSixNQUFNLEVBQUUsTUFBTTt3QkFDZCx5QkFBeUIsRUFBRSx5QkFBeUI7d0JBQ3BELGtCQUFrQixFQUFFLGFBQWE7d0JBQ2pDLHNCQUFzQixFQUFFLHlDQUF5Qzt3QkFDakUsc0JBQXNCLEVBQUUsa0JBQWtCO3dCQUMxQyxzQkFBc0IsRUFBRSxhQUFhO3FCQUN0QztpQkFDRjs4QkFxQkMsTUFBTTtzQkFETCxZQUFZO3VCQUFDLE9BQU8iLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIENvcHlyaWdodCDCqSAyMDI0IEFuZ3VsYXIgUHJpbWl0aXZlcy5cbiAqIGh0dHBzOi8vZ2l0aHViLmNvbS9uZy1wcmltaXRpdmVzL25nLXByaW1pdGl2ZXNcbiAqXG4gKiBUaGlzIHNvdXJjZSBjb2RlIGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgMi4wIGxpY2Vuc2UgZm91bmQgaW4gdGhlXG4gKiBMSUNFTlNFIGZpbGUgaW4gdGhlIHJvb3QgZGlyZWN0b3J5IG9mIHRoaXMgc291cmNlIHRyZWUuXG4gKi9cbmltcG9ydCB7IERpcmVjdGl2ZSwgSG9zdExpc3RlbmVyLCBpbnB1dCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgdW5pcXVlSWQgfSBmcm9tICduZy1wcmltaXRpdmVzL3V0aWxzJztcbmltcG9ydCB7IGluamVjdEFjY29yZGlvbkl0ZW0gfSBmcm9tICcuLi9hY2NvcmRpb24taXRlbS9hY2NvcmRpb24taXRlbS50b2tlbic7XG5pbXBvcnQgeyBpbmplY3RBY2NvcmRpb24gfSBmcm9tICcuLi9hY2NvcmRpb24vYWNjb3JkaW9uLnRva2VuJztcbmltcG9ydCB7IE5ncEFjY29yZGlvblRyaWdnZXJUb2tlbiB9IGZyb20gJy4vYWNjb3JkaW9uLXRyaWdnZXIudG9rZW4nO1xuXG5ARGlyZWN0aXZlKHtcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgc2VsZWN0b3I6ICdbbmdwQWNjb3JkaW9uVHJpZ2dlcl0nLFxuICBleHBvcnRBczogJ25ncEFjY29yZGlvblRyaWdnZXInLFxuICBwcm92aWRlcnM6IFt7IHByb3ZpZGU6IE5ncEFjY29yZGlvblRyaWdnZXJUb2tlbiwgdXNlRXhpc3Rpbmc6IE5ncEFjY29yZGlvblRyaWdnZXIgfV0sXG4gIGhvc3Q6IHtcbiAgICAnW2lkXSc6ICdpZCgpJyxcbiAgICAnW2F0dHIuZGF0YS1vcmllbnRhdGlvbl0nOiAnYWNjb3JkaW9uLm9yaWVudGF0aW9uKCknLFxuICAgICdbYXR0ci5kYXRhLW9wZW5dJzogJ2l0ZW0ub3BlbigpJyxcbiAgICAnW2F0dHIuZGF0YS1kaXNhYmxlZF0nOiAnaXRlbS5kaXNhYmxlZCgpIHx8IGFjY29yZGlvbi5kaXNhYmxlZCgpJyxcbiAgICAnW2F0dHIuYXJpYS1jb250cm9sc10nOiAnaXRlbS5jb250ZW50SWQoKScsXG4gICAgJ1thdHRyLmFyaWEtZXhwYW5kZWRdJzogJ2l0ZW0ub3BlbigpJyxcbiAgfSxcbn0pXG5leHBvcnQgY2xhc3MgTmdwQWNjb3JkaW9uVHJpZ2dlciB7XG4gIC8qKlxuICAgKiBBY2Nlc3MgdGhlIHBhcmVudCBhY2NvcmRpb24uXG4gICAqL1xuICBwcm90ZWN0ZWQgcmVhZG9ubHkgYWNjb3JkaW9uID0gaW5qZWN0QWNjb3JkaW9uKCk7XG5cbiAgLyoqXG4gICAqIFRoZSBpdGVtIGluc3RhbmNlLlxuICAgKi9cbiAgcHJvdGVjdGVkIHJlYWRvbmx5IGl0ZW0gPSBpbmplY3RBY2NvcmRpb25JdGVtKCk7XG5cbiAgLyoqXG4gICAqIFRoZSBpZCBvZiB0aGUgdHJpZ2dlci5cbiAgICovXG4gIHJlYWRvbmx5IGlkID0gaW5wdXQ8c3RyaW5nPih1bmlxdWVJZCgnbmdwLWFjY29yZGlvbi10cmlnZ2VyJykpO1xuXG4gIC8qKlxuICAgKiBUb2dnbGUgdGhlIGFjY29yZGlvbiBpdGVtLlxuICAgKi9cbiAgQEhvc3RMaXN0ZW5lcignY2xpY2snKVxuICB0b2dnbGUoKTogdm9pZCB7XG4gICAgaWYgKHRoaXMuaXRlbS5kaXNhYmxlZCgpIHx8IHRoaXMuYWNjb3JkaW9uLmRpc2FibGVkKCkpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICB0aGlzLmFjY29yZGlvbi50b2dnbGUodGhpcy5pdGVtLnZhbHVlKCkpO1xuICB9XG59XG4iXX0=
|
|
@@ -6,9 +6,10 @@
|
|
|
6
6
|
* LICENSE file in the root directory of this source tree.
|
|
7
7
|
*/
|
|
8
8
|
import { FocusMonitor } from '@angular/cdk/a11y';
|
|
9
|
-
import { Directive, ElementRef,
|
|
9
|
+
import { Directive, ElementRef, booleanAttribute, inject, input, output, signal, } from '@angular/core';
|
|
10
10
|
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
|
11
11
|
import { injectDisabled } from 'ng-primitives/internal';
|
|
12
|
+
import { onBooleanChange } from 'ng-primitives/utils';
|
|
12
13
|
import { NgpFocusVisibleToken } from './focus-visible.token';
|
|
13
14
|
import * as i0 from "@angular/core";
|
|
14
15
|
export class NgpFocusVisible {
|
|
@@ -42,10 +43,15 @@ export class NgpFocusVisible {
|
|
|
42
43
|
* Whether the element is currently focused.
|
|
43
44
|
*/
|
|
44
45
|
this.isFocused = signal(false);
|
|
46
|
+
// handle focus state
|
|
45
47
|
this.focusMonitor
|
|
46
48
|
.monitor(this.elementRef.nativeElement)
|
|
47
49
|
.pipe(takeUntilDestroyed())
|
|
48
|
-
.subscribe(origin =>
|
|
50
|
+
.subscribe(origin =>
|
|
51
|
+
// null indicates the element was blurred
|
|
52
|
+
origin === null ? this.onBlur() : this.onFocus(origin));
|
|
53
|
+
// if the component becomes disabled and it is focused, hide the focus
|
|
54
|
+
onBooleanChange(this.isDisabled, () => this.focus(false));
|
|
49
55
|
}
|
|
50
56
|
onFocus(origin) {
|
|
51
57
|
if (this.isDisabled() || this.isFocused()) {
|
|
@@ -53,26 +59,30 @@ export class NgpFocusVisible {
|
|
|
53
59
|
}
|
|
54
60
|
// for some elements the focus visible state should always appear on focus
|
|
55
61
|
if (this.alwaysShowFocus()) {
|
|
56
|
-
this.
|
|
57
|
-
this.focusChange.emit(true);
|
|
62
|
+
this.focus(true);
|
|
58
63
|
return;
|
|
59
64
|
}
|
|
60
|
-
// if the focus origin is keyboard, then the focus is visible
|
|
61
|
-
if (origin === 'keyboard') {
|
|
62
|
-
this.
|
|
63
|
-
this.focusChange.emit(true);
|
|
65
|
+
// if the focus origin is keyboard or program(focused programmatically), then the focus is visible
|
|
66
|
+
if (origin === 'keyboard' || origin === 'program') {
|
|
67
|
+
this.focus(true);
|
|
64
68
|
return;
|
|
65
69
|
}
|
|
66
70
|
}
|
|
67
|
-
/**
|
|
68
|
-
* Listen for blur events.
|
|
69
|
-
*/
|
|
70
71
|
onBlur() {
|
|
71
72
|
if (this.isDisabled() || !this.isFocused()) {
|
|
72
73
|
return;
|
|
73
74
|
}
|
|
74
|
-
this.
|
|
75
|
-
|
|
75
|
+
this.focus(false);
|
|
76
|
+
}
|
|
77
|
+
/**
|
|
78
|
+
* Trigger the focus signal along with the focusChange event.
|
|
79
|
+
*/
|
|
80
|
+
focus(value) {
|
|
81
|
+
if (this.isFocused() === value) {
|
|
82
|
+
return;
|
|
83
|
+
}
|
|
84
|
+
this.isFocused.set(value);
|
|
85
|
+
this.focusChange.emit(value);
|
|
76
86
|
}
|
|
77
87
|
alwaysShowFocus() {
|
|
78
88
|
const nonTextInputTypes = [
|
|
@@ -103,7 +113,7 @@ export class NgpFocusVisible {
|
|
|
103
113
|
return false;
|
|
104
114
|
}
|
|
105
115
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.1", ngImport: i0, type: NgpFocusVisible, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
106
|
-
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.1.1", type: NgpFocusVisible, isStandalone: true, selector: "[ngpFocusVisible]", inputs: { disabled: { classPropertyName: "disabled", publicName: "ngpFocusVisibleDisabled", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { focusChange: "ngpFocusVisible" }, host: {
|
|
116
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.1.1", type: NgpFocusVisible, isStandalone: true, selector: "[ngpFocusVisible]", inputs: { disabled: { classPropertyName: "disabled", publicName: "ngpFocusVisibleDisabled", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { focusChange: "ngpFocusVisible" }, host: { properties: { "attr.data-focus-visible": "isFocused()" } }, providers: [{ provide: NgpFocusVisibleToken, useExisting: NgpFocusVisible }], exportAs: ["ngpFocusVisible"], ngImport: i0 }); }
|
|
107
117
|
}
|
|
108
118
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.1", ngImport: i0, type: NgpFocusVisible, decorators: [{
|
|
109
119
|
type: Directive,
|
|
@@ -113,11 +123,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.1", ngImpor
|
|
|
113
123
|
exportAs: 'ngpFocusVisible',
|
|
114
124
|
providers: [{ provide: NgpFocusVisibleToken, useExisting: NgpFocusVisible }],
|
|
115
125
|
host: {
|
|
116
|
-
'[attr.data-focus-visible]': 'isFocused()
|
|
126
|
+
'[attr.data-focus-visible]': 'isFocused()',
|
|
117
127
|
},
|
|
118
128
|
}]
|
|
119
|
-
}], ctorParameters: () => []
|
|
120
|
-
|
|
121
|
-
args: ['blur']
|
|
122
|
-
}] } });
|
|
123
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"focus-visible.directive.js","sourceRoot":"","sources":["../../../../../../packages/ng-primitives/interactions/src/focus-visible/focus-visible.directive.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Copyright © 2024 Angular Primitives.\n * https://github.com/ng-primitives/ng-primitives\n *\n * This source code is licensed under the Apache 2.0 license found in the\n * LICENSE file in the root directory of this source tree.\n */\nimport { FocusMonitor, FocusOrigin } from '@angular/cdk/a11y';\nimport { BooleanInput } from '@angular/cdk/coercion';\nimport {\n  Directive,\n  ElementRef,\n  HostListener,\n  booleanAttribute,\n  inject,\n  input,\n  output,\n  signal,\n} from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { injectDisabled } from 'ng-primitives/internal';\nimport { NgpFocusVisibleToken } from './focus-visible.token';\n\n@Directive({\n  standalone: true,\n  selector: '[ngpFocusVisible]',\n  exportAs: 'ngpFocusVisible',\n  providers: [{ provide: NgpFocusVisibleToken, useExisting: NgpFocusVisible }],\n  host: {\n    '[attr.data-focus-visible]': 'isFocused() && !isDisabled()',\n  },\n})\nexport class NgpFocusVisible {\n  /**\n   * Access the element that the directive is applied to.\n   */\n  private readonly elementRef = inject<ElementRef<HTMLElement>>(ElementRef);\n\n  /**\n   * Access the focus monitor.\n   */\n  private readonly focusMonitor = inject(FocusMonitor);\n\n  /**\n   * Whether focus events are listened to.\n   */\n  readonly disabled = input<boolean, BooleanInput>(false, {\n    alias: 'ngpFocusVisibleDisabled',\n    transform: booleanAttribute,\n  });\n\n  /**\n   * Access the disabled state from any parent.\n   */\n  private readonly isDisabled = injectDisabled(this.disabled);\n\n  /**\n   * Emit when the element is visually focused.\n   */\n  readonly focusChange = output<boolean>({\n    alias: 'ngpFocusVisible',\n  });\n\n  /**\n   * Whether the element is currently focused.\n   */\n  protected readonly isFocused = signal<boolean>(false);\n\n  constructor() {\n    this.focusMonitor\n      .monitor(this.elementRef.nativeElement)\n      .pipe(takeUntilDestroyed())\n      .subscribe(origin => this.onFocus(origin));\n  }\n\n  private onFocus(origin: FocusOrigin): void {\n    if (this.isDisabled() || this.isFocused()) {\n      return;\n    }\n\n    // for some elements the focus visible state should always appear on focus\n    if (this.alwaysShowFocus()) {\n      this.isFocused.set(true);\n      this.focusChange.emit(true);\n      return;\n    }\n\n    // if the focus origin is keyboard, then the focus is visible\n    if (origin === 'keyboard') {\n      this.isFocused.set(true);\n      this.focusChange.emit(true);\n      return;\n    }\n  }\n\n  /**\n   * Listen for blur events.\n   */\n  @HostListener('blur')\n  protected onBlur(): void {\n    if (this.isDisabled() || !this.isFocused()) {\n      return;\n    }\n\n    this.isFocused.set(false);\n    this.focusChange.emit(false);\n  }\n\n  private alwaysShowFocus(): boolean {\n    const nonTextInputTypes = [\n      'checkbox',\n      'radio',\n      'range',\n      'color',\n      'file',\n      'image',\n      'button',\n      'submit',\n      'reset',\n    ];\n\n    // if this is an input element and it is a text input\n    if (\n      this.elementRef.nativeElement instanceof HTMLInputElement &&\n      !nonTextInputTypes.includes(this.elementRef.nativeElement.type)\n    ) {\n      return true;\n    }\n\n    // if this is a textarea\n    if (this.elementRef.nativeElement instanceof HTMLTextAreaElement) {\n      return true;\n    }\n\n    // if this is an element with contenteditable\n    if (\n      this.elementRef.nativeElement.isContentEditable ||\n      this.elementRef.nativeElement.hasAttribute('contenteditable')\n    ) {\n      return true;\n    }\n\n    return false;\n  }\n}\n"]}
|
|
129
|
+
}], ctorParameters: () => [] });
|
|
130
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"focus-visible.directive.js","sourceRoot":"","sources":["../../../../../../packages/ng-primitives/interactions/src/focus-visible/focus-visible.directive.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Copyright © 2024 Angular Primitives.\n * https://github.com/ng-primitives/ng-primitives\n *\n * This source code is licensed under the Apache 2.0 license found in the\n * LICENSE file in the root directory of this source tree.\n */\nimport { FocusMonitor, FocusOrigin } from '@angular/cdk/a11y';\nimport { BooleanInput } from '@angular/cdk/coercion';\nimport {\n  Directive,\n  ElementRef,\n  booleanAttribute,\n  inject,\n  input,\n  output,\n  signal,\n} from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { injectDisabled } from 'ng-primitives/internal';\nimport { onBooleanChange } from 'ng-primitives/utils';\nimport { NgpFocusVisibleToken } from './focus-visible.token';\n\n@Directive({\n  standalone: true,\n  selector: '[ngpFocusVisible]',\n  exportAs: 'ngpFocusVisible',\n  providers: [{ provide: NgpFocusVisibleToken, useExisting: NgpFocusVisible }],\n  host: {\n    '[attr.data-focus-visible]': 'isFocused()',\n  },\n})\nexport class NgpFocusVisible {\n  /**\n   * Access the element that the directive is applied to.\n   */\n  private readonly elementRef = inject<ElementRef<HTMLElement>>(ElementRef);\n\n  /**\n   * Access the focus monitor.\n   */\n  private readonly focusMonitor = inject(FocusMonitor);\n\n  /**\n   * Whether focus events are listened to.\n   */\n  readonly disabled = input<boolean, BooleanInput>(false, {\n    alias: 'ngpFocusVisibleDisabled',\n    transform: booleanAttribute,\n  });\n\n  /**\n   * Access the disabled state from any parent.\n   */\n  private readonly isDisabled = injectDisabled(this.disabled);\n\n  /**\n   * Emit when the element is visually focused.\n   */\n  readonly focusChange = output<boolean>({\n    alias: 'ngpFocusVisible',\n  });\n\n  /**\n   * Whether the element is currently focused.\n   */\n  protected readonly isFocused = signal<boolean>(false);\n\n  constructor() {\n    // handle focus state\n    this.focusMonitor\n      .monitor(this.elementRef.nativeElement)\n      .pipe(takeUntilDestroyed())\n      .subscribe(origin =>\n        // null indicates the element was blurred\n        origin === null ? this.onBlur() : this.onFocus(origin),\n      );\n\n    // if the component becomes disabled and it is focused, hide the focus\n    onBooleanChange(this.isDisabled, () => this.focus(false));\n  }\n\n  private onFocus(origin: FocusOrigin): void {\n    if (this.isDisabled() || this.isFocused()) {\n      return;\n    }\n\n    // for some elements the focus visible state should always appear on focus\n    if (this.alwaysShowFocus()) {\n      this.focus(true);\n      return;\n    }\n\n    // if the focus origin is keyboard or program(focused programmatically), then the focus is visible\n    if (origin === 'keyboard' || origin === 'program') {\n      this.focus(true);\n      return;\n    }\n  }\n\n  private onBlur(): void {\n    if (this.isDisabled() || !this.isFocused()) {\n      return;\n    }\n\n    this.focus(false);\n  }\n\n  /**\n   * Trigger the focus signal along with the focusChange event.\n   */\n  private focus(value: boolean) {\n    if (this.isFocused() === value) {\n      return;\n    }\n\n    this.isFocused.set(value);\n    this.focusChange.emit(value);\n  }\n\n  private alwaysShowFocus(): boolean {\n    const nonTextInputTypes = [\n      'checkbox',\n      'radio',\n      'range',\n      'color',\n      'file',\n      'image',\n      'button',\n      'submit',\n      'reset',\n    ];\n\n    // if this is an input element and it is a text input\n    if (\n      this.elementRef.nativeElement instanceof HTMLInputElement &&\n      !nonTextInputTypes.includes(this.elementRef.nativeElement.type)\n    ) {\n      return true;\n    }\n\n    // if this is a textarea\n    if (this.elementRef.nativeElement instanceof HTMLTextAreaElement) {\n      return true;\n    }\n\n    // if this is an element with contenteditable\n    if (\n      this.elementRef.nativeElement.isContentEditable ||\n      this.elementRef.nativeElement.hasAttribute('contenteditable')\n    ) {\n      return true;\n    }\n\n    return false;\n  }\n}\n"]}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Copyright © 2024 Angular Primitives.
|
|
3
|
+
* https://github.com/ng-primitives/ng-primitives
|
|
4
|
+
*
|
|
5
|
+
* This source code is licensed under the Apache 2.0 license found in the
|
|
6
|
+
* LICENSE file in the root directory of this source tree.
|
|
7
|
+
*/
|
|
8
|
+
import { InjectionToken, inject } from '@angular/core';
|
|
9
|
+
export const defaultPopoverConfig = {
|
|
10
|
+
offset: 4,
|
|
11
|
+
placement: 'bottom',
|
|
12
|
+
showDelay: 0,
|
|
13
|
+
hideDelay: 0,
|
|
14
|
+
flip: true,
|
|
15
|
+
container: null,
|
|
16
|
+
closeOnOutsideClick: true,
|
|
17
|
+
};
|
|
18
|
+
export const NgpPopoverConfigToken = new InjectionToken('NgpPopoverConfigToken');
|
|
19
|
+
/**
|
|
20
|
+
* Provide the default Popover configuration
|
|
21
|
+
* @param config The Popover configuration
|
|
22
|
+
* @returns The provider
|
|
23
|
+
*/
|
|
24
|
+
export function providePopoverConfig(config) {
|
|
25
|
+
return [
|
|
26
|
+
{
|
|
27
|
+
provide: NgpPopoverConfigToken,
|
|
28
|
+
useValue: { ...defaultPopoverConfig, ...config },
|
|
29
|
+
},
|
|
30
|
+
];
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* Inject the Popover configuration
|
|
34
|
+
* @returns The global Popover configuration
|
|
35
|
+
*/
|
|
36
|
+
export function injectPopoverConfig() {
|
|
37
|
+
return inject(NgpPopoverConfigToken, { optional: true }) ?? defaultPopoverConfig;
|
|
38
|
+
}
|
|
39
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicG9wb3Zlci5jb25maWcuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9uZy1wcmltaXRpdmVzL3BvcG92ZXIvc3JjL2NvbmZpZy9wb3BvdmVyLmNvbmZpZy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7O0dBTUc7QUFDSCxPQUFPLEVBQUUsY0FBYyxFQUFZLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQStDakUsTUFBTSxDQUFDLE1BQU0sb0JBQW9CLEdBQXFCO0lBQ3BELE1BQU0sRUFBRSxDQUFDO0lBQ1QsU0FBUyxFQUFFLFFBQVE7SUFDbkIsU0FBUyxFQUFFLENBQUM7SUFDWixTQUFTLEVBQUUsQ0FBQztJQUNaLElBQUksRUFBRSxJQUFJO0lBQ1YsU0FBUyxFQUFFLElBQUk7SUFDZixtQkFBbUIsRUFBRSxJQUFJO0NBQzFCLENBQUM7QUFFRixNQUFNLENBQUMsTUFBTSxxQkFBcUIsR0FBRyxJQUFJLGNBQWMsQ0FBbUIsdUJBQXVCLENBQUMsQ0FBQztBQUVuRzs7OztHQUlHO0FBQ0gsTUFBTSxVQUFVLG9CQUFvQixDQUFDLE1BQWlDO0lBQ3BFLE9BQU87UUFDTDtZQUNFLE9BQU8sRUFBRSxxQkFBcUI7WUFDOUIsUUFBUSxFQUFFLEVBQUUsR0FBRyxvQkFBb0IsRUFBRSxHQUFHLE1BQU0sRUFBRTtTQUNqRDtLQUNGLENBQUM7QUFDSixDQUFDO0FBRUQ7OztHQUdHO0FBQ0gsTUFBTSxVQUFVLG1CQUFtQjtJQUNqQyxPQUFPLE1BQU0sQ0FBQyxxQkFBcUIsRUFBRSxFQUFFLFFBQVEsRUFBRSxJQUFJLEVBQUUsQ0FBQyxJQUFJLG9CQUFvQixDQUFDO0FBQ25GLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIENvcHlyaWdodCDCqSAyMDI0IEFuZ3VsYXIgUHJpbWl0aXZlcy5cbiAqIGh0dHBzOi8vZ2l0aHViLmNvbS9uZy1wcmltaXRpdmVzL25nLXByaW1pdGl2ZXNcbiAqXG4gKiBUaGlzIHNvdXJjZSBjb2RlIGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgMi4wIGxpY2Vuc2UgZm91bmQgaW4gdGhlXG4gKiBMSUNFTlNFIGZpbGUgaW4gdGhlIHJvb3QgZGlyZWN0b3J5IG9mIHRoaXMgc291cmNlIHRyZWUuXG4gKi9cbmltcG9ydCB7IEluamVjdGlvblRva2VuLCBQcm92aWRlciwgaW5qZWN0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyB0eXBlIFBsYWNlbWVudCB9IGZyb20gJ0BmbG9hdGluZy11aS9kb20nO1xuXG5leHBvcnQgaW50ZXJmYWNlIE5ncFBvcG92ZXJDb25maWcge1xuICAvKipcbiAgICogRGVmaW5lIHRoZSBvZmZzZXQgb2YgdGhlIHBvcG92ZXIgcmVsYXRpdmUgdG8gdGhlIHRyaWdnZXIuXG4gICAqIEBkZWZhdWx0IDRcbiAgICovXG4gIG9mZnNldDogbnVtYmVyO1xuXG4gIC8qKlxuICAgKiBEZWZpbmUgdGhlIHBsYWNlbWVudCBvZiB0aGUgcG9wb3ZlciByZWxhdGl2ZSB0byB0aGUgdHJpZ2dlci5cbiAgICogQGRlZmF1bHQgJ2JvdHRvbSdcbiAgICovXG4gIHBsYWNlbWVudDogUGxhY2VtZW50O1xuXG4gIC8qKlxuICAgKiBEZWZpbmUgdGhlIGRlbGF5IGJlZm9yZSB0aGUgcG9wb3ZlciBpcyBzaG93bi5cbiAgICogQGRlZmF1bHQgMFxuICAgKi9cbiAgc2hvd0RlbGF5OiBudW1iZXI7XG5cbiAgLyoqXG4gICAqIERlZmluZSB0aGUgZGVsYXkgYmVmb3JlIHRoZSBwb3BvdmVyIGlzIGhpZGRlbi5cbiAgICogQGRlZmF1bHQgMFxuICAgKi9cbiAgaGlkZURlbGF5OiBudW1iZXI7XG5cbiAgLyoqXG4gICAqIERlZmluZSB3aGV0aGVyIHRoZSBwb3BvdmVyIHNob3VsZCBmbGlwIHdoZW4gdGhlcmUgaXMgbm90IGVub3VnaCBzcGFjZSBmb3IgdGhlIHBvcG92ZXIuXG4gICAqIEBkZWZhdWx0IHRydWVcbiAgICovXG4gIGZsaXA6IGJvb2xlYW47XG5cbiAgLyoqXG4gICAqIERlZmluZSB0aGUgY29udGFpbmVyIGluIHRvIHdoaWNoIHRoZSBwb3BvdmVyIHNob3VsZCBiZSBhdHRhY2hlZC5cbiAgICogQGRlZmF1bHQgZG9jdW1lbnQuYm9keVxuICAgKi9cbiAgY29udGFpbmVyOiBIVE1MRWxlbWVudCB8IG51bGw7XG5cbiAgLyoqXG4gICAqIERlZmluZSB3aGV0aGVyIHRoZSBwb3BvdmVyIHNob3VsZCBjbG9zZSB3aGVuIGNsaWNraW5nIG91dHNpZGUgb2YgaXQuXG4gICAqIEBkZWZhdWx0IHRydWVcbiAgICovXG4gIGNsb3NlT25PdXRzaWRlQ2xpY2s6IGJvb2xlYW47XG59XG5cbmV4cG9ydCBjb25zdCBkZWZhdWx0UG9wb3ZlckNvbmZpZzogTmdwUG9wb3ZlckNvbmZpZyA9IHtcbiAgb2Zmc2V0OiA0LFxuICBwbGFjZW1lbnQ6ICdib3R0b20nLFxuICBzaG93RGVsYXk6IDAsXG4gIGhpZGVEZWxheTogMCxcbiAgZmxpcDogdHJ1ZSxcbiAgY29udGFpbmVyOiBudWxsLFxuICBjbG9zZU9uT3V0c2lkZUNsaWNrOiB0cnVlLFxufTtcblxuZXhwb3J0IGNvbnN0IE5ncFBvcG92ZXJDb25maWdUb2tlbiA9IG5ldyBJbmplY3Rpb25Ub2tlbjxOZ3BQb3BvdmVyQ29uZmlnPignTmdwUG9wb3ZlckNvbmZpZ1Rva2VuJyk7XG5cbi8qKlxuICogUHJvdmlkZSB0aGUgZGVmYXVsdCBQb3BvdmVyIGNvbmZpZ3VyYXRpb25cbiAqIEBwYXJhbSBjb25maWcgVGhlIFBvcG92ZXIgY29uZmlndXJhdGlvblxuICogQHJldHVybnMgVGhlIHByb3ZpZGVyXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBwcm92aWRlUG9wb3ZlckNvbmZpZyhjb25maWc6IFBhcnRpYWw8TmdwUG9wb3ZlckNvbmZpZz4pOiBQcm92aWRlcltdIHtcbiAgcmV0dXJuIFtcbiAgICB7XG4gICAgICBwcm92aWRlOiBOZ3BQb3BvdmVyQ29uZmlnVG9rZW4sXG4gICAgICB1c2VWYWx1ZTogeyAuLi5kZWZhdWx0UG9wb3ZlckNvbmZpZywgLi4uY29uZmlnIH0sXG4gICAgfSxcbiAgXTtcbn1cblxuLyoqXG4gKiBJbmplY3QgdGhlIFBvcG92ZXIgY29uZmlndXJhdGlvblxuICogQHJldHVybnMgVGhlIGdsb2JhbCBQb3BvdmVyIGNvbmZpZ3VyYXRpb25cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGluamVjdFBvcG92ZXJDb25maWcoKTogTmdwUG9wb3ZlckNvbmZpZyB7XG4gIHJldHVybiBpbmplY3QoTmdwUG9wb3ZlckNvbmZpZ1Rva2VuLCB7IG9wdGlvbmFsOiB0cnVlIH0pID8/IGRlZmF1bHRQb3BvdmVyQ29uZmlnO1xufVxuIl19
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Copyright © 2024 Angular Primitives.
|
|
3
|
+
* https://github.com/ng-primitives/ng-primitives
|
|
4
|
+
*
|
|
5
|
+
* This source code is licensed under the Apache 2.0 license found in the
|
|
6
|
+
* LICENSE file in the root directory of this source tree.
|
|
7
|
+
*/
|
|
8
|
+
export { providePopoverConfig } from './config/popover.config';
|
|
9
|
+
export { NgpPopoverTrigger } from './popover-trigger/popover-trigger.directive';
|
|
10
|
+
export { NgpPopoverTriggerToken } from './popover-trigger/popover-trigger.token';
|
|
11
|
+
export { NgpPopover } from './popover/popover.directive';
|
|
12
|
+
export { NgpPopoverToken } from './popover/popover.token';
|
|
13
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9uZy1wcmltaXRpdmVzL3BvcG92ZXIvc3JjL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7R0FNRztBQUVILE9BQU8sRUFBb0Isb0JBQW9CLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQUNqRixPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSw2Q0FBNkMsQ0FBQztBQUNoRixPQUFPLEVBQUUsc0JBQXNCLEVBQUUsTUFBTSx5Q0FBeUMsQ0FBQztBQUNqRixPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sNkJBQTZCLENBQUM7QUFDekQsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLHlCQUF5QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDb3B5cmlnaHQgwqkgMjAyNCBBbmd1bGFyIFByaW1pdGl2ZXMuXG4gKiBodHRwczovL2dpdGh1Yi5jb20vbmctcHJpbWl0aXZlcy9uZy1wcmltaXRpdmVzXG4gKlxuICogVGhpcyBzb3VyY2UgY29kZSBpcyBsaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIDIuMCBsaWNlbnNlIGZvdW5kIGluIHRoZVxuICogTElDRU5TRSBmaWxlIGluIHRoZSByb290IGRpcmVjdG9yeSBvZiB0aGlzIHNvdXJjZSB0cmVlLlxuICovXG5cbmV4cG9ydCB7IE5ncFBvcG92ZXJDb25maWcsIHByb3ZpZGVQb3BvdmVyQ29uZmlnIH0gZnJvbSAnLi9jb25maWcvcG9wb3Zlci5jb25maWcnO1xuZXhwb3J0IHsgTmdwUG9wb3ZlclRyaWdnZXIgfSBmcm9tICcuL3BvcG92ZXItdHJpZ2dlci9wb3BvdmVyLXRyaWdnZXIuZGlyZWN0aXZlJztcbmV4cG9ydCB7IE5ncFBvcG92ZXJUcmlnZ2VyVG9rZW4gfSBmcm9tICcuL3BvcG92ZXItdHJpZ2dlci9wb3BvdmVyLXRyaWdnZXIudG9rZW4nO1xuZXhwb3J0IHsgTmdwUG9wb3ZlciB9IGZyb20gJy4vcG9wb3Zlci9wb3BvdmVyLmRpcmVjdGl2ZSc7XG5leHBvcnQgeyBOZ3BQb3BvdmVyVG9rZW4gfSBmcm9tICcuL3BvcG92ZXIvcG9wb3Zlci50b2tlbic7XG4iXX0=
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Generated bundle index. Do not edit.
|
|
3
|
+
*/
|
|
4
|
+
export * from './index';
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmctcHJpbWl0aXZlcy1wb3BvdmVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvbmctcHJpbWl0aXZlcy9wb3BvdmVyL3NyYy9uZy1wcmltaXRpdmVzLXBvcG92ZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFFSCxjQUFjLFNBQVMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogR2VuZXJhdGVkIGJ1bmRsZSBpbmRleC4gRG8gbm90IGVkaXQuXG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9pbmRleCc7XG4iXX0=
|
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Copyright © 2024 Angular Primitives.
|
|
3
|
+
* https://github.com/ng-primitives/ng-primitives
|
|
4
|
+
*
|
|
5
|
+
* This source code is licensed under the Apache 2.0 license found in the
|
|
6
|
+
* LICENSE file in the root directory of this source tree.
|
|
7
|
+
*/
|
|
8
|
+
import { Directive, ElementRef, computed, inject, isDevMode } from '@angular/core';
|
|
9
|
+
import { NgpFocusTrap } from 'ng-primitives/focus-trap';
|
|
10
|
+
import { injectPopoverTrigger } from '../popover-trigger/popover-trigger.token';
|
|
11
|
+
import { NgpPopoverToken } from './popover.token';
|
|
12
|
+
import * as i0 from "@angular/core";
|
|
13
|
+
import * as i1 from "ng-primitives/focus-trap";
|
|
14
|
+
export class NgpPopover {
|
|
15
|
+
constructor() {
|
|
16
|
+
/**
|
|
17
|
+
* Access the popover element.
|
|
18
|
+
*/
|
|
19
|
+
this.popover = inject((ElementRef));
|
|
20
|
+
/**
|
|
21
|
+
* Access the trigger instance.
|
|
22
|
+
*/
|
|
23
|
+
this.trigger = injectPopoverTrigger();
|
|
24
|
+
/**
|
|
25
|
+
* Compute the x position of the popover.
|
|
26
|
+
*/
|
|
27
|
+
this.x = computed(() => this.trigger.position().x);
|
|
28
|
+
/**
|
|
29
|
+
* Compute the y position of the popover.
|
|
30
|
+
*/
|
|
31
|
+
this.y = computed(() => this.trigger.position().y);
|
|
32
|
+
}
|
|
33
|
+
ngOnInit() {
|
|
34
|
+
// if the element does not have a fixed position then throw a warning in dev mode
|
|
35
|
+
if (isDevMode()) {
|
|
36
|
+
const { position } = getComputedStyle(this.popover.nativeElement);
|
|
37
|
+
if (position !== 'absolute') {
|
|
38
|
+
console.warn(`The popover element must have an absolute position. The current position is ${position}.`);
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.1", ngImport: i0, type: NgpPopover, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
43
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.1.1", type: NgpPopover, isStandalone: true, selector: "[ngpPopover]", host: { attributes: { "role": "menu" }, properties: { "style.left.px": "x()", "style.top.px": "y()" } }, providers: [{ provide: NgpPopoverToken, useExisting: NgpPopover }], exportAs: ["ngpPopover"], hostDirectives: [{ directive: i1.NgpFocusTrap }], ngImport: i0 }); }
|
|
44
|
+
}
|
|
45
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.1", ngImport: i0, type: NgpPopover, decorators: [{
|
|
46
|
+
type: Directive,
|
|
47
|
+
args: [{
|
|
48
|
+
standalone: true,
|
|
49
|
+
selector: '[ngpPopover]',
|
|
50
|
+
exportAs: 'ngpPopover',
|
|
51
|
+
hostDirectives: [NgpFocusTrap],
|
|
52
|
+
providers: [{ provide: NgpPopoverToken, useExisting: NgpPopover }],
|
|
53
|
+
host: {
|
|
54
|
+
role: 'menu',
|
|
55
|
+
'[style.left.px]': 'x()',
|
|
56
|
+
'[style.top.px]': 'y()',
|
|
57
|
+
},
|
|
58
|
+
}]
|
|
59
|
+
}] });
|
|
60
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicG9wb3Zlci5kaXJlY3RpdmUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9uZy1wcmltaXRpdmVzL3BvcG92ZXIvc3JjL3BvcG92ZXIvcG9wb3Zlci5kaXJlY3RpdmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7OztHQU1HO0FBQ0gsT0FBTyxFQUFFLFNBQVMsRUFBRSxVQUFVLEVBQVUsUUFBUSxFQUFFLE1BQU0sRUFBRSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDM0YsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLDBCQUEwQixDQUFDO0FBQ3hELE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLDBDQUEwQyxDQUFDO0FBQ2hGLE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQzs7O0FBY2xELE1BQU0sT0FBTyxVQUFVO0lBWnZCO1FBYUU7O1dBRUc7UUFDYyxZQUFPLEdBQUcsTUFBTSxDQUFDLENBQUEsVUFBdUIsQ0FBQSxDQUFDLENBQUM7UUFFM0Q7O1dBRUc7UUFDYyxZQUFPLEdBQUcsb0JBQW9CLEVBQUUsQ0FBQztRQUVsRDs7V0FFRztRQUNnQixNQUFDLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRSxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsUUFBUSxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFFakU7O1dBRUc7UUFDZ0IsTUFBQyxHQUFHLFFBQVEsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLFFBQVEsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDO0tBY2xFO0lBWkMsUUFBUTtRQUNOLGlGQUFpRjtRQUNqRixJQUFJLFNBQVMsRUFBRSxFQUFFLENBQUM7WUFDaEIsTUFBTSxFQUFFLFFBQVEsRUFBRSxHQUFHLGdCQUFnQixDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsYUFBYSxDQUFDLENBQUM7WUFFbEUsSUFBSSxRQUFRLEtBQUssVUFBVSxFQUFFLENBQUM7Z0JBQzVCLE9BQU8sQ0FBQyxJQUFJLENBQ1YsK0VBQStFLFFBQVEsR0FBRyxDQUMzRixDQUFDO1lBQ0osQ0FBQztRQUNILENBQUM7SUFDSCxDQUFDOzhHQWhDVSxVQUFVO2tHQUFWLFVBQVUsb0tBUFYsQ0FBQyxFQUFFLE9BQU8sRUFBRSxlQUFlLEVBQUUsV0FBVyxFQUFFLFVBQVUsRUFBRSxDQUFDOzsyRkFPdkQsVUFBVTtrQkFadEIsU0FBUzttQkFBQztvQkFDVCxVQUFVLEVBQUUsSUFBSTtvQkFDaEIsUUFBUSxFQUFFLGNBQWM7b0JBQ3hCLFFBQVEsRUFBRSxZQUFZO29CQUN0QixjQUFjLEVBQUUsQ0FBQyxZQUFZLENBQUM7b0JBQzlCLFNBQVMsRUFBRSxDQUFDLEVBQUUsT0FBTyxFQUFFLGVBQWUsRUFBRSxXQUFXLFlBQVksRUFBRSxDQUFDO29CQUNsRSxJQUFJLEVBQUU7d0JBQ0osSUFBSSxFQUFFLE1BQU07d0JBQ1osaUJBQWlCLEVBQUUsS0FBSzt3QkFDeEIsZ0JBQWdCLEVBQUUsS0FBSztxQkFDeEI7aUJBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIENvcHlyaWdodCDCqSAyMDI0IEFuZ3VsYXIgUHJpbWl0aXZlcy5cbiAqIGh0dHBzOi8vZ2l0aHViLmNvbS9uZy1wcmltaXRpdmVzL25nLXByaW1pdGl2ZXNcbiAqXG4gKiBUaGlzIHNvdXJjZSBjb2RlIGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgMi4wIGxpY2Vuc2UgZm91bmQgaW4gdGhlXG4gKiBMSUNFTlNFIGZpbGUgaW4gdGhlIHJvb3QgZGlyZWN0b3J5IG9mIHRoaXMgc291cmNlIHRyZWUuXG4gKi9cbmltcG9ydCB7IERpcmVjdGl2ZSwgRWxlbWVudFJlZiwgT25Jbml0LCBjb21wdXRlZCwgaW5qZWN0LCBpc0Rldk1vZGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IE5ncEZvY3VzVHJhcCB9IGZyb20gJ25nLXByaW1pdGl2ZXMvZm9jdXMtdHJhcCc7XG5pbXBvcnQgeyBpbmplY3RQb3BvdmVyVHJpZ2dlciB9IGZyb20gJy4uL3BvcG92ZXItdHJpZ2dlci9wb3BvdmVyLXRyaWdnZXIudG9rZW4nO1xuaW1wb3J0IHsgTmdwUG9wb3ZlclRva2VuIH0gZnJvbSAnLi9wb3BvdmVyLnRva2VuJztcblxuQERpcmVjdGl2ZSh7XG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIHNlbGVjdG9yOiAnW25ncFBvcG92ZXJdJyxcbiAgZXhwb3J0QXM6ICduZ3BQb3BvdmVyJyxcbiAgaG9zdERpcmVjdGl2ZXM6IFtOZ3BGb2N1c1RyYXBdLFxuICBwcm92aWRlcnM6IFt7IHByb3ZpZGU6IE5ncFBvcG92ZXJUb2tlbiwgdXNlRXhpc3Rpbmc6IE5ncFBvcG92ZXIgfV0sXG4gIGhvc3Q6IHtcbiAgICByb2xlOiAnbWVudScsXG4gICAgJ1tzdHlsZS5sZWZ0LnB4XSc6ICd4KCknLFxuICAgICdbc3R5bGUudG9wLnB4XSc6ICd5KCknLFxuICB9LFxufSlcbmV4cG9ydCBjbGFzcyBOZ3BQb3BvdmVyIGltcGxlbWVudHMgT25Jbml0IHtcbiAgLyoqXG4gICAqIEFjY2VzcyB0aGUgcG9wb3ZlciBlbGVtZW50LlxuICAgKi9cbiAgcHJpdmF0ZSByZWFkb25seSBwb3BvdmVyID0gaW5qZWN0KEVsZW1lbnRSZWY8SFRNTEVsZW1lbnQ+KTtcblxuICAvKipcbiAgICogQWNjZXNzIHRoZSB0cmlnZ2VyIGluc3RhbmNlLlxuICAgKi9cbiAgcHJpdmF0ZSByZWFkb25seSB0cmlnZ2VyID0gaW5qZWN0UG9wb3ZlclRyaWdnZXIoKTtcblxuICAvKipcbiAgICogQ29tcHV0ZSB0aGUgeCBwb3NpdGlvbiBvZiB0aGUgcG9wb3Zlci5cbiAgICovXG4gIHByb3RlY3RlZCByZWFkb25seSB4ID0gY29tcHV0ZWQoKCkgPT4gdGhpcy50cmlnZ2VyLnBvc2l0aW9uKCkueCk7XG5cbiAgLyoqXG4gICAqIENvbXB1dGUgdGhlIHkgcG9zaXRpb24gb2YgdGhlIHBvcG92ZXIuXG4gICAqL1xuICBwcm90ZWN0ZWQgcmVhZG9ubHkgeSA9IGNvbXB1dGVkKCgpID0+IHRoaXMudHJpZ2dlci5wb3NpdGlvbigpLnkpO1xuXG4gIG5nT25Jbml0KCk6IHZvaWQge1xuICAgIC8vIGlmIHRoZSBlbGVtZW50IGRvZXMgbm90IGhhdmUgYSBmaXhlZCBwb3NpdGlvbiB0aGVuIHRocm93IGEgd2FybmluZyBpbiBkZXYgbW9kZVxuICAgIGlmIChpc0Rldk1vZGUoKSkge1xuICAgICAgY29uc3QgeyBwb3NpdGlvbiB9ID0gZ2V0Q29tcHV0ZWRTdHlsZSh0aGlzLnBvcG92ZXIubmF0aXZlRWxlbWVudCk7XG5cbiAgICAgIGlmIChwb3NpdGlvbiAhPT0gJ2Fic29sdXRlJykge1xuICAgICAgICBjb25zb2xlLndhcm4oXG4gICAgICAgICAgYFRoZSBwb3BvdmVyIGVsZW1lbnQgbXVzdCBoYXZlIGFuIGFic29sdXRlIHBvc2l0aW9uLiBUaGUgY3VycmVudCBwb3NpdGlvbiBpcyAke3Bvc2l0aW9ufS5gLFxuICAgICAgICApO1xuICAgICAgfVxuICAgIH1cbiAgfVxufVxuIl19
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Copyright © 2024 Angular Primitives.
|
|
3
|
+
* https://github.com/ng-primitives/ng-primitives
|
|
4
|
+
*
|
|
5
|
+
* This source code is licensed under the Apache 2.0 license found in the
|
|
6
|
+
* LICENSE file in the root directory of this source tree.
|
|
7
|
+
*/
|
|
8
|
+
import { InjectionToken, inject } from '@angular/core';
|
|
9
|
+
export const NgpPopoverToken = new InjectionToken('NgpPopoverToken');
|
|
10
|
+
/**
|
|
11
|
+
* Inject the Popover directive instance
|
|
12
|
+
*/
|
|
13
|
+
export function injectPopover() {
|
|
14
|
+
return inject(NgpPopoverToken);
|
|
15
|
+
}
|
|
16
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicG9wb3Zlci50b2tlbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL25nLXByaW1pdGl2ZXMvcG9wb3Zlci9zcmMvcG9wb3Zlci9wb3BvdmVyLnRva2VuLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7R0FNRztBQUNILE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBR3ZELE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBRyxJQUFJLGNBQWMsQ0FBYSxpQkFBaUIsQ0FBQyxDQUFDO0FBRWpGOztHQUVHO0FBQ0gsTUFBTSxVQUFVLGFBQWE7SUFDM0IsT0FBTyxNQUFNLENBQUMsZUFBZSxDQUFDLENBQUM7QUFDakMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQ29weXJpZ2h0IMKpIDIwMjQgQW5ndWxhciBQcmltaXRpdmVzLlxuICogaHR0cHM6Ly9naXRodWIuY29tL25nLXByaW1pdGl2ZXMvbmctcHJpbWl0aXZlc1xuICpcbiAqIFRoaXMgc291cmNlIGNvZGUgaXMgbGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSAyLjAgbGljZW5zZSBmb3VuZCBpbiB0aGVcbiAqIExJQ0VOU0UgZmlsZSBpbiB0aGUgcm9vdCBkaXJlY3Rvcnkgb2YgdGhpcyBzb3VyY2UgdHJlZS5cbiAqL1xuaW1wb3J0IHsgSW5qZWN0aW9uVG9rZW4sIGluamVjdCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHR5cGUgeyBOZ3BQb3BvdmVyIH0gZnJvbSAnLi9wb3BvdmVyLmRpcmVjdGl2ZSc7XG5cbmV4cG9ydCBjb25zdCBOZ3BQb3BvdmVyVG9rZW4gPSBuZXcgSW5qZWN0aW9uVG9rZW48TmdwUG9wb3Zlcj4oJ05ncFBvcG92ZXJUb2tlbicpO1xuXG4vKipcbiAqIEluamVjdCB0aGUgUG9wb3ZlciBkaXJlY3RpdmUgaW5zdGFuY2VcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGluamVjdFBvcG92ZXIoKTogTmdwUG9wb3ZlciB7XG4gIHJldHVybiBpbmplY3QoTmdwUG9wb3ZlclRva2VuKTtcbn1cbiJdfQ==
|
|
@@ -0,0 +1,222 @@
|
|
|
1
|
+
import { DomPortalOutlet, TemplatePortal } from '@angular/cdk/portal';
|
|
2
|
+
import { DOCUMENT } from '@angular/common';
|
|
3
|
+
import { Directive, ElementRef, Injector, ViewContainerRef, booleanAttribute, computed, inject, input, model, numberAttribute, signal, } from '@angular/core';
|
|
4
|
+
import { autoUpdate, computePosition, flip, offset, shift, } from '@floating-ui/dom';
|
|
5
|
+
import { injectDisposables, onBooleanChange } from 'ng-primitives/utils';
|
|
6
|
+
import { injectPopoverConfig } from '../config/popover.config';
|
|
7
|
+
import { NgpPopoverTriggerToken, providePopoverTrigger } from './popover-trigger.token';
|
|
8
|
+
import * as i0 from "@angular/core";
|
|
9
|
+
export class NgpPopoverTrigger {
|
|
10
|
+
constructor() {
|
|
11
|
+
/**
|
|
12
|
+
* Access the trigger element
|
|
13
|
+
*/
|
|
14
|
+
this.trigger = inject((ElementRef));
|
|
15
|
+
/**
|
|
16
|
+
* Access the view container ref.
|
|
17
|
+
*/
|
|
18
|
+
this.viewContainerRef = inject(ViewContainerRef);
|
|
19
|
+
/**
|
|
20
|
+
* Access the document.
|
|
21
|
+
*/
|
|
22
|
+
this.document = inject(DOCUMENT);
|
|
23
|
+
/**
|
|
24
|
+
* Access the injector.
|
|
25
|
+
*/
|
|
26
|
+
this.injector = inject(Injector);
|
|
27
|
+
/**
|
|
28
|
+
* Access the global popover configuration.
|
|
29
|
+
*/
|
|
30
|
+
this.config = injectPopoverConfig();
|
|
31
|
+
/**
|
|
32
|
+
* Access the disposable utilities
|
|
33
|
+
*/
|
|
34
|
+
this.disposables = injectDisposables();
|
|
35
|
+
/**
|
|
36
|
+
* Access the popover template ref.
|
|
37
|
+
*/
|
|
38
|
+
this.popover = input.required({
|
|
39
|
+
alias: 'ngpPopoverTrigger',
|
|
40
|
+
});
|
|
41
|
+
/**
|
|
42
|
+
* The open state of the popover.
|
|
43
|
+
* @default false
|
|
44
|
+
*/
|
|
45
|
+
this.open = model(false, {
|
|
46
|
+
alias: 'ngpPopoverTriggerOpen',
|
|
47
|
+
});
|
|
48
|
+
/**
|
|
49
|
+
* Define if the trigger should be disabled.
|
|
50
|
+
* @default false
|
|
51
|
+
*/
|
|
52
|
+
this.disabled = input(false, {
|
|
53
|
+
alias: 'ngpPopoverTriggerDisabled',
|
|
54
|
+
transform: booleanAttribute,
|
|
55
|
+
});
|
|
56
|
+
/**
|
|
57
|
+
* Define the placement of the popover relative to the trigger.
|
|
58
|
+
* @default 'top'
|
|
59
|
+
*/
|
|
60
|
+
this.placement = input(this.config.placement, {
|
|
61
|
+
alias: 'ngpPopoverTriggerPlacement',
|
|
62
|
+
});
|
|
63
|
+
/**
|
|
64
|
+
* Define the offset of the popover relative to the trigger.
|
|
65
|
+
* @default 0
|
|
66
|
+
*/
|
|
67
|
+
this.offset = input(this.config.offset, {
|
|
68
|
+
alias: 'ngpPopoverTriggerOffset',
|
|
69
|
+
transform: numberAttribute,
|
|
70
|
+
});
|
|
71
|
+
/**
|
|
72
|
+
* Define the delay before the popover is displayed.
|
|
73
|
+
* @default 0
|
|
74
|
+
*/
|
|
75
|
+
this.showDelay = input(this.config.showDelay, {
|
|
76
|
+
alias: 'ngpPopoverTriggerShowDelay',
|
|
77
|
+
transform: numberAttribute,
|
|
78
|
+
});
|
|
79
|
+
/**
|
|
80
|
+
* Define the delay before the popover is hidden.
|
|
81
|
+
* @default 0
|
|
82
|
+
*/
|
|
83
|
+
this.hideDelay = input(this.config.hideDelay, {
|
|
84
|
+
alias: 'ngpPopoverTriggerHideDelay',
|
|
85
|
+
transform: numberAttribute,
|
|
86
|
+
});
|
|
87
|
+
/**
|
|
88
|
+
* Define whether the popover should flip when there is not enough space for the popover.
|
|
89
|
+
* @default true
|
|
90
|
+
*/
|
|
91
|
+
this.flip = input(this.config.flip, {
|
|
92
|
+
alias: 'ngpPopoverTriggerFlip',
|
|
93
|
+
transform: booleanAttribute,
|
|
94
|
+
});
|
|
95
|
+
/**
|
|
96
|
+
* Define the container in which the popover should be attached.
|
|
97
|
+
* @default document.body
|
|
98
|
+
*/
|
|
99
|
+
this.container = input(this.config.container, {
|
|
100
|
+
alias: 'ngpPopoverTriggerContainer',
|
|
101
|
+
});
|
|
102
|
+
/**
|
|
103
|
+
* Define whether the popover should close when clicking outside of it.
|
|
104
|
+
* @default true
|
|
105
|
+
*/
|
|
106
|
+
this.closeOnOutsideClick = input(this.config.closeOnOutsideClick, {
|
|
107
|
+
alias: 'ngpPopoverTriggerCloseOnOutsideClick',
|
|
108
|
+
transform: booleanAttribute,
|
|
109
|
+
});
|
|
110
|
+
/**
|
|
111
|
+
* Store the popover view ref.
|
|
112
|
+
*/
|
|
113
|
+
this.viewRef = null;
|
|
114
|
+
/**
|
|
115
|
+
* Derive the popover middleware from the provided configuration.
|
|
116
|
+
*/
|
|
117
|
+
this.middleware = computed(() => {
|
|
118
|
+
const middleware = [offset(this.offset()), shift()];
|
|
119
|
+
if (this.flip()) {
|
|
120
|
+
middleware.push(flip());
|
|
121
|
+
}
|
|
122
|
+
return middleware;
|
|
123
|
+
});
|
|
124
|
+
/**
|
|
125
|
+
* Store the computed position of the popover.
|
|
126
|
+
* @internal
|
|
127
|
+
*/
|
|
128
|
+
this.position = signal({
|
|
129
|
+
x: 0,
|
|
130
|
+
y: 0,
|
|
131
|
+
});
|
|
132
|
+
/**
|
|
133
|
+
* Store the state of the popover.
|
|
134
|
+
* @internal
|
|
135
|
+
*/
|
|
136
|
+
this.state = signal('closed');
|
|
137
|
+
// any time the open state changes then show or hide the popover
|
|
138
|
+
onBooleanChange(this.open, this.show.bind(this), this.hide.bind(this));
|
|
139
|
+
}
|
|
140
|
+
ngOnDestroy() {
|
|
141
|
+
this.destroyPopover();
|
|
142
|
+
}
|
|
143
|
+
toggleOpenState() {
|
|
144
|
+
this.open.update(open => !open);
|
|
145
|
+
}
|
|
146
|
+
show() {
|
|
147
|
+
// if the trigger is disabled or the popover is already open then do not show the popover
|
|
148
|
+
if (this.disabled() || this.state() === 'open' || this.state() === 'opening') {
|
|
149
|
+
return;
|
|
150
|
+
}
|
|
151
|
+
this.state.set('opening');
|
|
152
|
+
this.disposables.setTimeout(() => this.createPopover(), this.showDelay());
|
|
153
|
+
// Add document click listener to detect outside clicks
|
|
154
|
+
if (this.closeOnOutsideClick()) {
|
|
155
|
+
this.documentClickListener = this.onDocumentClick.bind(this);
|
|
156
|
+
this.document.addEventListener('click', this.documentClickListener, true);
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
hide() {
|
|
160
|
+
// if the trigger is disabled or the popover is already closed then do not hide the popover
|
|
161
|
+
if (this.disabled() || this.state() === 'closed' || this.state() === 'closing') {
|
|
162
|
+
return;
|
|
163
|
+
}
|
|
164
|
+
this.state.set('closing');
|
|
165
|
+
this.disposables.setTimeout(() => this.destroyPopover(), this.hideDelay());
|
|
166
|
+
// Remove the document click listener when the popover is hidden
|
|
167
|
+
if (this.documentClickListener) {
|
|
168
|
+
this.document.removeEventListener('click', this.documentClickListener, true);
|
|
169
|
+
}
|
|
170
|
+
}
|
|
171
|
+
onDocumentClick(event) {
|
|
172
|
+
const target = event.target;
|
|
173
|
+
// Check if the click is outside the trigger or the popover
|
|
174
|
+
const isOutside = !this.trigger.nativeElement.contains(target) &&
|
|
175
|
+
!(this.viewRef?.rootNodes[0]).contains(target);
|
|
176
|
+
if (isOutside) {
|
|
177
|
+
// Close the popover
|
|
178
|
+
this.open.set(false);
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
createPopover() {
|
|
182
|
+
const portal = new TemplatePortal(this.popover(), this.viewContainerRef, undefined, this.injector);
|
|
183
|
+
const domOutlet = new DomPortalOutlet(this.container() ?? this.document.body, undefined, undefined, Injector.create({
|
|
184
|
+
parent: this.injector,
|
|
185
|
+
providers: [providePopoverTrigger(this)],
|
|
186
|
+
}));
|
|
187
|
+
this.viewRef = domOutlet.attach(portal);
|
|
188
|
+
this.viewRef.detectChanges();
|
|
189
|
+
const outletElement = this.viewRef.rootNodes[0];
|
|
190
|
+
this.dispose = autoUpdate(this.trigger.nativeElement, outletElement, async () => {
|
|
191
|
+
const position = await computePosition(this.trigger.nativeElement, outletElement, {
|
|
192
|
+
placement: this.placement(),
|
|
193
|
+
middleware: this.middleware(),
|
|
194
|
+
});
|
|
195
|
+
this.position.set({ x: position.x, y: position.y });
|
|
196
|
+
});
|
|
197
|
+
this.state.set('open');
|
|
198
|
+
}
|
|
199
|
+
destroyPopover() {
|
|
200
|
+
this.viewRef?.destroy();
|
|
201
|
+
this.viewRef = null;
|
|
202
|
+
this.dispose?.();
|
|
203
|
+
this.state.set('closed');
|
|
204
|
+
}
|
|
205
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.1", ngImport: i0, type: NgpPopoverTrigger, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
206
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.1.1", type: NgpPopoverTrigger, isStandalone: true, selector: "[ngpPopoverTrigger]", inputs: { popover: { classPropertyName: "popover", publicName: "ngpPopoverTrigger", isSignal: true, isRequired: true, transformFunction: null }, open: { classPropertyName: "open", publicName: "ngpPopoverTriggerOpen", isSignal: true, isRequired: false, transformFunction: null }, disabled: { classPropertyName: "disabled", publicName: "ngpPopoverTriggerDisabled", isSignal: true, isRequired: false, transformFunction: null }, placement: { classPropertyName: "placement", publicName: "ngpPopoverTriggerPlacement", isSignal: true, isRequired: false, transformFunction: null }, offset: { classPropertyName: "offset", publicName: "ngpPopoverTriggerOffset", isSignal: true, isRequired: false, transformFunction: null }, showDelay: { classPropertyName: "showDelay", publicName: "ngpPopoverTriggerShowDelay", isSignal: true, isRequired: false, transformFunction: null }, hideDelay: { classPropertyName: "hideDelay", publicName: "ngpPopoverTriggerHideDelay", isSignal: true, isRequired: false, transformFunction: null }, flip: { classPropertyName: "flip", publicName: "ngpPopoverTriggerFlip", isSignal: true, isRequired: false, transformFunction: null }, container: { classPropertyName: "container", publicName: "ngpPopoverTriggerContainer", isSignal: true, isRequired: false, transformFunction: null }, closeOnOutsideClick: { classPropertyName: "closeOnOutsideClick", publicName: "ngpPopoverTriggerCloseOnOutsideClick", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { open: "ngpPopoverTriggerOpenChange" }, host: { listeners: { "click": "toggleOpenState()" }, properties: { "attr.data-state": "state()", "attr.data-disabled": "disabled()" } }, providers: [{ provide: NgpPopoverTriggerToken, useExisting: NgpPopoverTrigger }], exportAs: ["ngpPopoverTrigger"], ngImport: i0 }); }
|
|
207
|
+
}
|
|
208
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.1", ngImport: i0, type: NgpPopoverTrigger, decorators: [{
|
|
209
|
+
type: Directive,
|
|
210
|
+
args: [{
|
|
211
|
+
standalone: true,
|
|
212
|
+
selector: '[ngpPopoverTrigger]',
|
|
213
|
+
exportAs: 'ngpPopoverTrigger',
|
|
214
|
+
providers: [{ provide: NgpPopoverTriggerToken, useExisting: NgpPopoverTrigger }],
|
|
215
|
+
host: {
|
|
216
|
+
'[attr.data-state]': 'state()',
|
|
217
|
+
'[attr.data-disabled]': 'disabled()',
|
|
218
|
+
'(click)': 'toggleOpenState()',
|
|
219
|
+
},
|
|
220
|
+
}]
|
|
221
|
+
}], ctorParameters: () => [] });
|
|
222
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"popover-trigger.directive.js","sourceRoot":"","sources":["../../../../../../packages/ng-primitives/popover/src/popover-trigger/popover-trigger.directive.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Copyright © 2024 Angular Primitives.\n * https://github.com/ng-primitives/ng-primitives\n *\n * This source code is licensed under the Apache 2.0 license found in the\n * LICENSE file in the root directory of this source tree.\n */\nimport { BooleanInput, NumberInput } from '@angular/cdk/coercion';\nimport { DomPortalOutlet, TemplatePortal } from '@angular/cdk/portal';\nimport { DOCUMENT } from '@angular/common';\nimport {\n  Directive,\n  ElementRef,\n  EmbeddedViewRef,\n  Injector,\n  OnDestroy,\n  TemplateRef,\n  ViewContainerRef,\n  booleanAttribute,\n  computed,\n  inject,\n  input,\n  model,\n  numberAttribute,\n  signal,\n} from '@angular/core';\nimport {\n  Middleware,\n  Placement,\n  autoUpdate,\n  computePosition,\n  flip,\n  offset,\n  shift,\n} from '@floating-ui/dom';\nimport { injectDisposables, onBooleanChange } from 'ng-primitives/utils';\nimport { injectPopoverConfig } from '../config/popover.config';\nimport { NgpPopoverTriggerToken, providePopoverTrigger } from './popover-trigger.token';\n\n@Directive({\n  standalone: true,\n  selector: '[ngpPopoverTrigger]',\n  exportAs: 'ngpPopoverTrigger',\n  providers: [{ provide: NgpPopoverTriggerToken, useExisting: NgpPopoverTrigger }],\n  host: {\n    '[attr.data-state]': 'state()',\n    '[attr.data-disabled]': 'disabled()',\n    '(click)': 'toggleOpenState()',\n  },\n})\nexport class NgpPopoverTrigger implements OnDestroy {\n  /**\n   * Access the trigger element\n   */\n  private readonly trigger = inject(ElementRef<HTMLElement>);\n\n  /**\n   * Access the view container ref.\n   */\n  private readonly viewContainerRef = inject(ViewContainerRef);\n\n  /**\n   * Access the document.\n   */\n  private readonly document = inject(DOCUMENT);\n\n  /**\n   * Access the injector.\n   */\n  private readonly injector = inject(Injector);\n\n  /**\n   * Access the global popover configuration.\n   */\n  private readonly config = injectPopoverConfig();\n\n  /**\n   * Access the disposable utilities\n   */\n  private readonly disposables = injectDisposables();\n\n  /**\n   * Access the popover template ref.\n   */\n  readonly popover = input.required<TemplateRef<void>>({\n    alias: 'ngpPopoverTrigger',\n  });\n\n  /**\n   * The open state of the popover.\n   * @default false\n   */\n  readonly open = model<boolean>(false, {\n    alias: 'ngpPopoverTriggerOpen',\n  });\n\n  /**\n   * Define if the trigger should be disabled.\n   * @default false\n   */\n  readonly disabled = input<boolean, BooleanInput>(false, {\n    alias: 'ngpPopoverTriggerDisabled',\n    transform: booleanAttribute,\n  });\n\n  /**\n   * Define the placement of the popover relative to the trigger.\n   * @default 'top'\n   */\n  readonly placement = input<Placement>(this.config.placement, {\n    alias: 'ngpPopoverTriggerPlacement',\n  });\n\n  /**\n   * Define the offset of the popover relative to the trigger.\n   * @default 0\n   */\n  readonly offset = input<number, NumberInput>(this.config.offset, {\n    alias: 'ngpPopoverTriggerOffset',\n    transform: numberAttribute,\n  });\n\n  /**\n   * Define the delay before the popover is displayed.\n   * @default 0\n   */\n  readonly showDelay = input<number, NumberInput>(this.config.showDelay, {\n    alias: 'ngpPopoverTriggerShowDelay',\n    transform: numberAttribute,\n  });\n\n  /**\n   * Define the delay before the popover is hidden.\n   * @default 0\n   */\n  readonly hideDelay = input<number, NumberInput>(this.config.hideDelay, {\n    alias: 'ngpPopoverTriggerHideDelay',\n    transform: numberAttribute,\n  });\n\n  /**\n   * Define whether the popover should flip when there is not enough space for the popover.\n   * @default true\n   */\n  readonly flip = input<boolean, BooleanInput>(this.config.flip, {\n    alias: 'ngpPopoverTriggerFlip',\n    transform: booleanAttribute,\n  });\n\n  /**\n   * Define the container in which the popover should be attached.\n   * @default document.body\n   */\n  readonly container = input<HTMLElement | null>(this.config.container, {\n    alias: 'ngpPopoverTriggerContainer',\n  });\n\n  /**\n   * Define whether the popover should close when clicking outside of it.\n   * @default true\n   */\n  readonly closeOnOutsideClick = input<boolean, BooleanInput>(this.config.closeOnOutsideClick, {\n    alias: 'ngpPopoverTriggerCloseOnOutsideClick',\n    transform: booleanAttribute,\n  });\n\n  /**\n   * Store the popover view ref.\n   */\n  private viewRef: EmbeddedViewRef<void> | null = null;\n\n  /**\n   * Derive the popover middleware from the provided configuration.\n   */\n  private readonly middleware = computed(() => {\n    const middleware: Middleware[] = [offset(this.offset()), shift()];\n\n    if (this.flip()) {\n      middleware.push(flip());\n    }\n\n    return middleware;\n  });\n\n  /**\n   * Store the computed position of the popover.\n   * @internal\n   */\n  readonly position = signal<{ x: number; y: number }>({\n    x: 0,\n    y: 0,\n  });\n\n  /**\n   * Store the state of the popover.\n   * @internal\n   */\n  readonly state = signal<PopoverState>('closed');\n\n  /**\n   * The dispose function to stop computing the position of the popover.\n   */\n  private dispose?: () => void;\n\n  /**\n   * A document-wide click listener that checks if the click\n   * occurred outside of the popover and trigger elements.\n   */\n  private documentClickListener?: (event: MouseEvent) => void;\n\n  constructor() {\n    // any time the open state changes then show or hide the popover\n    onBooleanChange(this.open, this.show.bind(this), this.hide.bind(this));\n  }\n\n  ngOnDestroy(): void {\n    this.destroyPopover();\n  }\n\n  toggleOpenState(): void {\n    this.open.update(open => !open);\n  }\n\n  private show(): void {\n    // if the trigger is disabled or the popover is already open then do not show the popover\n    if (this.disabled() || this.state() === 'open' || this.state() === 'opening') {\n      return;\n    }\n\n    this.state.set('opening');\n    this.disposables.setTimeout(() => this.createPopover(), this.showDelay());\n\n    // Add document click listener to detect outside clicks\n    if (this.closeOnOutsideClick()) {\n      this.documentClickListener = this.onDocumentClick.bind(this);\n      this.document.addEventListener('click', this.documentClickListener, true);\n    }\n  }\n\n  private hide(): void {\n    // if the trigger is disabled or the popover is already closed then do not hide the popover\n    if (this.disabled() || this.state() === 'closed' || this.state() === 'closing') {\n      return;\n    }\n\n    this.state.set('closing');\n    this.disposables.setTimeout(() => this.destroyPopover(), this.hideDelay());\n\n    // Remove the document click listener when the popover is hidden\n    if (this.documentClickListener) {\n      this.document.removeEventListener('click', this.documentClickListener, true);\n    }\n  }\n\n  private onDocumentClick(event: MouseEvent): void {\n    const target = event.target as HTMLElement;\n\n    // Check if the click is outside the trigger or the popover\n    const isOutside =\n      !this.trigger.nativeElement.contains(target) &&\n      !(this.viewRef?.rootNodes[0] as HTMLElement).contains(target);\n\n    if (isOutside) {\n      // Close the popover\n      this.open.set(false);\n    }\n  }\n\n  private createPopover(): void {\n    const portal = new TemplatePortal(\n      this.popover(),\n      this.viewContainerRef,\n      undefined,\n      this.injector,\n    );\n\n    const domOutlet = new DomPortalOutlet(\n      this.container() ?? this.document.body,\n      undefined,\n      undefined,\n      Injector.create({\n        parent: this.injector,\n        providers: [providePopoverTrigger(this)],\n      }),\n    );\n\n    this.viewRef = domOutlet.attach(portal);\n    this.viewRef.detectChanges();\n\n    const outletElement = this.viewRef.rootNodes[0];\n\n    this.dispose = autoUpdate(this.trigger.nativeElement, outletElement, async () => {\n      const position = await computePosition(this.trigger.nativeElement, outletElement, {\n        placement: this.placement(),\n        middleware: this.middleware(),\n      });\n\n      this.position.set({ x: position.x, y: position.y });\n    });\n\n    this.state.set('open');\n  }\n\n  private destroyPopover(): void {\n    this.viewRef?.destroy();\n    this.viewRef = null;\n    this.dispose?.();\n    this.state.set('closed');\n  }\n}\n\nexport type PopoverState = 'closed' | 'opening' | 'open' | 'closing';\n"]}
|