@ngutil/aria 0.0.80 → 0.0.82
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/fesm2022/ngutil-aria.mjs +88 -65
- package/fesm2022/ngutil-aria.mjs.map +1 -1
- package/focus/focus-state.directive.d.ts +1 -2
- package/focus/focus.service.d.ts +1 -1
- package/focus/focusable.directive.d.ts +0 -1
- package/package.json +7 -9
- package/ui-state/abstract.d.ts +1 -1
- package/ui-state/dirty.directive.d.ts +9 -0
- package/ui-state/index.d.ts +5 -3
- package/ui-state/progress-state.d.ts +1 -1
- package/esm2022/activity/activity.service.mjs +0 -49
- package/esm2022/activity/index.mjs +0 -2
- package/esm2022/focus/focus-state.directive.mjs +0 -63
- package/esm2022/focus/focus-trap.directive.mjs +0 -25
- package/esm2022/focus/focus.service.mjs +0 -145
- package/esm2022/focus/focusable.directive.mjs +0 -52
- package/esm2022/focus/index.mjs +0 -5
- package/esm2022/gestures/gesture-drag.mjs +0 -81
- package/esm2022/gestures/gesture-event.mjs +0 -44
- package/esm2022/gestures/gesture-longtap.mjs +0 -34
- package/esm2022/gestures/gesture-tap.mjs +0 -37
- package/esm2022/gestures/gesture.mjs +0 -66
- package/esm2022/gestures/gesture.service.mjs +0 -279
- package/esm2022/gestures/index.mjs +0 -7
- package/esm2022/index.mjs +0 -6
- package/esm2022/keystroke/index.mjs +0 -2
- package/esm2022/keystroke/keystroke.service.mjs +0 -92
- package/esm2022/ngutil-aria.mjs +0 -5
- package/esm2022/ui-state/abstract.mjs +0 -41
- package/esm2022/ui-state/busy.directive.mjs +0 -27
- package/esm2022/ui-state/disabled.directive.mjs +0 -28
- package/esm2022/ui-state/index.mjs +0 -25
- package/esm2022/ui-state/progress-state.mjs +0 -141
- package/esm2022/ui-state/readonly.directive.mjs +0 -27
- package/esm2022/ui-state/selector.mjs +0 -308
- package/esm2022/ui-state/ui-state.mjs +0 -91
|
@@ -1,6 +1,5 @@
|
|
|
1
1
|
import { Observable } from "rxjs";
|
|
2
|
-
import { ElementInput } from "
|
|
3
|
-
import { ConnectProtocol } from "@ngutil/common";
|
|
2
|
+
import { ConnectProtocol, ElementInput } from "@ngutil/common";
|
|
4
3
|
import { FocusChanges } from "./focus.service";
|
|
5
4
|
import * as i0 from "@angular/core";
|
|
6
5
|
export declare class FocusState implements ConnectProtocol {
|
package/focus/focus.service.d.ts
CHANGED
|
@@ -21,7 +21,7 @@ export declare class FocusService {
|
|
|
21
21
|
queryFocusable(inside: ElementInput): FocusableElement[];
|
|
22
22
|
getFirstFocusable(inside: ElementInput): FocusableElement | undefined;
|
|
23
23
|
isFocusable(node: ElementInput): boolean;
|
|
24
|
-
focusTrap(inside: ElementInput, deferCaptureElements?: boolean): import("@angular/cdk/a11y").FocusTrap;
|
|
24
|
+
focusTrap(inside: ElementInput, deferCaptureElements?: boolean): import("@angular/cdk/a11y-module.d-DrV0SO0k").FocusTrap;
|
|
25
25
|
static ɵfac: i0.ɵɵFactoryDeclaration<FocusService, never>;
|
|
26
26
|
static ɵprov: i0.ɵɵInjectableDeclaration<FocusService>;
|
|
27
27
|
}
|
|
@@ -7,7 +7,6 @@ export declare class Focusable implements ConnectProtocol {
|
|
|
7
7
|
readonly focusable: import("@angular/core").InputSignal<number | boolean>;
|
|
8
8
|
readonly tabindex: import("@angular/core").InputSignalWithTransform<number, any>;
|
|
9
9
|
readonly _tabindex: import("@angular/core").Signal<number>;
|
|
10
|
-
constructor();
|
|
11
10
|
connect(value: Focusable | FocusState | ElementInput): import("rxjs").Observable<unknown>;
|
|
12
11
|
static ɵfac: i0.ɵɵFactoryDeclaration<Focusable, never>;
|
|
13
12
|
static ɵdir: i0.ɵɵDirectiveDeclaration<Focusable, "[nuFocusable]", ["nuFocusable"], { "focusable": { "alias": "nuFocusable"; "required": false; "isSignal": true; }; "tabindex": { "alias": "tabindex"; "required": false; "isSignal": true; }; }, {}, never, never, true, [{ directive: typeof i1.FocusState; inputs: {}; outputs: {}; }]>;
|
package/package.json
CHANGED
|
@@ -1,16 +1,16 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@ngutil/aria",
|
|
3
|
-
"version": "0.0.
|
|
3
|
+
"version": "0.0.82",
|
|
4
4
|
"peerDependencies": {
|
|
5
|
-
"@angular/cdk": "
|
|
6
|
-
"@angular/common": "
|
|
7
|
-
"@angular/core": "
|
|
8
|
-
"
|
|
5
|
+
"@angular/cdk": "19.2.11",
|
|
6
|
+
"@angular/common": "19.2.8",
|
|
7
|
+
"@angular/core": "19.2.8",
|
|
8
|
+
"es-toolkit": "^1.35.0",
|
|
9
9
|
"rxjs": "^7.8.1",
|
|
10
10
|
"tabbable": "^6.2.0",
|
|
11
11
|
"utility-types": "^3.11.0",
|
|
12
|
-
"@ngutil/common": "0.0.
|
|
13
|
-
"@ngutil/style": "0.0.
|
|
12
|
+
"@ngutil/common": "0.0.82",
|
|
13
|
+
"@ngutil/style": "0.0.82"
|
|
14
14
|
},
|
|
15
15
|
"publishConfig": {
|
|
16
16
|
"access": "public",
|
|
@@ -25,8 +25,6 @@
|
|
|
25
25
|
},
|
|
26
26
|
".": {
|
|
27
27
|
"types": "./index.d.ts",
|
|
28
|
-
"esm2022": "./esm2022/ngutil-aria.mjs",
|
|
29
|
-
"esm": "./esm2022/ngutil-aria.mjs",
|
|
30
28
|
"default": "./fesm2022/ngutil-aria.mjs"
|
|
31
29
|
}
|
|
32
30
|
},
|
package/ui-state/abstract.d.ts
CHANGED
|
@@ -15,5 +15,5 @@ export declare abstract class AbstractUiState<N extends string> {
|
|
|
15
15
|
intercept(source: string): <S>(src: Observable<S>) => Observable<S>;
|
|
16
16
|
wrap<T>(observable: Observable<T>, source: string): Observable<T>;
|
|
17
17
|
static ɵfac: i0.ɵɵFactoryDeclaration<AbstractUiState<any>, never>;
|
|
18
|
-
static ɵdir: i0.ɵɵDirectiveDeclaration<AbstractUiState<any>, never, never, {}, {}, never, never,
|
|
18
|
+
static ɵdir: i0.ɵɵDirectiveDeclaration<AbstractUiState<any>, never, never, {}, {}, never, never, true, never>;
|
|
19
19
|
}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import { AbstractUiState } from "./abstract";
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
export declare class DirtyDirective extends AbstractUiState<"dirty"> {
|
|
4
|
+
readonly input: import("@angular/core").InputSignal<boolean>;
|
|
5
|
+
readonly when: import("@angular/core").InputSignal<string>;
|
|
6
|
+
constructor();
|
|
7
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<DirtyDirective, never>;
|
|
8
|
+
static ɵdir: i0.ɵɵDirectiveDeclaration<DirtyDirective, "[nuDirty], [nuDirtyWhen]", ["dirty"], { "input": { "alias": "nuDirty"; "required": false; "isSignal": true; }; "when": { "alias": "nuDirtyWhen"; "required": false; "isSignal": true; }; }, {}, never, never, true, never>;
|
|
9
|
+
}
|
package/ui-state/index.d.ts
CHANGED
|
@@ -1,15 +1,17 @@
|
|
|
1
1
|
import * as i0 from "@angular/core";
|
|
2
2
|
import * as i1 from "./busy.directive";
|
|
3
|
-
import * as i2 from "./
|
|
4
|
-
import * as i3 from "./
|
|
3
|
+
import * as i2 from "./dirty.directive";
|
|
4
|
+
import * as i3 from "./disabled.directive";
|
|
5
|
+
import * as i4 from "./readonly.directive";
|
|
5
6
|
export * from "./abstract";
|
|
6
7
|
export * from "./busy.directive";
|
|
7
8
|
export * from "./disabled.directive";
|
|
8
9
|
export * from "./progress-state";
|
|
9
10
|
export * from "./readonly.directive";
|
|
11
|
+
export * from "./dirty.directive";
|
|
10
12
|
export * from "./ui-state";
|
|
11
13
|
export declare class UiStateModule {
|
|
12
14
|
static ɵfac: i0.ɵɵFactoryDeclaration<UiStateModule, never>;
|
|
13
|
-
static ɵmod: i0.ɵɵNgModuleDeclaration<UiStateModule, never, [typeof i1.BusyDirective, typeof i2.
|
|
15
|
+
static ɵmod: i0.ɵɵNgModuleDeclaration<UiStateModule, never, [typeof i1.BusyDirective, typeof i2.DirtyDirective, typeof i3.DisabledDirective, typeof i4.ReadonlyDirective], [typeof i1.BusyDirective, typeof i2.DirtyDirective, typeof i3.DisabledDirective, typeof i4.ReadonlyDirective]>;
|
|
14
16
|
static ɵinj: i0.ɵɵInjectorDeclaration<UiStateModule>;
|
|
15
17
|
}
|
|
@@ -57,7 +57,7 @@ export declare class ProgressState {
|
|
|
57
57
|
segment(name: string, distribution?: number): ProgressSegmentRef;
|
|
58
58
|
connect(state: ProgressState, name: string, distribution?: number): Observable<void>;
|
|
59
59
|
static ɵfac: i0.ɵɵFactoryDeclaration<ProgressState, never>;
|
|
60
|
-
static ɵdir: i0.ɵɵDirectiveDeclaration<ProgressState, never, never, {}, {}, never, never,
|
|
60
|
+
static ɵdir: i0.ɵɵDirectiveDeclaration<ProgressState, never, never, {}, {}, never, never, true, never>;
|
|
61
61
|
}
|
|
62
62
|
export declare class ProgressSegmentRef {
|
|
63
63
|
#private;
|
|
@@ -1,49 +0,0 @@
|
|
|
1
|
-
import { DOCUMENT } from "@angular/common";
|
|
2
|
-
import { inject, Injectable, NgZone } from "@angular/core";
|
|
3
|
-
import { distinctUntilChanged, filter, fromEvent, map, merge, share, shareReplay, startWith, switchMap, take, throttleTime, timer } from "rxjs";
|
|
4
|
-
import * as i0 from "@angular/core";
|
|
5
|
-
const EVENT_OPTIONS = {
|
|
6
|
-
capture: true,
|
|
7
|
-
passive: true
|
|
8
|
-
};
|
|
9
|
-
const EVENT_ORIGIN = {
|
|
10
|
-
keydown: "keyboard",
|
|
11
|
-
mousedown: "mouse",
|
|
12
|
-
mousemove: "mouse",
|
|
13
|
-
touchstart: "touch"
|
|
14
|
-
};
|
|
15
|
-
export class ActivityService {
|
|
16
|
-
constructor() {
|
|
17
|
-
this.#zone = inject(NgZone);
|
|
18
|
-
this.#doc = inject(DOCUMENT);
|
|
19
|
-
this.events$ = this.#zone.runOutsideAngular(() => merge(fromEvent(this.#doc, "keydown", EVENT_OPTIONS), fromEvent(this.#doc, "mousedown", EVENT_OPTIONS), fromEvent(this.#doc, "mousemove", EVENT_OPTIONS), fromEvent(this.#doc, "touchstart", EVENT_OPTIONS)).pipe(map(event => {
|
|
20
|
-
const type = event.type;
|
|
21
|
-
return {
|
|
22
|
-
origin: EVENT_ORIGIN[type],
|
|
23
|
-
type: type,
|
|
24
|
-
node: event.target
|
|
25
|
-
};
|
|
26
|
-
}), share()));
|
|
27
|
-
}
|
|
28
|
-
#zone;
|
|
29
|
-
#doc;
|
|
30
|
-
watchActivity(node) {
|
|
31
|
-
if (node) {
|
|
32
|
-
return this.events$.pipe(filter(event => event.node != null &&
|
|
33
|
-
(event.node === node || (event.node instanceof HTMLElement && event.node.contains(node)))), shareReplay(1));
|
|
34
|
-
}
|
|
35
|
-
else {
|
|
36
|
-
return this.events$;
|
|
37
|
-
}
|
|
38
|
-
}
|
|
39
|
-
watchInactvity(timeout) {
|
|
40
|
-
return this.events$.pipe(startWith(null), throttleTime(timeout / 2), switchMap(() => timer(0, timeout).pipe(take(2))), map(v => v !== 0), distinctUntilChanged(), shareReplay(1));
|
|
41
|
-
}
|
|
42
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.6", ngImport: i0, type: ActivityService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
43
|
-
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.6", ngImport: i0, type: ActivityService, providedIn: "root" }); }
|
|
44
|
-
}
|
|
45
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.6", ngImport: i0, type: ActivityService, decorators: [{
|
|
46
|
-
type: Injectable,
|
|
47
|
-
args: [{ providedIn: "root" }]
|
|
48
|
-
}] });
|
|
49
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWN0aXZpdHkuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL2FyaWEvc3JjL2FjdGl2aXR5L2FjdGl2aXR5LnNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLGlCQUFpQixDQUFBO0FBQzFDLE9BQU8sRUFBRSxNQUFNLEVBQUUsVUFBVSxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQTtBQUUxRCxPQUFPLEVBQ0gsb0JBQW9CLEVBQ3BCLE1BQU0sRUFDTixTQUFTLEVBQ1QsR0FBRyxFQUNILEtBQUssRUFFTCxLQUFLLEVBQ0wsV0FBVyxFQUNYLFNBQVMsRUFDVCxTQUFTLEVBQ1QsSUFBSSxFQUNKLFlBQVksRUFDWixLQUFLLEVBQ1IsTUFBTSxNQUFNLENBQUE7O0FBV2IsTUFBTSxhQUFhLEdBQTRCO0lBQzNDLE9BQU8sRUFBRSxJQUFJO0lBQ2IsT0FBTyxFQUFFLElBQUk7Q0FDaEIsQ0FBQTtBQUVELE1BQU0sWUFBWSxHQUFHO0lBQ2pCLE9BQU8sRUFBRSxVQUFVO0lBQ25CLFNBQVMsRUFBRSxPQUFPO0lBQ2xCLFNBQVMsRUFBRSxPQUFPO0lBQ2xCLFVBQVUsRUFBRSxPQUFPO0NBQ3RCLENBQUE7QUFHRCxNQUFNLE9BQU8sZUFBZTtJQUQ1QjtRQUVhLFVBQUssR0FBRyxNQUFNLENBQUMsTUFBTSxDQUFDLENBQUE7UUFDdEIsU0FBSSxHQUFHLE1BQU0sQ0FBQyxRQUFRLENBQUMsQ0FBQTtRQUV2QixZQUFPLEdBQThCLElBQUksQ0FBQyxLQUFLLENBQUMsaUJBQWlCLENBQUMsR0FBRyxFQUFFLENBQzVFLEtBQUssQ0FDRCxTQUFTLENBQUMsSUFBSSxDQUFDLElBQUksRUFBRSxTQUFTLEVBQUUsYUFBYSxDQUFDLEVBQzlDLFNBQVMsQ0FBQyxJQUFJLENBQUMsSUFBSSxFQUFFLFdBQVcsRUFBRSxhQUFhLENBQUMsRUFDaEQsU0FBUyxDQUFDLElBQUksQ0FBQyxJQUFJLEVBQUUsV0FBVyxFQUFFLGFBQWEsQ0FBQyxFQUNoRCxTQUFTLENBQUMsSUFBSSxDQUFDLElBQUksRUFBRSxZQUFZLEVBQUUsYUFBYSxDQUFDLENBQ3BELENBQUMsSUFBSSxDQUNGLEdBQUcsQ0FBQyxLQUFLLENBQUMsRUFBRTtZQUNSLE1BQU0sSUFBSSxHQUFHLEtBQUssQ0FBQyxJQUE2QixDQUFBO1lBQ2hELE9BQU87Z0JBQ0gsTUFBTSxFQUFFLFlBQVksQ0FBQyxJQUFJLENBQW1CO2dCQUM1QyxJQUFJLEVBQUUsSUFBSTtnQkFDVixJQUFJLEVBQUUsS0FBSyxDQUFDLE1BQWE7YUFDNUIsQ0FBQTtRQUNMLENBQUMsQ0FBQyxFQUNGLEtBQUssRUFBRSxDQUNWLENBQ0osQ0FBQTtLQTJCSjtJQS9DWSxLQUFLLENBQWlCO0lBQ3RCLElBQUksQ0FBbUI7SUFxQmhDLGFBQWEsQ0FBQyxJQUFrQjtRQUM1QixJQUFJLElBQUksRUFBRSxDQUFDO1lBQ1AsT0FBTyxJQUFJLENBQUMsT0FBTyxDQUFDLElBQUksQ0FDcEIsTUFBTSxDQUNGLEtBQUssQ0FBQyxFQUFFLENBQ0osS0FBSyxDQUFDLElBQUksSUFBSSxJQUFJO2dCQUNsQixDQUFDLEtBQUssQ0FBQyxJQUFJLEtBQUssSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksWUFBWSxXQUFXLElBQUksS0FBSyxDQUFDLElBQUksQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUNoRyxFQUNELFdBQVcsQ0FBQyxDQUFDLENBQUMsQ0FDakIsQ0FBQTtRQUNMLENBQUM7YUFBTSxDQUFDO1lBQ0osT0FBTyxJQUFJLENBQUMsT0FBTyxDQUFBO1FBQ3ZCLENBQUM7SUFDTCxDQUFDO0lBRUQsY0FBYyxDQUFDLE9BQWU7UUFDMUIsT0FBTyxJQUFJLENBQUMsT0FBTyxDQUFDLElBQUksQ0FDcEIsU0FBUyxDQUFDLElBQUksQ0FBQyxFQUNmLFlBQVksQ0FBQyxPQUFPLEdBQUcsQ0FBQyxDQUFDLEVBQ3pCLFNBQVMsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxLQUFLLENBQUMsQ0FBQyxFQUFFLE9BQU8sQ0FBQyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxFQUNoRCxHQUFHLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFDLEVBQ2pCLG9CQUFvQixFQUFFLEVBQ3RCLFdBQVcsQ0FBQyxDQUFDLENBQUMsQ0FDakIsQ0FBQTtJQUNMLENBQUM7OEdBL0NRLGVBQWU7a0hBQWYsZUFBZSxjQURGLE1BQU07OzJGQUNuQixlQUFlO2tCQUQzQixVQUFVO21CQUFDLEVBQUUsVUFBVSxFQUFFLE1BQU0sRUFBRSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IERPQ1VNRU5UIH0gZnJvbSBcIkBhbmd1bGFyL2NvbW1vblwiXG5pbXBvcnQgeyBpbmplY3QsIEluamVjdGFibGUsIE5nWm9uZSB9IGZyb20gXCJAYW5ndWxhci9jb3JlXCJcblxuaW1wb3J0IHtcbiAgICBkaXN0aW5jdFVudGlsQ2hhbmdlZCxcbiAgICBmaWx0ZXIsXG4gICAgZnJvbUV2ZW50LFxuICAgIG1hcCxcbiAgICBtZXJnZSxcbiAgICBPYnNlcnZhYmxlLFxuICAgIHNoYXJlLFxuICAgIHNoYXJlUmVwbGF5LFxuICAgIHN0YXJ0V2l0aCxcbiAgICBzd2l0Y2hNYXAsXG4gICAgdGFrZSxcbiAgICB0aHJvdHRsZVRpbWUsXG4gICAgdGltZXJcbn0gZnJvbSBcInJ4anNcIlxuXG4vLyBUT0RPOiBkZXRlY3QgcHJvZ3JhbSBhY3Rpdml0eVxuZXhwb3J0IHR5cGUgQWN0aXZpdHlPcmlnaW4gPSBcIm1vdXNlXCIgfCBcImtleWJvYXJkXCIgfCBcInRvdWNoXCIgfCBcInByb2dyYW1cIlxuXG5leHBvcnQgaW50ZXJmYWNlIEFjdGl2aXR5RXZlbnQge1xuICAgIG9yaWdpbjogQWN0aXZpdHlPcmlnaW5cbiAgICB0eXBlOiBrZXlvZiB0eXBlb2YgRVZFTlRfT1JJR0lOXG4gICAgbm9kZT86IE5vZGVcbn1cblxuY29uc3QgRVZFTlRfT1BUSU9OUzogQWRkRXZlbnRMaXN0ZW5lck9wdGlvbnMgPSB7XG4gICAgY2FwdHVyZTogdHJ1ZSxcbiAgICBwYXNzaXZlOiB0cnVlXG59XG5cbmNvbnN0IEVWRU5UX09SSUdJTiA9IHtcbiAgICBrZXlkb3duOiBcImtleWJvYXJkXCIsXG4gICAgbW91c2Vkb3duOiBcIm1vdXNlXCIsXG4gICAgbW91c2Vtb3ZlOiBcIm1vdXNlXCIsXG4gICAgdG91Y2hzdGFydDogXCJ0b3VjaFwiXG59XG5cbkBJbmplY3RhYmxlKHsgcHJvdmlkZWRJbjogXCJyb290XCIgfSlcbmV4cG9ydCBjbGFzcyBBY3Rpdml0eVNlcnZpY2Uge1xuICAgIHJlYWRvbmx5ICN6b25lID0gaW5qZWN0KE5nWm9uZSlcbiAgICByZWFkb25seSAjZG9jID0gaW5qZWN0KERPQ1VNRU5UKVxuXG4gICAgcmVhZG9ubHkgZXZlbnRzJDogT2JzZXJ2YWJsZTxBY3Rpdml0eUV2ZW50PiA9IHRoaXMuI3pvbmUucnVuT3V0c2lkZUFuZ3VsYXIoKCkgPT5cbiAgICAgICAgbWVyZ2UoXG4gICAgICAgICAgICBmcm9tRXZlbnQodGhpcy4jZG9jLCBcImtleWRvd25cIiwgRVZFTlRfT1BUSU9OUyksXG4gICAgICAgICAgICBmcm9tRXZlbnQodGhpcy4jZG9jLCBcIm1vdXNlZG93blwiLCBFVkVOVF9PUFRJT05TKSxcbiAgICAgICAgICAgIGZyb21FdmVudCh0aGlzLiNkb2MsIFwibW91c2Vtb3ZlXCIsIEVWRU5UX09QVElPTlMpLFxuICAgICAgICAgICAgZnJvbUV2ZW50KHRoaXMuI2RvYywgXCJ0b3VjaHN0YXJ0XCIsIEVWRU5UX09QVElPTlMpXG4gICAgICAgICkucGlwZShcbiAgICAgICAgICAgIG1hcChldmVudCA9PiB7XG4gICAgICAgICAgICAgICAgY29uc3QgdHlwZSA9IGV2ZW50LnR5cGUgYXMgQWN0aXZpdHlFdmVudFtcInR5cGVcIl1cbiAgICAgICAgICAgICAgICByZXR1cm4ge1xuICAgICAgICAgICAgICAgICAgICBvcmlnaW46IEVWRU5UX09SSUdJTlt0eXBlXSBhcyBBY3Rpdml0eU9yaWdpbixcbiAgICAgICAgICAgICAgICAgICAgdHlwZTogdHlwZSxcbiAgICAgICAgICAgICAgICAgICAgbm9kZTogZXZlbnQudGFyZ2V0IGFzIGFueVxuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH0pLFxuICAgICAgICAgICAgc2hhcmUoKVxuICAgICAgICApXG4gICAgKVxuXG4gICAgd2F0Y2hBY3Rpdml0eShub2RlPzogSFRNTEVsZW1lbnQpIHtcbiAgICAgICAgaWYgKG5vZGUpIHtcbiAgICAgICAgICAgIHJldHVybiB0aGlzLmV2ZW50cyQucGlwZShcbiAgICAgICAgICAgICAgICBmaWx0ZXIoXG4gICAgICAgICAgICAgICAgICAgIGV2ZW50ID0+XG4gICAgICAgICAgICAgICAgICAgICAgICBldmVudC5ub2RlICE9IG51bGwgJiZcbiAgICAgICAgICAgICAgICAgICAgICAgIChldmVudC5ub2RlID09PSBub2RlIHx8IChldmVudC5ub2RlIGluc3RhbmNlb2YgSFRNTEVsZW1lbnQgJiYgZXZlbnQubm9kZS5jb250YWlucyhub2RlKSkpXG4gICAgICAgICAgICAgICAgKSxcbiAgICAgICAgICAgICAgICBzaGFyZVJlcGxheSgxKVxuICAgICAgICAgICAgKVxuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgcmV0dXJuIHRoaXMuZXZlbnRzJFxuICAgICAgICB9XG4gICAgfVxuXG4gICAgd2F0Y2hJbmFjdHZpdHkodGltZW91dDogbnVtYmVyKSB7XG4gICAgICAgIHJldHVybiB0aGlzLmV2ZW50cyQucGlwZShcbiAgICAgICAgICAgIHN0YXJ0V2l0aChudWxsKSxcbiAgICAgICAgICAgIHRocm90dGxlVGltZSh0aW1lb3V0IC8gMiksXG4gICAgICAgICAgICBzd2l0Y2hNYXAoKCkgPT4gdGltZXIoMCwgdGltZW91dCkucGlwZSh0YWtlKDIpKSksXG4gICAgICAgICAgICBtYXAodiA9PiB2ICE9PSAwKSxcbiAgICAgICAgICAgIGRpc3RpbmN0VW50aWxDaGFuZ2VkKCksXG4gICAgICAgICAgICBzaGFyZVJlcGxheSgxKVxuICAgICAgICApXG4gICAgfVxufVxuIl19
|
|
@@ -1,2 +0,0 @@
|
|
|
1
|
-
export * from "./activity.service";
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9hcmlhL3NyYy9hY3Rpdml0eS9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLG9CQUFvQixDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSBcIi4vYWN0aXZpdHkuc2VydmljZVwiXG4iXX0=
|
|
@@ -1,63 +0,0 @@
|
|
|
1
|
-
var _a;
|
|
2
|
-
import { Directive, effect, ElementRef, inject } from "@angular/core";
|
|
3
|
-
import { takeUntilDestroyed, toSignal } from "@angular/core/rxjs-interop";
|
|
4
|
-
import { BehaviorSubject, combineLatest, debounceTime, map, Observable, of, shareReplay, switchMap } from "rxjs";
|
|
5
|
-
import { isElementInput } from "@ngutil/common";
|
|
6
|
-
import { FocusService } from "./focus.service";
|
|
7
|
-
import * as i0 from "@angular/core";
|
|
8
|
-
export class FocusState {
|
|
9
|
-
#focus;
|
|
10
|
-
#el;
|
|
11
|
-
#parent;
|
|
12
|
-
#default;
|
|
13
|
-
#self;
|
|
14
|
-
#connected;
|
|
15
|
-
#connEvent;
|
|
16
|
-
constructor() {
|
|
17
|
-
this.#focus = inject(FocusService);
|
|
18
|
-
this.#el = inject(ElementRef);
|
|
19
|
-
this.#parent = inject(_a, { optional: true, skipSelf: true });
|
|
20
|
-
this.#default = { element: this.#el.nativeElement, origin: null };
|
|
21
|
-
this.#self = this.#focus.watch(this.#el).pipe(takeUntilDestroyed(), shareReplay(1));
|
|
22
|
-
this.#connected = new BehaviorSubject([]);
|
|
23
|
-
this.#connEvent = this.#connected.pipe(switchMap(values => (values.length === 0 ? of([]) : combineLatest(values))), takeUntilDestroyed(), map(values => values.filter(v => v.origin != null)), map(values => values[0] || this.#default), shareReplay(1));
|
|
24
|
-
this.event$ = combineLatest([this.#self, this.#connEvent]).pipe(debounceTime(100), map(values => values.find(v => v.origin != null) || this.#default), shareReplay(1));
|
|
25
|
-
this.origin$ = this.event$.pipe(map(event => event.origin));
|
|
26
|
-
this.within$ = this.event$.pipe(map(event => (event.element !== this.#el.nativeElement ? event.origin : null)));
|
|
27
|
-
this.event = toSignal(this.event$, { rejectErrors: true, manualCleanup: true });
|
|
28
|
-
this.origin = toSignal(this.origin$, { rejectErrors: true, manualCleanup: true });
|
|
29
|
-
this.within = toSignal(this.within$, { rejectErrors: true, manualCleanup: true });
|
|
30
|
-
// TODO: miért kell ez?, ha nincs itt akkor nem frissül
|
|
31
|
-
effect(() => this.origin(), { allowSignalWrites: false });
|
|
32
|
-
this.#parent?.connect(this).pipe(takeUntilDestroyed()).subscribe();
|
|
33
|
-
}
|
|
34
|
-
connect(value) {
|
|
35
|
-
return new Observable(() => {
|
|
36
|
-
const src = (isElementInput(value) ? this.#focus.watch(value) : value.event$).pipe(shareReplay(1));
|
|
37
|
-
this.#connected.next([...this.#connected.value, src]);
|
|
38
|
-
return () => {
|
|
39
|
-
let current = this.#connected.value;
|
|
40
|
-
const idx = current.indexOf(src);
|
|
41
|
-
if (idx >= 0) {
|
|
42
|
-
current = [...current];
|
|
43
|
-
current.splice(idx, 1);
|
|
44
|
-
this.#connected.next(current);
|
|
45
|
-
}
|
|
46
|
-
};
|
|
47
|
-
});
|
|
48
|
-
}
|
|
49
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.6", ngImport: i0, type: FocusState, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
50
|
-
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.6", type: FocusState, isStandalone: true, host: { properties: { "attr.focus": "origin()", "attr.focusWithin": "within()" } }, ngImport: i0 }); }
|
|
51
|
-
}
|
|
52
|
-
_a = FocusState;
|
|
53
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.6", ngImport: i0, type: FocusState, decorators: [{
|
|
54
|
-
type: Directive,
|
|
55
|
-
args: [{
|
|
56
|
-
standalone: true,
|
|
57
|
-
host: {
|
|
58
|
-
"[attr.focus]": "origin()",
|
|
59
|
-
"[attr.focusWithin]": "within()"
|
|
60
|
-
}
|
|
61
|
-
}]
|
|
62
|
-
}], ctorParameters: () => [] });
|
|
63
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"focus-state.directive.js","sourceRoot":"","sources":["../../../../../packages/aria/src/focus/focus-state.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directive, effect, ElementRef, inject } from \"@angular/core\"\nimport { takeUntilDestroyed, toSignal } from \"@angular/core/rxjs-interop\"\n\nimport { BehaviorSubject, combineLatest, debounceTime, map, Observable, of, shareReplay, switchMap } from \"rxjs\"\n\nimport { ElementInput } from \"dist/packages/common\"\n\nimport { ConnectProtocol, isElementInput } from \"@ngutil/common\"\n\nimport { FocusChanges, FocusService } from \"./focus.service\"\n\n@Directive({\n    standalone: true,\n    host: {\n        \"[attr.focus]\": \"origin()\",\n        \"[attr.focusWithin]\": \"within()\"\n    }\n})\nexport class FocusState implements ConnectProtocol {\n    readonly #focus = inject(FocusService)\n    readonly #el = inject<ElementRef<HTMLElement>>(ElementRef)\n    readonly #parent = inject(FocusState, { optional: true, skipSelf: true })\n    readonly #default: FocusChanges = { element: this.#el.nativeElement, origin: null }\n\n    readonly #self = this.#focus.watch(this.#el).pipe(takeUntilDestroyed(), shareReplay(1))\n\n    readonly #connected = new BehaviorSubject<Observable<FocusChanges>[]>([])\n    readonly #connEvent = this.#connected.pipe(\n        switchMap(values => (values.length === 0 ? of([]) : combineLatest(values))),\n        takeUntilDestroyed(),\n        map(values => values.filter(v => v.origin != null)),\n        map(values => values[0] || this.#default),\n        shareReplay(1)\n    )\n\n    readonly event$: Observable<FocusChanges> = combineLatest([this.#self, this.#connEvent]).pipe(\n        debounceTime(100),\n        map(values => values.find(v => v.origin != null) || this.#default),\n        shareReplay(1)\n    )\n\n    readonly origin$ = this.event$.pipe(map(event => event.origin))\n    readonly within$ = this.event$.pipe(map(event => (event.element !== this.#el.nativeElement ? event.origin : null)))\n\n    readonly event = toSignal(this.event$, { rejectErrors: true, manualCleanup: true })\n    readonly origin = toSignal(this.origin$, { rejectErrors: true, manualCleanup: true })\n    readonly within = toSignal(this.within$, { rejectErrors: true, manualCleanup: true })\n\n    constructor() {\n        // TODO: miért kell ez?, ha nincs itt akkor nem frissül\n        effect(() => this.origin(), { allowSignalWrites: false })\n        this.#parent?.connect(this).pipe(takeUntilDestroyed()).subscribe()\n    }\n\n    connect(value: FocusState | ElementInput) {\n        return new Observable(() => {\n            const src = (isElementInput(value) ? this.#focus.watch(value) : value.event$).pipe(shareReplay(1))\n            this.#connected.next([...this.#connected.value, src])\n\n            return () => {\n                let current = this.#connected.value\n                const idx = current.indexOf(src)\n                if (idx >= 0) {\n                    current = [...current]\n                    current.splice(idx, 1)\n                    this.#connected.next(current)\n                }\n            }\n        })\n    }\n}\n"]}
|
|
@@ -1,25 +0,0 @@
|
|
|
1
|
-
import { ElementRef, inject } from "@angular/core";
|
|
2
|
-
import { BehaviorSubject } from "rxjs";
|
|
3
|
-
import { Destructible } from "@ngutil/common";
|
|
4
|
-
import { FocusService } from "./focus.service";
|
|
5
|
-
// TODO: implement
|
|
6
|
-
export class FocusTrap extends Destructible {
|
|
7
|
-
#manager = inject(FocusService);
|
|
8
|
-
#el = inject((ElementRef));
|
|
9
|
-
get enabled() {
|
|
10
|
-
return this.#enabled.value;
|
|
11
|
-
}
|
|
12
|
-
set enabled(val) {
|
|
13
|
-
if (this.#enabled.value !== val) {
|
|
14
|
-
this.#enabled.next(val);
|
|
15
|
-
}
|
|
16
|
-
}
|
|
17
|
-
#enabled = new BehaviorSubject(true);
|
|
18
|
-
enable() {
|
|
19
|
-
this.enabled = true;
|
|
20
|
-
}
|
|
21
|
-
disable() {
|
|
22
|
-
this.enabled = false;
|
|
23
|
-
}
|
|
24
|
-
}
|
|
25
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9jdXMtdHJhcC5kaXJlY3RpdmUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9hcmlhL3NyYy9mb2N1cy9mb2N1cy10cmFwLmRpcmVjdGl2ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQTtBQUVsRCxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sTUFBTSxDQUFBO0FBRXRDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQTtBQUU3QyxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUE7QUFFOUMsa0JBQWtCO0FBQ2xCLE1BQU0sT0FBTyxTQUFVLFNBQVEsWUFBWTtJQUN2QyxRQUFRLEdBQUcsTUFBTSxDQUFDLFlBQVksQ0FBQyxDQUFBO0lBQy9CLEdBQUcsR0FBRyxNQUFNLENBQUMsQ0FBQSxVQUF1QixDQUFBLENBQUMsQ0FBQTtJQUVyQyxJQUFJLE9BQU87UUFDUCxPQUFPLElBQUksQ0FBQyxRQUFRLENBQUMsS0FBSyxDQUFBO0lBQzlCLENBQUM7SUFDRCxJQUFJLE9BQU8sQ0FBQyxHQUFZO1FBQ3BCLElBQUksSUFBSSxDQUFDLFFBQVEsQ0FBQyxLQUFLLEtBQUssR0FBRyxFQUFFLENBQUM7WUFDOUIsSUFBSSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUE7UUFDM0IsQ0FBQztJQUNMLENBQUM7SUFDRCxRQUFRLEdBQUcsSUFBSSxlQUFlLENBQVUsSUFBSSxDQUFDLENBQUE7SUFFN0MsTUFBTTtRQUNGLElBQUksQ0FBQyxPQUFPLEdBQUcsSUFBSSxDQUFBO0lBQ3ZCLENBQUM7SUFFRCxPQUFPO1FBQ0gsSUFBSSxDQUFDLE9BQU8sR0FBRyxLQUFLLENBQUE7SUFDeEIsQ0FBQztDQUNKIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRWxlbWVudFJlZiwgaW5qZWN0IH0gZnJvbSBcIkBhbmd1bGFyL2NvcmVcIlxuXG5pbXBvcnQgeyBCZWhhdmlvclN1YmplY3QgfSBmcm9tIFwicnhqc1wiXG5cbmltcG9ydCB7IERlc3RydWN0aWJsZSB9IGZyb20gXCJAbmd1dGlsL2NvbW1vblwiXG5cbmltcG9ydCB7IEZvY3VzU2VydmljZSB9IGZyb20gXCIuL2ZvY3VzLnNlcnZpY2VcIlxuXG4vLyBUT0RPOiBpbXBsZW1lbnRcbmV4cG9ydCBjbGFzcyBGb2N1c1RyYXAgZXh0ZW5kcyBEZXN0cnVjdGlibGUge1xuICAgICNtYW5hZ2VyID0gaW5qZWN0KEZvY3VzU2VydmljZSlcbiAgICAjZWwgPSBpbmplY3QoRWxlbWVudFJlZjxIVE1MRWxlbWVudD4pXG5cbiAgICBnZXQgZW5hYmxlZCgpIHtcbiAgICAgICAgcmV0dXJuIHRoaXMuI2VuYWJsZWQudmFsdWVcbiAgICB9XG4gICAgc2V0IGVuYWJsZWQodmFsOiBib29sZWFuKSB7XG4gICAgICAgIGlmICh0aGlzLiNlbmFibGVkLnZhbHVlICE9PSB2YWwpIHtcbiAgICAgICAgICAgIHRoaXMuI2VuYWJsZWQubmV4dCh2YWwpXG4gICAgICAgIH1cbiAgICB9XG4gICAgI2VuYWJsZWQgPSBuZXcgQmVoYXZpb3JTdWJqZWN0PGJvb2xlYW4+KHRydWUpXG5cbiAgICBlbmFibGUoKSB7XG4gICAgICAgIHRoaXMuZW5hYmxlZCA9IHRydWVcbiAgICB9XG5cbiAgICBkaXNhYmxlKCkge1xuICAgICAgICB0aGlzLmVuYWJsZWQgPSBmYWxzZVxuICAgIH1cbn1cbiJdfQ==
|
|
@@ -1,145 +0,0 @@
|
|
|
1
|
-
import { FocusTrapFactory } from "@angular/cdk/a11y";
|
|
2
|
-
import { DOCUMENT } from "@angular/common";
|
|
3
|
-
import { inject, Injectable, NgZone } from "@angular/core";
|
|
4
|
-
import { takeUntilDestroyed } from "@angular/core/rxjs-interop";
|
|
5
|
-
import { BehaviorSubject, combineLatest, connect, distinctUntilChanged, filter, finalize, map, merge, Observable, shareReplay, startWith, Subject, tap } from "rxjs";
|
|
6
|
-
import { isEqual } from "lodash-es";
|
|
7
|
-
import { focusable, isFocusable } from "tabbable";
|
|
8
|
-
import { coerceElement } from "@ngutil/common";
|
|
9
|
-
import { ActivityService } from "../activity";
|
|
10
|
-
import * as i0 from "@angular/core";
|
|
11
|
-
const EVENT_OPTIONS = {
|
|
12
|
-
capture: true,
|
|
13
|
-
passive: true
|
|
14
|
-
};
|
|
15
|
-
export class FocusService {
|
|
16
|
-
constructor() {
|
|
17
|
-
this.#activity = inject(ActivityService);
|
|
18
|
-
this.#focusTrap = inject(FocusTrapFactory);
|
|
19
|
-
this.#zone = inject(NgZone);
|
|
20
|
-
this.#document = inject(DOCUMENT);
|
|
21
|
-
this.#originOverrides = new BehaviorSubject(new Map());
|
|
22
|
-
this.#focus = listener(this.#document, this.#zone, "focus", EVENT_OPTIONS).pipe(connect(focus => this.#zone.runOutsideAngular(() => {
|
|
23
|
-
let lastFocused = null;
|
|
24
|
-
const sideEffect = new Subject();
|
|
25
|
-
// if element removed form document, emit blur & focus
|
|
26
|
-
const mutation = new MutationObserver(mutations => {
|
|
27
|
-
for (const mutation of mutations) {
|
|
28
|
-
if (mutation.type === "childList" && mutation.removedNodes.length > 0) {
|
|
29
|
-
for (const removed of Array.from(mutation.removedNodes)) {
|
|
30
|
-
if (removed === lastFocused || removed.contains(lastFocused)) {
|
|
31
|
-
this.#blurSide.next(lastFocused);
|
|
32
|
-
sideEffect.next(this.#document.activeElement);
|
|
33
|
-
return;
|
|
34
|
-
}
|
|
35
|
-
}
|
|
36
|
-
}
|
|
37
|
-
}
|
|
38
|
-
});
|
|
39
|
-
mutation.observe(this.#document, { subtree: true, childList: true });
|
|
40
|
-
return merge(focus.pipe(tap(node => (lastFocused = node)), finalize(() => mutation.disconnect())), sideEffect);
|
|
41
|
-
})), distinctUntilChanged(strictEq));
|
|
42
|
-
this.#blurEvent = listener(this.#document, this.#zone, "blur", EVENT_OPTIONS).pipe(startWith(null));
|
|
43
|
-
this.#blurSide = new Subject();
|
|
44
|
-
this.#blur = merge(this.#blurEvent, this.#blurSide).pipe(distinctUntilChanged(strictEq));
|
|
45
|
-
this.events = this.#zone.runOutsideAngular(() => combineLatest({
|
|
46
|
-
activity: this.#activity.events$.pipe(filter(event => event.type !== "mousemove")),
|
|
47
|
-
focus: this.#focus,
|
|
48
|
-
blur: this.#blur.pipe(tap(el => this.#delOrigin(el))),
|
|
49
|
-
overrides: this.#originOverrides
|
|
50
|
-
}).pipe(takeUntilDestroyed(), map(({ activity, focus, blur, overrides }) => {
|
|
51
|
-
const override = overrides.get(focus);
|
|
52
|
-
// If focus in with alt+tab
|
|
53
|
-
if (blur === document && activity.origin === "keyboard") {
|
|
54
|
-
return { origin: override || "program", element: focus };
|
|
55
|
-
}
|
|
56
|
-
if (focus === document) {
|
|
57
|
-
return { origin: override || "program", element: focus };
|
|
58
|
-
}
|
|
59
|
-
// If press tab button, first fire the event in the currently focused element
|
|
60
|
-
if (focus === blur) {
|
|
61
|
-
return null;
|
|
62
|
-
}
|
|
63
|
-
// When press tab, the activity is on the current fucesd element,
|
|
64
|
-
// so when blur is changed to it, the focus change is completed
|
|
65
|
-
if (activity.origin === "keyboard" && isActivityElement(activity.node, blur)) {
|
|
66
|
-
return { origin: override || "keyboard", element: focus };
|
|
67
|
-
}
|
|
68
|
-
if (isActivityElement(activity.node, focus)) {
|
|
69
|
-
return { origin: override || activity.origin, element: focus };
|
|
70
|
-
}
|
|
71
|
-
else {
|
|
72
|
-
return { origin: override || "program", element: focus };
|
|
73
|
-
}
|
|
74
|
-
}), filter(v => !!v), distinctUntilChanged(isEqual), shareReplay({ bufferSize: 1, refCount: true })));
|
|
75
|
-
}
|
|
76
|
-
#activity;
|
|
77
|
-
#focusTrap;
|
|
78
|
-
#zone;
|
|
79
|
-
#document;
|
|
80
|
-
#originOverrides;
|
|
81
|
-
#focus;
|
|
82
|
-
#blurEvent;
|
|
83
|
-
#blurSide;
|
|
84
|
-
#blur;
|
|
85
|
-
watch(element) {
|
|
86
|
-
const el = coerceElement(element);
|
|
87
|
-
return this.events.pipe(map(event => {
|
|
88
|
-
if (event.element &&
|
|
89
|
-
(event.element === el || (typeof el.contains === "function" && el.contains(event.element)))) {
|
|
90
|
-
return event;
|
|
91
|
-
}
|
|
92
|
-
return { element: el, origin: null };
|
|
93
|
-
}));
|
|
94
|
-
}
|
|
95
|
-
focus(node, origin) {
|
|
96
|
-
this.#setOrigin(node, origin);
|
|
97
|
-
coerceElement(node).focus();
|
|
98
|
-
}
|
|
99
|
-
queryFocusable(inside) {
|
|
100
|
-
return focusable(coerceElement(inside), { includeContainer: false });
|
|
101
|
-
}
|
|
102
|
-
getFirstFocusable(inside) {
|
|
103
|
-
return this.queryFocusable(inside)[0];
|
|
104
|
-
}
|
|
105
|
-
isFocusable(node) {
|
|
106
|
-
return isFocusable(coerceElement(node));
|
|
107
|
-
}
|
|
108
|
-
focusTrap(inside, deferCaptureElements = false) {
|
|
109
|
-
return this.#focusTrap.create(coerceElement(inside), deferCaptureElements);
|
|
110
|
-
}
|
|
111
|
-
#setOrigin(el, origin) {
|
|
112
|
-
const target = coerceElement(el);
|
|
113
|
-
const map = this.#originOverrides.value;
|
|
114
|
-
map.set(target, origin);
|
|
115
|
-
this.#originOverrides.next(map);
|
|
116
|
-
}
|
|
117
|
-
#delOrigin(el) {
|
|
118
|
-
const target = coerceElement(el);
|
|
119
|
-
const map = this.#originOverrides.value;
|
|
120
|
-
map.delete(target);
|
|
121
|
-
this.#originOverrides.next(map);
|
|
122
|
-
}
|
|
123
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.6", ngImport: i0, type: FocusService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
124
|
-
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.6", ngImport: i0, type: FocusService, providedIn: "root" }); }
|
|
125
|
-
}
|
|
126
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.6", ngImport: i0, type: FocusService, decorators: [{
|
|
127
|
-
type: Injectable,
|
|
128
|
-
args: [{ providedIn: "root" }]
|
|
129
|
-
}] });
|
|
130
|
-
function isActivityElement(activityEl, focused) {
|
|
131
|
-
return activityEl != null && focused != null && (activityEl === focused || focused.contains(activityEl));
|
|
132
|
-
}
|
|
133
|
-
function listener(doc, zone, type, options) {
|
|
134
|
-
return new Observable((dst) => zone.runOutsideAngular(() => {
|
|
135
|
-
const handler = (e) => {
|
|
136
|
-
dst.next(e.target);
|
|
137
|
-
};
|
|
138
|
-
document.addEventListener(type, handler, options);
|
|
139
|
-
return () => document.removeEventListener(type, handler, options);
|
|
140
|
-
}));
|
|
141
|
-
}
|
|
142
|
-
function strictEq(a, b) {
|
|
143
|
-
return a === b;
|
|
144
|
-
}
|
|
145
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"focus.service.js","sourceRoot":"","sources":["../../../../../packages/aria/src/focus/focus.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { FocusTrapFactory } from \"@angular/cdk/a11y\"\nimport { DOCUMENT } from \"@angular/common\"\nimport { inject, Injectable, NgZone } from \"@angular/core\"\nimport { takeUntilDestroyed } from \"@angular/core/rxjs-interop\"\n\nimport {\n    BehaviorSubject,\n    combineLatest,\n    connect,\n    distinctUntilChanged,\n    filter,\n    finalize,\n    map,\n    merge,\n    Observable,\n    shareReplay,\n    startWith,\n    Subject,\n    Subscriber,\n    tap\n} from \"rxjs\"\n\nimport { isEqual } from \"lodash-es\"\nimport { focusable, type FocusableElement, isFocusable } from \"tabbable\"\n\nimport { coerceElement, ElementInput } from \"@ngutil/common\"\n\nimport { ActivityOrigin, ActivityService } from \"../activity\"\n\nconst EVENT_OPTIONS: AddEventListenerOptions = {\n    capture: true,\n    passive: true\n}\n\nexport type FocusOrigin = ActivityOrigin | null\n\nexport interface FocusChanges {\n    origin: FocusOrigin\n    element: HTMLElement\n}\n\nexport interface FocusableEvent {\n    origin: FocusOrigin\n    exact: boolean\n    node: Node\n}\n\n@Injectable({ providedIn: \"root\" })\nexport class FocusService {\n    readonly #activity = inject(ActivityService)\n    readonly #focusTrap = inject(FocusTrapFactory)\n    readonly #zone = inject(NgZone)\n    readonly #document = inject(DOCUMENT)\n\n    readonly #originOverrides = new BehaviorSubject<Map<HTMLElement, FocusOrigin>>(new Map())\n\n    readonly #focus = listener(this.#document, this.#zone, \"focus\", EVENT_OPTIONS).pipe(\n        connect(focus =>\n            this.#zone.runOutsideAngular(() => {\n                let lastFocused: Node | null = null\n                const sideEffect = new Subject<Node>()\n\n                // if element removed form document, emit blur & focus\n                const mutation = new MutationObserver(mutations => {\n                    for (const mutation of mutations) {\n                        if (mutation.type === \"childList\" && mutation.removedNodes.length > 0) {\n                            for (const removed of Array.from(mutation.removedNodes)) {\n                                if (removed === lastFocused || removed.contains(lastFocused)) {\n                                    this.#blurSide.next(lastFocused!)\n                                    sideEffect.next(this.#document.activeElement!)\n                                    return\n                                }\n                            }\n                        }\n                    }\n                })\n\n                mutation.observe(this.#document, { subtree: true, childList: true })\n                return merge(\n                    focus.pipe(\n                        tap(node => (lastFocused = node)),\n                        finalize(() => mutation.disconnect())\n                    ),\n                    sideEffect\n                )\n            })\n        ),\n\n        distinctUntilChanged(strictEq)\n    )\n\n    readonly #blurEvent = listener(this.#document, this.#zone, \"blur\", EVENT_OPTIONS).pipe(startWith(null))\n    readonly #blurSide = new Subject<Node>()\n    readonly #blur = merge(this.#blurEvent, this.#blurSide).pipe(distinctUntilChanged(strictEq))\n\n    readonly events: Observable<FocusChanges> = this.#zone.runOutsideAngular(() =>\n        combineLatest({\n            activity: this.#activity.events$.pipe(filter(event => event.type !== \"mousemove\")),\n            focus: this.#focus,\n            blur: this.#blur.pipe(tap(el => this.#delOrigin(el as any))),\n            overrides: this.#originOverrides\n        }).pipe(\n            takeUntilDestroyed(),\n            map(({ activity, focus, blur, overrides }) => {\n                const override = overrides.get(focus as any)\n\n                // If focus in with alt+tab\n                if (blur === document && activity.origin === \"keyboard\") {\n                    return { origin: override || \"program\", element: focus } satisfies FocusChanges\n                }\n\n                if (focus === document) {\n                    return { origin: override || \"program\", element: focus } satisfies FocusChanges\n                }\n\n                // If press tab button, first fire the event in the currently focused element\n                if (focus === blur) {\n                    return null\n                }\n\n                // When press tab, the activity is on the current fucesd element,\n                // so when blur is changed to it, the focus change is completed\n                if (activity.origin === \"keyboard\" && isActivityElement(activity.node, blur)) {\n                    return { origin: override || \"keyboard\", element: focus } satisfies FocusChanges\n                }\n\n                if (isActivityElement(activity.node, focus)) {\n                    return { origin: override || activity.origin, element: focus }\n                } else {\n                    return { origin: override || \"program\", element: focus } satisfies FocusChanges\n                }\n            }),\n            filter(v => !!v),\n            distinctUntilChanged(isEqual),\n            shareReplay({ bufferSize: 1, refCount: true })\n        )\n    )\n\n    watch(element: ElementInput) {\n        const el = coerceElement(element)\n        return this.events.pipe(\n            map(event => {\n                if (\n                    event.element &&\n                    (event.element === el || (typeof el.contains === \"function\" && el.contains(event.element)))\n                ) {\n                    return event\n                }\n                return { element: el, origin: null } as FocusChanges\n            })\n        )\n    }\n\n    focus(node: ElementInput, origin: FocusOrigin | null) {\n        this.#setOrigin(node, origin)\n        coerceElement(node).focus()\n    }\n\n    queryFocusable(inside: ElementInput): FocusableElement[] {\n        return focusable(coerceElement(inside), { includeContainer: false })\n    }\n\n    getFirstFocusable(inside: ElementInput): FocusableElement | undefined {\n        return this.queryFocusable(inside)[0]\n    }\n\n    isFocusable(node: ElementInput): boolean {\n        return isFocusable(coerceElement(node))\n    }\n\n    focusTrap(inside: ElementInput, deferCaptureElements: boolean = false) {\n        return this.#focusTrap.create(coerceElement(inside), deferCaptureElements)\n    }\n\n    #setOrigin(el: ElementInput, origin: FocusOrigin) {\n        const target = coerceElement(el)\n        const map = this.#originOverrides.value\n        map.set(target, origin)\n        this.#originOverrides.next(map)\n    }\n\n    #delOrigin(el: ElementInput) {\n        const target = coerceElement(el)\n        const map = this.#originOverrides.value\n        map.delete(target)\n        this.#originOverrides.next(map)\n    }\n}\n\nfunction isActivityElement(activityEl?: Node | null, focused?: Node | null): boolean {\n    return activityEl != null && focused != null && (activityEl === focused || focused.contains(activityEl))\n}\n\nfunction listener(doc: Document, zone: NgZone, type: Event[\"type\"], options: AddEventListenerOptions) {\n    return new Observable((dst: Subscriber<Node>) =>\n        zone.runOutsideAngular(() => {\n            const handler = (e: Event) => {\n                dst.next(e.target as Node)\n            }\n            document.addEventListener(type, handler, options)\n            return () => document.removeEventListener(type, handler, options)\n        })\n    )\n}\n\nfunction strictEq(a: any, b: any): boolean {\n    return a === b\n}\n"]}
|
|
@@ -1,52 +0,0 @@
|
|
|
1
|
-
import { computed, Directive, effect, inject, input } from "@angular/core";
|
|
2
|
-
import { isElementInput } from "@ngutil/common";
|
|
3
|
-
import { FocusState } from "./focus-state.directive";
|
|
4
|
-
import * as i0 from "@angular/core";
|
|
5
|
-
import * as i1 from "./focus-state.directive";
|
|
6
|
-
// TODO: what happens when disabled is changed
|
|
7
|
-
export class Focusable {
|
|
8
|
-
constructor() {
|
|
9
|
-
this.state = inject(FocusState);
|
|
10
|
-
this.focusable = input(true, { alias: "nuFocusable" });
|
|
11
|
-
this.tabindex = input(0, { transform: Number });
|
|
12
|
-
this._tabindex = computed(() => {
|
|
13
|
-
const focusable = this.focusable();
|
|
14
|
-
const tabindex = this.tabindex();
|
|
15
|
-
if (focusable === false) {
|
|
16
|
-
return -1;
|
|
17
|
-
}
|
|
18
|
-
if (typeof focusable === "number") {
|
|
19
|
-
return focusable;
|
|
20
|
-
}
|
|
21
|
-
if (focusable === true && tabindex != null && !isNaN(tabindex)) {
|
|
22
|
-
return tabindex;
|
|
23
|
-
}
|
|
24
|
-
return 0;
|
|
25
|
-
});
|
|
26
|
-
// TODO: miért kell ez?, ha nincs itt akkor nem frissül
|
|
27
|
-
effect(() => this._tabindex(), { allowSignalWrites: false });
|
|
28
|
-
}
|
|
29
|
-
connect(value) {
|
|
30
|
-
if (value instanceof FocusState || isElementInput(value)) {
|
|
31
|
-
return this.state.connect(value);
|
|
32
|
-
}
|
|
33
|
-
else {
|
|
34
|
-
return this.state.connect(value.state);
|
|
35
|
-
}
|
|
36
|
-
}
|
|
37
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.6", ngImport: i0, type: Focusable, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
38
|
-
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.2.6", type: Focusable, isStandalone: true, selector: "[nuFocusable]", inputs: { focusable: { classPropertyName: "focusable", publicName: "nuFocusable", isSignal: true, isRequired: false, transformFunction: null }, tabindex: { classPropertyName: "tabindex", publicName: "tabindex", isSignal: true, isRequired: false, transformFunction: null } }, host: { properties: { "attr.tabindex": "_tabindex()" } }, exportAs: ["nuFocusable"], hostDirectives: [{ directive: i1.FocusState }], ngImport: i0 }); }
|
|
39
|
-
}
|
|
40
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.6", ngImport: i0, type: Focusable, decorators: [{
|
|
41
|
-
type: Directive,
|
|
42
|
-
args: [{
|
|
43
|
-
standalone: true,
|
|
44
|
-
selector: "[nuFocusable]",
|
|
45
|
-
exportAs: "nuFocusable",
|
|
46
|
-
host: {
|
|
47
|
-
"[attr.tabindex]": "_tabindex()"
|
|
48
|
-
},
|
|
49
|
-
hostDirectives: [FocusState]
|
|
50
|
-
}]
|
|
51
|
-
}], ctorParameters: () => [] });
|
|
52
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9jdXNhYmxlLmRpcmVjdGl2ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL2FyaWEvc3JjL2ZvY3VzL2ZvY3VzYWJsZS5kaXJlY3RpdmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFFBQVEsRUFBRSxTQUFTLEVBQUUsTUFBTSxFQUFFLE1BQU0sRUFBRSxLQUFLLEVBQUUsTUFBTSxlQUFlLENBQUE7QUFFMUUsT0FBTyxFQUFpQyxjQUFjLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQTtBQUU5RSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0seUJBQXlCLENBQUE7OztBQUVwRCw4Q0FBOEM7QUFXOUMsTUFBTSxPQUFPLFNBQVM7SUF5QmxCO1FBeEJTLFVBQUssR0FBRyxNQUFNLENBQUMsVUFBVSxDQUFDLENBQUE7UUFFMUIsY0FBUyxHQUFHLEtBQUssQ0FBbUIsSUFBSSxFQUFFLEVBQUUsS0FBSyxFQUFFLGFBQWEsRUFBRSxDQUFDLENBQUE7UUFDbkUsYUFBUSxHQUFHLEtBQUssQ0FBQyxDQUFDLEVBQUUsRUFBRSxTQUFTLEVBQUUsTUFBTSxFQUFFLENBQUMsQ0FBQTtRQUUxQyxjQUFTLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRTtZQUMvQixNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsU0FBUyxFQUFFLENBQUE7WUFDbEMsTUFBTSxRQUFRLEdBQUcsSUFBSSxDQUFDLFFBQVEsRUFBRSxDQUFBO1lBRWhDLElBQUksU0FBUyxLQUFLLEtBQUssRUFBRSxDQUFDO2dCQUN0QixPQUFPLENBQUMsQ0FBQyxDQUFBO1lBQ2IsQ0FBQztZQUVELElBQUksT0FBTyxTQUFTLEtBQUssUUFBUSxFQUFFLENBQUM7Z0JBQ2hDLE9BQU8sU0FBUyxDQUFBO1lBQ3BCLENBQUM7WUFFRCxJQUFJLFNBQVMsS0FBSyxJQUFJLElBQUksUUFBUSxJQUFJLElBQUksSUFBSSxDQUFDLEtBQUssQ0FBQyxRQUFRLENBQUMsRUFBRSxDQUFDO2dCQUM3RCxPQUFPLFFBQVEsQ0FBQTtZQUNuQixDQUFDO1lBRUQsT0FBTyxDQUFDLENBQUE7UUFDWixDQUFDLENBQUMsQ0FBQTtRQUdFLHVEQUF1RDtRQUN2RCxNQUFNLENBQUMsR0FBRyxFQUFFLENBQUMsSUFBSSxDQUFDLFNBQVMsRUFBRSxFQUFFLEVBQUUsaUJBQWlCLEVBQUUsS0FBSyxFQUFFLENBQUMsQ0FBQTtJQUNoRSxDQUFDO0lBRUQsT0FBTyxDQUFDLEtBQTRDO1FBQ2hELElBQUksS0FBSyxZQUFZLFVBQVUsSUFBSSxjQUFjLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQztZQUN2RCxPQUFPLElBQUksQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLEtBQUssQ0FBQyxDQUFBO1FBQ3BDLENBQUM7YUFBTSxDQUFDO1lBQ0osT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUMsS0FBSyxDQUFDLENBQUE7UUFDMUMsQ0FBQztJQUNMLENBQUM7OEdBcENRLFNBQVM7a0dBQVQsU0FBUzs7MkZBQVQsU0FBUztrQkFUckIsU0FBUzttQkFBQztvQkFDUCxVQUFVLEVBQUUsSUFBSTtvQkFDaEIsUUFBUSxFQUFFLGVBQWU7b0JBQ3pCLFFBQVEsRUFBRSxhQUFhO29CQUN2QixJQUFJLEVBQUU7d0JBQ0YsaUJBQWlCLEVBQUUsYUFBYTtxQkFDbkM7b0JBQ0QsY0FBYyxFQUFFLENBQUMsVUFBVSxDQUFDO2lCQUMvQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGNvbXB1dGVkLCBEaXJlY3RpdmUsIGVmZmVjdCwgaW5qZWN0LCBpbnB1dCB9IGZyb20gXCJAYW5ndWxhci9jb3JlXCJcblxuaW1wb3J0IHsgQ29ubmVjdFByb3RvY29sLCBFbGVtZW50SW5wdXQsIGlzRWxlbWVudElucHV0IH0gZnJvbSBcIkBuZ3V0aWwvY29tbW9uXCJcblxuaW1wb3J0IHsgRm9jdXNTdGF0ZSB9IGZyb20gXCIuL2ZvY3VzLXN0YXRlLmRpcmVjdGl2ZVwiXG5cbi8vIFRPRE86IHdoYXQgaGFwcGVucyB3aGVuIGRpc2FibGVkIGlzIGNoYW5nZWRcblxuQERpcmVjdGl2ZSh7XG4gICAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgICBzZWxlY3RvcjogXCJbbnVGb2N1c2FibGVdXCIsXG4gICAgZXhwb3J0QXM6IFwibnVGb2N1c2FibGVcIixcbiAgICBob3N0OiB7XG4gICAgICAgIFwiW2F0dHIudGFiaW5kZXhdXCI6IFwiX3RhYmluZGV4KClcIlxuICAgIH0sXG4gICAgaG9zdERpcmVjdGl2ZXM6IFtGb2N1c1N0YXRlXVxufSlcbmV4cG9ydCBjbGFzcyBGb2N1c2FibGUgaW1wbGVtZW50cyBDb25uZWN0UHJvdG9jb2wge1xuICAgIHJlYWRvbmx5IHN0YXRlID0gaW5qZWN0KEZvY3VzU3RhdGUpXG5cbiAgICByZWFkb25seSBmb2N1c2FibGUgPSBpbnB1dDxib29sZWFuIHwgbnVtYmVyPih0cnVlLCB7IGFsaWFzOiBcIm51Rm9jdXNhYmxlXCIgfSlcbiAgICByZWFkb25seSB0YWJpbmRleCA9IGlucHV0KDAsIHsgdHJhbnNmb3JtOiBOdW1iZXIgfSlcblxuICAgIHJlYWRvbmx5IF90YWJpbmRleCA9IGNvbXB1dGVkKCgpID0+IHtcbiAgICAgICAgY29uc3QgZm9jdXNhYmxlID0gdGhpcy5mb2N1c2FibGUoKVxuICAgICAgICBjb25zdCB0YWJpbmRleCA9IHRoaXMudGFiaW5kZXgoKVxuXG4gICAgICAgIGlmIChmb2N1c2FibGUgPT09IGZhbHNlKSB7XG4gICAgICAgICAgICByZXR1cm4gLTFcbiAgICAgICAgfVxuXG4gICAgICAgIGlmICh0eXBlb2YgZm9jdXNhYmxlID09PSBcIm51bWJlclwiKSB7XG4gICAgICAgICAgICByZXR1cm4gZm9jdXNhYmxlXG4gICAgICAgIH1cblxuICAgICAgICBpZiAoZm9jdXNhYmxlID09PSB0cnVlICYmIHRhYmluZGV4ICE9IG51bGwgJiYgIWlzTmFOKHRhYmluZGV4KSkge1xuICAgICAgICAgICAgcmV0dXJuIHRhYmluZGV4XG4gICAgICAgIH1cblxuICAgICAgICByZXR1cm4gMFxuICAgIH0pXG5cbiAgICBjb25zdHJ1Y3RvcigpIHtcbiAgICAgICAgLy8gVE9ETzogbWnDqXJ0IGtlbGwgZXo/LCBoYSBuaW5jcyBpdHQgYWtrb3IgbmVtIGZyaXNzw7xsXG4gICAgICAgIGVmZmVjdCgoKSA9PiB0aGlzLl90YWJpbmRleCgpLCB7IGFsbG93U2lnbmFsV3JpdGVzOiBmYWxzZSB9KVxuICAgIH1cblxuICAgIGNvbm5lY3QodmFsdWU6IEZvY3VzYWJsZSB8IEZvY3VzU3RhdGUgfCBFbGVtZW50SW5wdXQpIHtcbiAgICAgICAgaWYgKHZhbHVlIGluc3RhbmNlb2YgRm9jdXNTdGF0ZSB8fCBpc0VsZW1lbnRJbnB1dCh2YWx1ZSkpIHtcbiAgICAgICAgICAgIHJldHVybiB0aGlzLnN0YXRlLmNvbm5lY3QodmFsdWUpXG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICByZXR1cm4gdGhpcy5zdGF0ZS5jb25uZWN0KHZhbHVlLnN0YXRlKVxuICAgICAgICB9XG4gICAgfVxufVxuIl19
|
package/esm2022/focus/index.mjs
DELETED
|
@@ -1,5 +0,0 @@
|
|
|
1
|
-
export * from "./focus.service";
|
|
2
|
-
export * from "./focus-trap.directive";
|
|
3
|
-
export * from "./focusable.directive";
|
|
4
|
-
export * from "./focus-state.directive";
|
|
5
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9hcmlhL3NyYy9mb2N1cy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLGlCQUFpQixDQUFBO0FBQy9CLGNBQWMsd0JBQXdCLENBQUE7QUFDdEMsY0FBYyx1QkFBdUIsQ0FBQTtBQUNyQyxjQUFjLHlCQUF5QixDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSBcIi4vZm9jdXMuc2VydmljZVwiXG5leHBvcnQgKiBmcm9tIFwiLi9mb2N1cy10cmFwLmRpcmVjdGl2ZVwiXG5leHBvcnQgKiBmcm9tIFwiLi9mb2N1c2FibGUuZGlyZWN0aXZlXCJcbmV4cG9ydCAqIGZyb20gXCIuL2ZvY3VzLXN0YXRlLmRpcmVjdGl2ZVwiXG4iXX0=
|