@ngutil/floating 0.0.51 → 0.0.53

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.
@@ -1,5 +1,5 @@
1
1
  import { Inject, Injectable, InjectionToken } from "@angular/core";
2
- import { debounceTime, EMPTY, filter, map, merge, ReplaySubject, shareReplay, take, takeUntil, takeWhile } from "rxjs";
2
+ import { combineLatest, debounceTime, EMPTY, filter, map, ReplaySubject, shareReplay, startWith, take, takeUntil, takeWhile } from "rxjs";
3
3
  import { Lifecycle } from "@ngutil/common";
4
4
  import { ContainerRef } from "../layer/container-ref";
5
5
  import { LayerService } from "../layer/layer.service";
@@ -7,6 +7,7 @@ import * as i0 from "@angular/core";
7
7
  import * as i1 from "../layer/layer.service";
8
8
  import * as i2 from "../layer/container-ref";
9
9
  export const TRAITS = new InjectionToken("TRAITS");
10
+ let UID_COUNTER = 0;
10
11
  export class FloatingRef {
11
12
  #traits;
12
13
  #untilCleanup;
@@ -27,8 +28,13 @@ export class FloatingRef {
27
28
  this.#traits = {};
28
29
  this.#untilCleanup = this.state.onExecute("cleanup");
29
30
  this.#untilDisposed = this.state.onExecute("disposed");
30
- container.nativeElement.style.overflow = "hidden";
31
- container.nativeElement.style.visibility = "hidden";
31
+ this.uid = `${++UID_COUNTER}`;
32
+ Object.assign(container.nativeElement.style, {
33
+ overflow: "hidden",
34
+ visibility: "hidden",
35
+ pointerEvents: "none"
36
+ });
37
+ container.nativeElement.setAttribute("data-floating", this.uid);
32
38
  this.#traits = traits;
33
39
  this.traitState$ = this.#traitState().pipe(shareReplay(1));
34
40
  this.state.current$.pipe(takeWhile(state => state !== "cleanup", true)).subscribe(state => {
@@ -38,6 +44,9 @@ export class FloatingRef {
38
44
  this.state.on("showing", () => {
39
45
  container.nativeElement.style.visibility = "visible";
40
46
  });
47
+ this.state.on("shown", () => {
48
+ container.nativeElement.style.pointerEvents = null;
49
+ });
41
50
  this.state.on("disposing", () => {
42
51
  container.nativeElement.style.pointerEvents = "none";
43
52
  });
@@ -72,23 +81,18 @@ export class FloatingRef {
72
81
  this.close(true).subscribe();
73
82
  }
74
83
  watchTrait(name) {
75
- return this.traitState$.pipe(takeUntil(this.#untilDisposed), filter(event => event.name === name), map(event => event.data), shareReplay(1));
84
+ return this.traitState$.pipe(takeUntil(this.#untilDisposed), map(state => state[name]), filter(value => value != null));
76
85
  }
77
86
  #traitState() {
78
- const src = [];
87
+ const src = {};
79
88
  for (const [k, v] of Object.entries(this.#traits)) {
80
- src.push(v.connect(this).pipe(takeUntil(this.#untilCleanup), map(result => {
81
- return { name: k, data: result };
82
- })));
89
+ src[k] = v.connect(this).pipe(takeUntil(this.#untilDisposed), startWith(null));
83
90
  }
84
- if (src.length === 0) {
91
+ if (Object.keys(src).length === 0) {
85
92
  return EMPTY;
86
93
  }
87
- else if (src.length === 1) {
88
- return src[0];
89
- }
90
94
  else {
91
- return merge(...src);
95
+ return combineLatest(src).pipe(shareReplay(1));
92
96
  }
93
97
  }
94
98
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.6", ngImport: i0, type: FloatingRef, deps: [{ token: i1.LayerService }, { token: i2.ContainerRef }, { token: TRAITS }], target: i0.ɵɵFactoryTarget.Injectable }); }
@@ -100,4 +104,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.6", ngImpor
100
104
  type: Inject,
101
105
  args: [TRAITS]
102
106
  }] }] });
103
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"floating-ref.js","sourceRoot":"","sources":["../../../../../packages/floating/src/floating/floating-ref.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable, InjectionToken } from \"@angular/core\"\n\nimport {\n    debounceTime,\n    EMPTY,\n    filter,\n    map,\n    merge,\n    Observable,\n    ReplaySubject,\n    shareReplay,\n    take,\n    takeUntil,\n    takeWhile\n} from \"rxjs\"\n\nimport { Lifecycle } from \"@ngutil/common\"\n\nimport { ContainerRef } from \"../layer/container-ref\"\nimport { LayerService } from \"../layer/layer.service\"\nimport { FloatingTrait } from \"./traits/_base\"\n\nexport type Traits = { [key: string]: FloatingTrait }\n\nexport const TRAITS = new InjectionToken<Traits>(\"TRAITS\")\n\nexport interface FloatingChannel {\n    floatingRef: FloatingRef\n    type: string\n    data?: any\n}\n\nexport interface FloatingTraitEvent {\n    name: string\n    data: object\n}\n\n@Injectable()\nexport class FloatingRef<C extends FloatingChannel = FloatingChannel, T extends HTMLElement = HTMLElement> {\n    readonly channel = new ReplaySubject<FloatingChannel>(1)\n\n    readonly state = new Lifecycle({\n        init: {},\n        showing: {},\n        shown: {},\n        closing: { cancellable: false, order: \"sequential\" },\n        disposing: { cancellable: false },\n        disposed: { cancellable: false, order: \"sequential\" },\n        cleanup: { cancellable: false, order: \"sequential\" }\n    })\n\n    readonly #traits: Traits = {}\n    readonly traitState$: Observable<FloatingTraitEvent>\n\n    readonly #untilCleanup = this.state.onExecute(\"cleanup\")\n    readonly #untilDisposed = this.state.onExecute(\"disposed\")\n\n    constructor(\n        readonly layerSvc: LayerService,\n        readonly container: ContainerRef,\n        @Inject(TRAITS) traits: Traits\n    ) {\n        container.nativeElement.style.overflow = \"hidden\"\n        container.nativeElement.style.visibility = \"hidden\"\n\n        this.#traits = traits\n        this.traitState$ = this.#traitState().pipe(shareReplay(1))\n\n        this.state.current$.pipe(takeWhile(state => state !== \"cleanup\", true)).subscribe(state => {\n            this.emit({ type: state } as C)\n        })\n        this.state.on(\"init\", () => this.traitState$.pipe(takeUntil(this.#untilCleanup), debounceTime(5), take(1)))\n        this.state.on(\"showing\", () => {\n            container.nativeElement.style.visibility = \"visible\"\n        })\n        this.state.on(\"disposing\", () => {\n            container.nativeElement.style.pointerEvents = \"none\"\n        })\n\n        this.state.control(container.state)\n\n        // TODO: remove\n        // this.state.status$.subscribe(status => {\n        //     console.log(\"floating\", status)\n        // })\n    }\n\n    show() {\n        return this.state.run(\"init\", \"showing\", \"shown\")\n    }\n\n    /**\n     * @deprecated\n     */\n    hide() {\n        return this.close(true)\n    }\n\n    close(force = false) {\n        if (force) {\n            return this.state.run(\"disposing\", \"disposed\", \"cleanup\")\n        } else {\n            return this.state.run(\"closing\", \"disposing\", \"disposed\", \"cleanup\")\n        }\n    }\n\n    emit(event: Omit<C, \"floatingRef\">) {\n        this.channel.next({ ...event, floatingRef: this } as any)\n    }\n\n    setResult(data: any) {\n        this.emit({ type: \"result\", data } as C)\n        this.close(true).subscribe()\n    }\n\n    watchTrait<T>(name: string): Observable<T> {\n        return this.traitState$.pipe(\n            takeUntil(this.#untilDisposed),\n            filter(event => event.name === name),\n            map(event => event.data as T),\n            shareReplay(1)\n        )\n    }\n\n    #traitState(): Observable<FloatingTraitEvent> {\n        const src = []\n\n        for (const [k, v] of Object.entries(this.#traits)) {\n            src.push(\n                v.connect(this).pipe(\n                    takeUntil(this.#untilCleanup),\n                    map(result => {\n                        return { name: k, data: result }\n                    })\n                )\n            )\n        }\n\n        if (src.length === 0) {\n            return EMPTY\n        } else if (src.length === 1) {\n            return src[0]\n        } else {\n            return merge(...src)\n        }\n    }\n}\n"]}
107
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"floating-ref.js","sourceRoot":"","sources":["../../../../../packages/floating/src/floating/floating-ref.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,UAAU,EAAE,cAAc,EAAE,MAAM,eAAe,CAAA;AAElE,OAAO,EACH,aAAa,EACb,YAAY,EACZ,KAAK,EACL,MAAM,EACN,GAAG,EAEH,aAAa,EACb,WAAW,EACX,SAAS,EACT,IAAI,EACJ,SAAS,EACT,SAAS,EACZ,MAAM,MAAM,CAAA;AAEb,OAAO,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAA;AAE1C,OAAO,EAAE,YAAY,EAAE,MAAM,wBAAwB,CAAA;AACrD,OAAO,EAAE,YAAY,EAAE,MAAM,wBAAwB,CAAA;;;;AAKrD,MAAM,CAAC,MAAM,MAAM,GAAG,IAAI,cAAc,CAAS,QAAQ,CAAC,CAAA;AAe1D,IAAI,WAAW,GAAG,CAAC,CAAA;AAGnB,MAAM,OAAO,WAAW;IAaX,OAAO,CAAa;IAGpB,aAAa,CAAkC;IAC/C,cAAc,CAAmC;IAI1D,YACa,QAAsB,EACtB,SAAuB,EAChB,MAAc;QAFrB,aAAQ,GAAR,QAAQ,CAAc;QACtB,cAAS,GAAT,SAAS,CAAc;QAtB3B,YAAO,GAAG,IAAI,aAAa,CAAkB,CAAC,CAAC,CAAA;QAE/C,UAAK,GAAG,IAAI,SAAS,CAAC;YAC3B,IAAI,EAAE,EAAE;YACR,OAAO,EAAE,EAAE;YACX,KAAK,EAAE,EAAE;YACT,OAAO,EAAE,EAAE,WAAW,EAAE,KAAK,EAAE,KAAK,EAAE,YAAY,EAAE;YACpD,SAAS,EAAE,EAAE,WAAW,EAAE,KAAK,EAAE;YACjC,QAAQ,EAAE,EAAE,WAAW,EAAE,KAAK,EAAE,KAAK,EAAE,YAAY,EAAE;YACrD,OAAO,EAAE,EAAE,WAAW,EAAE,KAAK,EAAE,KAAK,EAAE,YAAY,EAAE;SACvD,CAAC,CAAA;QAEO,YAAO,GAAW,EAAE,CAAA;QAGpB,kBAAa,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,SAAS,CAAC,CAAA;QAC/C,mBAAc,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,UAAU,CAAC,CAAA;QAEjD,QAAG,GAAG,GAAG,EAAE,WAAW,EAAE,CAAA;QAO7B,MAAM,CAAC,MAAM,CAAC,SAAS,CAAC,aAAa,CAAC,KAAK,EAAE;YACzC,QAAQ,EAAE,QAAQ;YAClB,UAAU,EAAE,QAAQ;YACpB,aAAa,EAAE,MAAM;SACxB,CAAC,CAAA;QACF,SAAS,CAAC,aAAa,CAAC,YAAY,CAAC,eAAe,EAAE,IAAI,CAAC,GAAG,CAAC,CAAA;QAE/D,IAAI,CAAC,OAAO,GAAG,MAAM,CAAA;QACrB,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,CAAA;QAE1D,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,KAAK,SAAS,EAAE,IAAI,CAAC,CAAC,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE;YACtF,IAAI,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,KAAK,EAAO,CAAC,CAAA;QACnC,CAAC,CAAC,CAAA;QACF,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,MAAM,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,aAAa,CAAC,EAAE,YAAY,CAAC,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAA;QAC3G,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,SAAS,EAAE,GAAG,EAAE;YAC1B,SAAS,CAAC,aAAa,CAAC,KAAK,CAAC,UAAU,GAAG,SAAS,CAAA;QACxD,CAAC,CAAC,CAAA;QACF,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,OAAO,EAAE,GAAG,EAAE;YACxB,SAAS,CAAC,aAAa,CAAC,KAAK,CAAC,aAAa,GAAG,IAAW,CAAA;QAC7D,CAAC,CAAC,CAAA;QACF,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,WAAW,EAAE,GAAG,EAAE;YAC5B,SAAS,CAAC,aAAa,CAAC,KAAK,CAAC,aAAa,GAAG,MAAM,CAAA;QACxD,CAAC,CAAC,CAAA;QAEF,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,SAAS,CAAC,KAAK,CAAC,CAAA;QAEnC,eAAe;QACf,2CAA2C;QAC3C,sCAAsC;QACtC,KAAK;IACT,CAAC;IAED,IAAI;QACA,OAAO,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,MAAM,EAAE,SAAS,EAAE,OAAO,CAAC,CAAA;IACrD,CAAC;IAED;;OAEG;IACH,IAAI;QACA,OAAO,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAA;IAC3B,CAAC;IAED,KAAK,CAAC,KAAK,GAAG,KAAK;QACf,IAAI,KAAK,EAAE,CAAC;YACR,OAAO,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,WAAW,EAAE,UAAU,EAAE,SAAS,CAAC,CAAA;QAC7D,CAAC;aAAM,CAAC;YACJ,OAAO,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,SAAS,EAAE,WAAW,EAAE,UAAU,EAAE,SAAS,CAAC,CAAA;QACxE,CAAC;IACL,CAAC;IAED,IAAI,CAAC,KAA6B;QAC9B,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,GAAG,KAAK,EAAE,WAAW,EAAE,IAAI,EAAS,CAAC,CAAA;IAC7D,CAAC;IAED,SAAS,CAAC,IAAS;QACf,IAAI,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAO,CAAC,CAAA;QACxC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,SAAS,EAAE,CAAA;IAChC,CAAC;IAED,UAAU,CAAI,IAAY;QACtB,OAAO,IAAI,CAAC,WAAW,CAAC,IAAI,CACxB,SAAS,CAAC,IAAI,CAAC,cAAc,CAAC,EAC9B,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,EACzB,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,IAAI,IAAI,CAAC,CAChB,CAAA;IACtB,CAAC;IAED,WAAW;QACP,MAAM,GAAG,GAA6D,EAAE,CAAA;QAExE,KAAK,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;YAChD,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,cAAc,CAAC,EAAE,SAAS,CAAC,IAAI,CAAC,CAAC,CAAA;QAClF,CAAC;QAED,IAAI,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAChC,OAAO,KAAK,CAAA;QAChB,CAAC;aAAM,CAAC;YACJ,OAAO,aAAa,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,CAAA;QAClD,CAAC;IACL,CAAC;8GA1GQ,WAAW,0EAwBR,MAAM;kHAxBT,WAAW;;2FAAX,WAAW;kBADvB,UAAU;;0BAyBF,MAAM;2BAAC,MAAM","sourcesContent":["import { Inject, Injectable, InjectionToken } from \"@angular/core\"\n\nimport {\n    combineLatest,\n    debounceTime,\n    EMPTY,\n    filter,\n    map,\n    Observable,\n    ReplaySubject,\n    shareReplay,\n    startWith,\n    take,\n    takeUntil,\n    takeWhile\n} from \"rxjs\"\n\nimport { Lifecycle } from \"@ngutil/common\"\n\nimport { ContainerRef } from \"../layer/container-ref\"\nimport { LayerService } from \"../layer/layer.service\"\nimport { FloatingTrait } from \"./traits/_base\"\n\nexport type Traits = { [key: string]: FloatingTrait }\n\nexport const TRAITS = new InjectionToken<Traits>(\"TRAITS\")\n\nexport interface FloatingChannel {\n    floatingRef: FloatingRef\n    type: string\n    data?: any\n}\n\nexport interface FloatingTraitEvent {\n    name: string\n    data: object\n}\n\ntype TraitState = { [key: string]: FloatingTraitEvent | null }\n\nlet UID_COUNTER = 0\n\n@Injectable()\nexport class FloatingRef<C extends FloatingChannel = FloatingChannel, T extends HTMLElement = HTMLElement> {\n    readonly channel = new ReplaySubject<FloatingChannel>(1)\n\n    readonly state = new Lifecycle({\n        init: {},\n        showing: {},\n        shown: {},\n        closing: { cancellable: false, order: \"sequential\" },\n        disposing: { cancellable: false },\n        disposed: { cancellable: false, order: \"sequential\" },\n        cleanup: { cancellable: false, order: \"sequential\" }\n    })\n\n    readonly #traits: Traits = {}\n    readonly traitState$: Observable<TraitState>\n\n    readonly #untilCleanup = this.state.onExecute(\"cleanup\")\n    readonly #untilDisposed = this.state.onExecute(\"disposed\")\n\n    readonly uid = `${++UID_COUNTER}`\n\n    constructor(\n        readonly layerSvc: LayerService,\n        readonly container: ContainerRef,\n        @Inject(TRAITS) traits: Traits\n    ) {\n        Object.assign(container.nativeElement.style, {\n            overflow: \"hidden\",\n            visibility: \"hidden\",\n            pointerEvents: \"none\"\n        })\n        container.nativeElement.setAttribute(\"data-floating\", this.uid)\n\n        this.#traits = traits\n        this.traitState$ = this.#traitState().pipe(shareReplay(1))\n\n        this.state.current$.pipe(takeWhile(state => state !== \"cleanup\", true)).subscribe(state => {\n            this.emit({ type: state } as C)\n        })\n        this.state.on(\"init\", () => this.traitState$.pipe(takeUntil(this.#untilCleanup), debounceTime(5), take(1)))\n        this.state.on(\"showing\", () => {\n            container.nativeElement.style.visibility = \"visible\"\n        })\n        this.state.on(\"shown\", () => {\n            container.nativeElement.style.pointerEvents = null as any\n        })\n        this.state.on(\"disposing\", () => {\n            container.nativeElement.style.pointerEvents = \"none\"\n        })\n\n        this.state.control(container.state)\n\n        // TODO: remove\n        // this.state.status$.subscribe(status => {\n        //     console.log(\"floating\", status)\n        // })\n    }\n\n    show() {\n        return this.state.run(\"init\", \"showing\", \"shown\")\n    }\n\n    /**\n     * @deprecated\n     */\n    hide() {\n        return this.close(true)\n    }\n\n    close(force = false) {\n        if (force) {\n            return this.state.run(\"disposing\", \"disposed\", \"cleanup\")\n        } else {\n            return this.state.run(\"closing\", \"disposing\", \"disposed\", \"cleanup\")\n        }\n    }\n\n    emit(event: Omit<C, \"floatingRef\">) {\n        this.channel.next({ ...event, floatingRef: this } as any)\n    }\n\n    setResult(data: any) {\n        this.emit({ type: \"result\", data } as C)\n        this.close(true).subscribe()\n    }\n\n    watchTrait<T>(name: string): Observable<T> {\n        return this.traitState$.pipe(\n            takeUntil(this.#untilDisposed),\n            map(state => state[name]),\n            filter(value => value != null)\n        ) as Observable<T>\n    }\n\n    #traitState(): Observable<TraitState> {\n        const src: { [key: string]: Observable<FloatingTraitEvent | null> } = {}\n\n        for (const [k, v] of Object.entries(this.#traits)) {\n            src[k] = v.connect(this).pipe(takeUntil(this.#untilDisposed), startWith(null))\n        }\n\n        if (Object.keys(src).length === 0) {\n            return EMPTY\n        } else {\n            return combineLatest(src).pipe(shareReplay(1))\n        }\n    }\n}\n"]}
@@ -1,25 +1,49 @@
1
1
  import { animate, AnimationBuilder, style } from "@angular/animations";
2
- import { Observable, tap } from "rxjs";
2
+ import { map, Observable, switchMap, take, tap, timer } from "rxjs";
3
3
  import { animationObservable } from "@ngutil/graphics";
4
- import { Duration, Ease } from "@ngutil/style";
4
+ import { alignmentToTransformOrigin, Duration, Ease } from "@ngutil/style";
5
5
  // https://tympanus.net/Development/ModalWindowEffects/
6
- const transitionDuration = Duration.FastMs;
6
+ const timing = `${Duration.FastMs}ms ${Ease.Deceleration}`;
7
7
  export class AnimationTrait {
8
- constructor(animation) {
8
+ constructor(animation, options) {
9
9
  this.animation = animation;
10
+ this.options = options;
10
11
  this.name = "animation";
11
12
  }
12
13
  connect(floatingRef) {
13
14
  return new Observable((dst) => {
14
15
  const builder = floatingRef.container.injector.get(AnimationBuilder);
15
16
  const element = floatingRef.container.nativeElement;
16
- floatingRef.state.on("showing", () => animationObservable({ builder, element, animation: this.animation.show }));
17
- floatingRef.state.on("disposing", () => animationObservable({ builder, element, animation: this.animation.hide }).pipe(tap(() => (element.style.display = "none"))));
17
+ const options = this.options || {};
18
+ floatingRef.state.on("showing", () => animationParams(floatingRef, 0, options.params).pipe(switchMap(params => animationObservable({
19
+ builder,
20
+ element,
21
+ animation: this.animation.show,
22
+ options: { ...options, params }
23
+ }))));
24
+ floatingRef.state.on("disposing", () => animationParams(floatingRef, 0, options.params).pipe(switchMap(params => animationObservable({
25
+ builder,
26
+ element,
27
+ animation: this.animation.hide,
28
+ options: { ...options, params }
29
+ })), tap(() => (element.style.display = "none"))));
18
30
  floatingRef.state.on("disposing", () => dst.complete());
19
31
  dst.next();
20
32
  });
21
33
  }
22
34
  }
35
+ function animationParams(floatingRef, delay, overrides) {
36
+ const src = delay > 0
37
+ ? timer(delay).pipe(switchMap(() => floatingRef.watchTrait("position")))
38
+ : floatingRef.watchTrait("position");
39
+ return src.pipe(take(1), map(position => {
40
+ const origin = position.computed ? alignmentToTransformOrigin(position.computed.content.align) : "center";
41
+ return {
42
+ origin,
43
+ ...overrides
44
+ };
45
+ }));
46
+ }
23
47
  export const FallAnimation = {
24
48
  show: [
25
49
  style({
@@ -27,27 +51,58 @@ export const FallAnimation = {
27
51
  visibility: "visible",
28
52
  opacity: "0"
29
53
  }),
30
- animate(`${transitionDuration}ms ${Ease.Deceleration}`, style({
54
+ animate(timing, style({
31
55
  transform: "scale(1)",
32
56
  opacity: "1"
33
57
  }))
34
58
  ],
35
59
  hide: [
36
- animate(`${transitionDuration}ms ${Ease.Deceleration}`, style({
60
+ animate(timing, style({
37
61
  transform: "scale(1.5)",
38
62
  visibility: "visible",
39
63
  opacity: "0"
40
64
  }))
41
65
  ]
42
66
  };
43
- export function fallAnimation() {
44
- return new AnimationTrait(FallAnimation);
67
+ export function fallAnimation(options) {
68
+ return new AnimationTrait(FallAnimation, options);
45
69
  }
46
70
  export const FadeAnimation = {
47
- show: [style({ opacity: 0 }), animate(`${transitionDuration}ms ${Ease.Deceleration}`, style({ opacity: 1 }))],
48
- hide: [animate(`${transitionDuration}ms ${Ease.Deceleration}`, style({ opacity: 0 }))]
71
+ show: [style({ opacity: 0 }), animate(timing, style({ opacity: 1 }))],
72
+ hide: [animate(timing, style({ opacity: 0 }))]
49
73
  };
50
- export function fadeAnimation() {
51
- return new AnimationTrait(FadeAnimation);
74
+ export function fadeAnimation(options) {
75
+ return new AnimationTrait(FadeAnimation, options);
76
+ }
77
+ export const DropAnimation = {
78
+ show: [
79
+ style({
80
+ transform: "translate({{ translateX }}, {{ translateY }})",
81
+ opacity: "0",
82
+ transformOrigin: "{{ origin }}",
83
+ visibility: "visible"
84
+ }),
85
+ animate(timing, style({
86
+ opacity: "1",
87
+ transform: "scale(1, 1) translate(0px, 0px)"
88
+ }))
89
+ ],
90
+ hide: [
91
+ animate(timing, style({ opacity: 0, transform: "translate(calc({{ translateX }} * -1), calc({{ translateY }} * -1))" }))
92
+ ]
93
+ };
94
+ export function dropAnimation(options) {
95
+ if (!options) {
96
+ options = {};
97
+ }
98
+ if (!options.params) {
99
+ options.params = {};
100
+ }
101
+ else {
102
+ options.params = { ...options.params };
103
+ }
104
+ options.params["translateX"] = "0px";
105
+ options.params["translateY"] = "-40px";
106
+ return new AnimationTrait(DropAnimation, options);
52
107
  }
53
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYW5pbWF0aW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvZmxvYXRpbmcvc3JjL2Zsb2F0aW5nL3RyYWl0cy9hbmltYXRpb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLE9BQU8sRUFBRSxnQkFBZ0IsRUFBcUIsS0FBSyxFQUFFLE1BQU0scUJBQXFCLENBQUE7QUFFekYsT0FBTyxFQUFFLFVBQVUsRUFBYyxHQUFHLEVBQUUsTUFBTSxNQUFNLENBQUE7QUFFbEQsT0FBTyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sa0JBQWtCLENBQUE7QUFDdEQsT0FBTyxFQUFFLFFBQVEsRUFBRSxJQUFJLEVBQUUsTUFBTSxlQUFlLENBQUE7QUFPOUMsdURBQXVEO0FBRXZELE1BQU0sa0JBQWtCLEdBQUcsUUFBUSxDQUFDLE1BQU0sQ0FBQTtBQUUxQyxNQUFNLE9BQU8sY0FBYztJQUd2QixZQUFxQixTQUF1QjtRQUF2QixjQUFTLEdBQVQsU0FBUyxDQUFjO1FBRm5DLFNBQUksR0FBRyxXQUFXLENBQUE7SUFFb0IsQ0FBQztJQUVoRCxPQUFPLENBQUMsV0FBd0I7UUFDNUIsT0FBTyxJQUFJLFVBQVUsQ0FBQyxDQUFDLEdBQXdCLEVBQUUsRUFBRTtZQUMvQyxNQUFNLE9BQU8sR0FBRyxXQUFXLENBQUMsU0FBUyxDQUFDLFFBQVEsQ0FBQyxHQUFHLENBQUMsZ0JBQWdCLENBQUMsQ0FBQTtZQUNwRSxNQUFNLE9BQU8sR0FBRyxXQUFXLENBQUMsU0FBUyxDQUFDLGFBQWEsQ0FBQTtZQUNuRCxXQUFXLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQyxTQUFTLEVBQUUsR0FBRyxFQUFFLENBQ2pDLG1CQUFtQixDQUFDLEVBQUUsT0FBTyxFQUFFLE9BQU8sRUFBRSxTQUFTLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLEVBQUUsQ0FBQyxDQUM1RSxDQUFBO1lBQ0QsV0FBVyxDQUFDLEtBQUssQ0FBQyxFQUFFLENBQUMsV0FBVyxFQUFFLEdBQUcsRUFBRSxDQUNuQyxtQkFBbUIsQ0FBQyxFQUFFLE9BQU8sRUFBRSxPQUFPLEVBQUUsU0FBUyxFQUFFLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxFQUFFLENBQUMsQ0FBQyxJQUFJLENBQzFFLEdBQUcsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUMsT0FBTyxHQUFHLE1BQU0sQ0FBQyxDQUFDLENBQzlDLENBQ0osQ0FBQTtZQUNELFdBQVcsQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDLFdBQVcsRUFBRSxHQUFHLEVBQUUsQ0FBQyxHQUFHLENBQUMsUUFBUSxFQUFFLENBQUMsQ0FBQTtZQUN2RCxHQUFHLENBQUMsSUFBSSxFQUFFLENBQUE7UUFDZCxDQUFDLENBQUMsQ0FBQTtJQUNOLENBQUM7Q0FDSjtBQUVELE1BQU0sQ0FBQyxNQUFNLGFBQWEsR0FBaUI7SUFDdkMsSUFBSSxFQUFFO1FBQ0YsS0FBSyxDQUFDO1lBQ0YsU0FBUyxFQUFFLFlBQVk7WUFDdkIsVUFBVSxFQUFFLFNBQVM7WUFDckIsT0FBTyxFQUFFLEdBQUc7U0FDZixDQUFDO1FBQ0YsT0FBTyxDQUNILEdBQUcsa0JBQWtCLE1BQU0sSUFBSSxDQUFDLFlBQVksRUFBRSxFQUM5QyxLQUFLLENBQUM7WUFDRixTQUFTLEVBQUUsVUFBVTtZQUNyQixPQUFPLEVBQUUsR0FBRztTQUNmLENBQUMsQ0FDTDtLQUNKO0lBQ0QsSUFBSSxFQUFFO1FBQ0YsT0FBTyxDQUNILEdBQUcsa0JBQWtCLE1BQU0sSUFBSSxDQUFDLFlBQVksRUFBRSxFQUM5QyxLQUFLLENBQUM7WUFDRixTQUFTLEVBQUUsWUFBWTtZQUN2QixVQUFVLEVBQUUsU0FBUztZQUNyQixPQUFPLEVBQUUsR0FBRztTQUNmLENBQUMsQ0FDTDtLQUNKO0NBQ0osQ0FBQTtBQUVELE1BQU0sVUFBVSxhQUFhO0lBQ3pCLE9BQU8sSUFBSSxjQUFjLENBQUMsYUFBYSxDQUFDLENBQUE7QUFDNUMsQ0FBQztBQUVELE1BQU0sQ0FBQyxNQUFNLGFBQWEsR0FBaUI7SUFDdkMsSUFBSSxFQUFFLENBQUMsS0FBSyxDQUFDLEVBQUUsT0FBTyxFQUFFLENBQUMsRUFBRSxDQUFDLEVBQUUsT0FBTyxDQUFDLEdBQUcsa0JBQWtCLE1BQU0sSUFBSSxDQUFDLFlBQVksRUFBRSxFQUFFLEtBQUssQ0FBQyxFQUFFLE9BQU8sRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7SUFDN0csSUFBSSxFQUFFLENBQUMsT0FBTyxDQUFDLEdBQUcsa0JBQWtCLE1BQU0sSUFBSSxDQUFDLFlBQVksRUFBRSxFQUFFLEtBQUssQ0FBQyxFQUFFLE9BQU8sRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7Q0FDekYsQ0FBQTtBQUVELE1BQU0sVUFBVSxhQUFhO0lBQ3pCLE9BQU8sSUFBSSxjQUFjLENBQUMsYUFBYSxDQUFDLENBQUE7QUFDNUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGFuaW1hdGUsIEFuaW1hdGlvbkJ1aWxkZXIsIEFuaW1hdGlvbk1ldGFkYXRhLCBzdHlsZSB9IGZyb20gXCJAYW5ndWxhci9hbmltYXRpb25zXCJcblxuaW1wb3J0IHsgT2JzZXJ2YWJsZSwgU3Vic2NyaWJlciwgdGFwIH0gZnJvbSBcInJ4anNcIlxuXG5pbXBvcnQgeyBhbmltYXRpb25PYnNlcnZhYmxlIH0gZnJvbSBcIkBuZ3V0aWwvZ3JhcGhpY3NcIlxuaW1wb3J0IHsgRHVyYXRpb24sIEVhc2UgfSBmcm9tIFwiQG5ndXRpbC9zdHlsZVwiXG5cbmltcG9ydCB7IEZsb2F0aW5nUmVmIH0gZnJvbSBcIi4uL2Zsb2F0aW5nLXJlZlwiXG5pbXBvcnQgeyBGbG9hdGluZ1RyYWl0IH0gZnJvbSBcIi4vX2Jhc2VcIlxuXG5leHBvcnQgdHlwZSBBbmltYXRpb25TZXQgPSB7IHNob3c6IEFuaW1hdGlvbk1ldGFkYXRhW107IGhpZGU6IEFuaW1hdGlvbk1ldGFkYXRhW10gfVxuXG4vLyBodHRwczovL3R5bXBhbnVzLm5ldC9EZXZlbG9wbWVudC9Nb2RhbFdpbmRvd0VmZmVjdHMvXG5cbmNvbnN0IHRyYW5zaXRpb25EdXJhdGlvbiA9IER1cmF0aW9uLkZhc3RNc1xuXG5leHBvcnQgY2xhc3MgQW5pbWF0aW9uVHJhaXQgaW1wbGVtZW50cyBGbG9hdGluZ1RyYWl0PHVua25vd24+IHtcbiAgICByZWFkb25seSBuYW1lID0gXCJhbmltYXRpb25cIlxuXG4gICAgY29uc3RydWN0b3IocmVhZG9ubHkgYW5pbWF0aW9uOiBBbmltYXRpb25TZXQpIHt9XG5cbiAgICBjb25uZWN0KGZsb2F0aW5nUmVmOiBGbG9hdGluZ1JlZik6IE9ic2VydmFibGU8dW5rbm93bj4ge1xuICAgICAgICByZXR1cm4gbmV3IE9ic2VydmFibGUoKGRzdDogU3Vic2NyaWJlcjx1bmtub3duPikgPT4ge1xuICAgICAgICAgICAgY29uc3QgYnVpbGRlciA9IGZsb2F0aW5nUmVmLmNvbnRhaW5lci5pbmplY3Rvci5nZXQoQW5pbWF0aW9uQnVpbGRlcilcbiAgICAgICAgICAgIGNvbnN0IGVsZW1lbnQgPSBmbG9hdGluZ1JlZi5jb250YWluZXIubmF0aXZlRWxlbWVudFxuICAgICAgICAgICAgZmxvYXRpbmdSZWYuc3RhdGUub24oXCJzaG93aW5nXCIsICgpID0+XG4gICAgICAgICAgICAgICAgYW5pbWF0aW9uT2JzZXJ2YWJsZSh7IGJ1aWxkZXIsIGVsZW1lbnQsIGFuaW1hdGlvbjogdGhpcy5hbmltYXRpb24uc2hvdyB9KVxuICAgICAgICAgICAgKVxuICAgICAgICAgICAgZmxvYXRpbmdSZWYuc3RhdGUub24oXCJkaXNwb3NpbmdcIiwgKCkgPT5cbiAgICAgICAgICAgICAgICBhbmltYXRpb25PYnNlcnZhYmxlKHsgYnVpbGRlciwgZWxlbWVudCwgYW5pbWF0aW9uOiB0aGlzLmFuaW1hdGlvbi5oaWRlIH0pLnBpcGUoXG4gICAgICAgICAgICAgICAgICAgIHRhcCgoKSA9PiAoZWxlbWVudC5zdHlsZS5kaXNwbGF5ID0gXCJub25lXCIpKVxuICAgICAgICAgICAgICAgIClcbiAgICAgICAgICAgIClcbiAgICAgICAgICAgIGZsb2F0aW5nUmVmLnN0YXRlLm9uKFwiZGlzcG9zaW5nXCIsICgpID0+IGRzdC5jb21wbGV0ZSgpKVxuICAgICAgICAgICAgZHN0Lm5leHQoKVxuICAgICAgICB9KVxuICAgIH1cbn1cblxuZXhwb3J0IGNvbnN0IEZhbGxBbmltYXRpb246IEFuaW1hdGlvblNldCA9IHtcbiAgICBzaG93OiBbXG4gICAgICAgIHN0eWxlKHtcbiAgICAgICAgICAgIHRyYW5zZm9ybTogXCJzY2FsZSgxLjUpXCIsXG4gICAgICAgICAgICB2aXNpYmlsaXR5OiBcInZpc2libGVcIixcbiAgICAgICAgICAgIG9wYWNpdHk6IFwiMFwiXG4gICAgICAgIH0pLFxuICAgICAgICBhbmltYXRlKFxuICAgICAgICAgICAgYCR7dHJhbnNpdGlvbkR1cmF0aW9ufW1zICR7RWFzZS5EZWNlbGVyYXRpb259YCxcbiAgICAgICAgICAgIHN0eWxlKHtcbiAgICAgICAgICAgICAgICB0cmFuc2Zvcm06IFwic2NhbGUoMSlcIixcbiAgICAgICAgICAgICAgICBvcGFjaXR5OiBcIjFcIlxuICAgICAgICAgICAgfSlcbiAgICAgICAgKVxuICAgIF0sXG4gICAgaGlkZTogW1xuICAgICAgICBhbmltYXRlKFxuICAgICAgICAgICAgYCR7dHJhbnNpdGlvbkR1cmF0aW9ufW1zICR7RWFzZS5EZWNlbGVyYXRpb259YCxcbiAgICAgICAgICAgIHN0eWxlKHtcbiAgICAgICAgICAgICAgICB0cmFuc2Zvcm06IFwic2NhbGUoMS41KVwiLFxuICAgICAgICAgICAgICAgIHZpc2liaWxpdHk6IFwidmlzaWJsZVwiLFxuICAgICAgICAgICAgICAgIG9wYWNpdHk6IFwiMFwiXG4gICAgICAgICAgICB9KVxuICAgICAgICApXG4gICAgXVxufVxuXG5leHBvcnQgZnVuY3Rpb24gZmFsbEFuaW1hdGlvbigpIHtcbiAgICByZXR1cm4gbmV3IEFuaW1hdGlvblRyYWl0KEZhbGxBbmltYXRpb24pXG59XG5cbmV4cG9ydCBjb25zdCBGYWRlQW5pbWF0aW9uOiBBbmltYXRpb25TZXQgPSB7XG4gICAgc2hvdzogW3N0eWxlKHsgb3BhY2l0eTogMCB9KSwgYW5pbWF0ZShgJHt0cmFuc2l0aW9uRHVyYXRpb259bXMgJHtFYXNlLkRlY2VsZXJhdGlvbn1gLCBzdHlsZSh7IG9wYWNpdHk6IDEgfSkpXSxcbiAgICBoaWRlOiBbYW5pbWF0ZShgJHt0cmFuc2l0aW9uRHVyYXRpb259bXMgJHtFYXNlLkRlY2VsZXJhdGlvbn1gLCBzdHlsZSh7IG9wYWNpdHk6IDAgfSkpXVxufVxuXG5leHBvcnQgZnVuY3Rpb24gZmFkZUFuaW1hdGlvbigpIHtcbiAgICByZXR1cm4gbmV3IEFuaW1hdGlvblRyYWl0KEZhZGVBbmltYXRpb24pXG59XG4iXX0=
108
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"animation.js","sourceRoot":"","sources":["../../../../../../packages/floating/src/floating/traits/animation.ts"],"names":[],"mappings":"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","sourcesContent":["import { animate, AnimationBuilder, AnimationMetadata, AnimationOptions, style } from \"@angular/animations\"\n\nimport { map, Observable, Subscriber, switchMap, take, tap, timer } from \"rxjs\"\n\nimport { animationObservable } from \"@ngutil/graphics\"\nimport { alignmentToTransformOrigin, Duration, Ease } from \"@ngutil/style\"\n\nimport { FloatingRef } from \"../floating-ref\"\nimport { FloatingTrait } from \"./_base\"\nimport { FloatingPosition } from \"./position\"\n\nexport type AnimationSet = { show: AnimationMetadata[]; hide: AnimationMetadata[] }\n\n// https://tympanus.net/Development/ModalWindowEffects/\n\nconst timing = `${Duration.FastMs}ms ${Ease.Deceleration}`\n\nexport class AnimationTrait implements FloatingTrait<unknown> {\n    readonly name = \"animation\"\n\n    constructor(\n        readonly animation: AnimationSet,\n        readonly options?: AnimationOptions\n    ) {}\n\n    connect(floatingRef: FloatingRef): Observable<unknown> {\n        return new Observable((dst: Subscriber<unknown>) => {\n            const builder = floatingRef.container.injector.get(AnimationBuilder)\n            const element = floatingRef.container.nativeElement\n            const options = this.options || {}\n\n            floatingRef.state.on(\"showing\", () =>\n                animationParams(floatingRef, 0, options.params).pipe(\n                    switchMap(params =>\n                        animationObservable({\n                            builder,\n                            element,\n                            animation: this.animation.show,\n                            options: { ...options, params }\n                        })\n                    )\n                )\n            )\n            floatingRef.state.on(\"disposing\", () =>\n                animationParams(floatingRef, 0, options.params).pipe(\n                    switchMap(params =>\n                        animationObservable({\n                            builder,\n                            element,\n                            animation: this.animation.hide,\n                            options: { ...options, params }\n                        })\n                    ),\n                    tap(() => (element.style.display = \"none\"))\n                )\n            )\n            floatingRef.state.on(\"disposing\", () => dst.complete())\n            dst.next()\n        })\n    }\n}\n\nfunction animationParams(\n    floatingRef: FloatingRef,\n    delay: number,\n    overrides: AnimationOptions[\"params\"]\n): Observable<AnimationOptions[\"params\"]> {\n    const src =\n        delay > 0\n            ? timer(delay).pipe(switchMap(() => floatingRef.watchTrait<FloatingPosition>(\"position\")))\n            : floatingRef.watchTrait<FloatingPosition>(\"position\")\n\n    return src.pipe(\n        take(1),\n        map(position => {\n            const origin = position.computed ? alignmentToTransformOrigin(position.computed!.content.align) : \"center\"\n            return {\n                origin,\n                ...overrides\n            }\n        })\n    )\n}\n\nexport const FallAnimation: AnimationSet = {\n    show: [\n        style({\n            transform: \"scale(1.5)\",\n            visibility: \"visible\",\n            opacity: \"0\"\n        }),\n        animate(\n            timing,\n            style({\n                transform: \"scale(1)\",\n                opacity: \"1\"\n            })\n        )\n    ],\n    hide: [\n        animate(\n            timing,\n            style({\n                transform: \"scale(1.5)\",\n                visibility: \"visible\",\n                opacity: \"0\"\n            })\n        )\n    ]\n}\n\nexport function fallAnimation(options?: AnimationOptions) {\n    return new AnimationTrait(FallAnimation, options)\n}\n\nexport const FadeAnimation: AnimationSet = {\n    show: [style({ opacity: 0 }), animate(timing, style({ opacity: 1 }))],\n    hide: [animate(timing, style({ opacity: 0 }))]\n}\n\nexport function fadeAnimation(options?: AnimationOptions) {\n    return new AnimationTrait(FadeAnimation, options)\n}\n\nexport const DropAnimation: AnimationSet = {\n    show: [\n        style({\n            transform: \"translate({{ translateX }}, {{ translateY }})\",\n            opacity: \"0\",\n            transformOrigin: \"{{ origin }}\",\n            visibility: \"visible\"\n        }),\n        animate(\n            timing,\n            style({\n                opacity: \"1\",\n                transform: \"scale(1, 1) translate(0px, 0px)\"\n            })\n        )\n    ],\n    hide: [\n        animate(\n            timing,\n            style({ opacity: 0, transform: \"translate(calc({{ translateX }} * -1), calc({{ translateY }} * -1))\" })\n        )\n    ]\n}\n\nexport function dropAnimation(options?: AnimationOptions) {\n    if (!options) {\n        options = {}\n    }\n\n    if (!options.params) {\n        options.params = {}\n    } else {\n        options.params = { ...options.params }\n    }\n\n    options.params[\"translateX\"] = \"0px\"\n    options.params[\"translateY\"] = \"-40px\"\n\n    return new AnimationTrait(DropAnimation, options)\n}\n"]}
@@ -1,12 +1,7 @@
1
1
  import { AnimationBuilder } from "@angular/animations";
2
- import { exhaustMap, map, Observable, Subject } from "rxjs";
2
+ import { Observable } from "rxjs";
3
3
  import { animationObservable } from "@ngutil/graphics";
4
4
  import { FadeAnimation } from "./animation";
5
- export class BackdropState {
6
- constructor() {
7
- this.onClick = new Subject();
8
- }
9
- }
10
5
  export class BackdropTrait {
11
6
  constructor(options) {
12
7
  this.options = options;
@@ -15,24 +10,10 @@ export class BackdropTrait {
15
10
  connect(floatingRef) {
16
11
  return new Observable((dest) => {
17
12
  const animationBuilder = floatingRef.container.injector.get(AnimationBuilder);
18
- const options = {
19
- type: this.options.type,
20
- under: floatingRef.container,
21
- color: this.options.color,
22
- style: this.options.style
23
- };
24
- if (this.options.type === "crop") {
25
- ;
26
- options.crop = floatingRef
27
- .watchTrait("position")
28
- .pipe(map(position => position.anchor));
29
- }
30
- const state = new BackdropState();
31
- const backdrop = floatingRef.layerSvc.newBackdrop(options);
32
- if (this.options.closeOnClick) {
33
- dest.add(this.#installClickHandler(floatingRef, backdrop, state));
34
- dest.add(state.onClick.pipe(exhaustMap(() => floatingRef.close())).subscribe());
35
- }
13
+ const options = { ...this.options };
14
+ const backdrop = floatingRef.layerSvc.newBackdrop(floatingRef.container, options);
15
+ floatingRef.container.nativeElement.setAttribute("data-floating-has-backdrop", "true");
16
+ backdrop.nativeElement.setAttribute("data-floating-backdrop", floatingRef.uid);
36
17
  backdrop.state.on("showing", () => animationObservable({
37
18
  builder: animationBuilder,
38
19
  element: backdrop.nativeElement,
@@ -46,26 +27,11 @@ export class BackdropTrait {
46
27
  backdrop.state.on("disposed", () => dest.complete());
47
28
  floatingRef.state.on("disposing", () => backdrop.dispose());
48
29
  dest.add(backdrop.show().subscribe());
49
- dest.next(state);
30
+ dest.next(backdrop);
50
31
  });
51
32
  }
52
- #installClickHandler(floatingRef, backdrop, state) {
53
- const handler = (event) => {
54
- if (event.defaultPrevented) {
55
- return;
56
- }
57
- if (event.target === backdrop.nativeElement || backdrop.nativeElement.contains(event.target)) {
58
- ;
59
- state.onClick.next();
60
- }
61
- };
62
- document.addEventListener("click", handler);
63
- return () => {
64
- document.removeEventListener("click", handler);
65
- };
66
- }
67
33
  }
68
34
  export function backdrop(options) {
69
35
  return new BackdropTrait(options);
70
36
  }
71
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"backdrop.js","sourceRoot":"","sources":["../../../../../../packages/floating/src/floating/traits/backdrop.ts"],"names":[],"mappings":"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","sourcesContent":["import { AnimationBuilder } from \"@angular/animations\"\n\nimport { exhaustMap, map, Observable, Subject, Subscriber } from \"rxjs\"\n\nimport { animationObservable } from \"@ngutil/graphics\"\n\nimport type { BackdropOptions, BackdropRef, CropBackdropOptions } from \"../../layer/backdrop-ref\"\nimport { type FloatingRef } from \"../floating-ref\"\nimport { FloatingTrait } from \"./_base\"\nimport { FadeAnimation } from \"./animation\"\nimport { FloatingPosition } from \"./position\"\n\nexport interface BackdropTraitOptions {\n    type: BackdropOptions[\"type\"]\n    color: BackdropOptions[\"color\"]\n    style?: Partial<CSSStyleDeclaration>\n    closeOnClick?: boolean\n    // TODO: maybe cropMargin\n}\n\nexport class BackdropState {\n    onClick: Observable<void> = new Subject<void>()\n}\n\nexport class BackdropTrait implements FloatingTrait<BackdropState> {\n    readonly name = \"backdrop\"\n\n    constructor(readonly options: BackdropTraitOptions) {}\n\n    connect(floatingRef: FloatingRef<any>): Observable<BackdropState> {\n        return new Observable((dest: Subscriber<BackdropState>) => {\n            const animationBuilder = floatingRef.container.injector.get(AnimationBuilder)\n            const options: BackdropOptions = {\n                type: this.options.type,\n                under: floatingRef.container,\n                color: this.options.color,\n                style: this.options.style\n            } as any\n\n            if (this.options.type === \"crop\") {\n                ;(options as CropBackdropOptions).crop = floatingRef\n                    .watchTrait<FloatingPosition>(\"position\")\n                    .pipe(map(position => position.anchor))\n            }\n\n            const state = new BackdropState()\n            const backdrop = floatingRef.layerSvc.newBackdrop(options)\n\n            if (this.options.closeOnClick) {\n                dest.add(this.#installClickHandler(floatingRef, backdrop, state))\n                dest.add(state.onClick.pipe(exhaustMap(() => floatingRef.close())).subscribe())\n            }\n\n            backdrop.state.on(\"showing\", () =>\n                animationObservable({\n                    builder: animationBuilder,\n                    element: backdrop.nativeElement,\n                    animation: FadeAnimation.show\n                })\n            )\n            backdrop.state.on(\"disposing\", () =>\n                animationObservable({\n                    builder: animationBuilder,\n                    element: backdrop.nativeElement,\n                    animation: FadeAnimation.hide\n                })\n            )\n\n            backdrop.state.on(\"disposed\", () => dest.complete())\n\n            floatingRef.state.on(\"disposing\", () => backdrop.dispose())\n\n            dest.add(backdrop.show().subscribe())\n            dest.next(state)\n        })\n    }\n\n    #installClickHandler(floatingRef: FloatingRef<any>, backdrop: BackdropRef, state: BackdropState) {\n        const handler = (event: MouseEvent) => {\n            if (event.defaultPrevented) {\n                return\n            }\n\n            if (event.target === backdrop.nativeElement || backdrop.nativeElement.contains(event.target as Node)) {\n                ;(state.onClick as Subject<void>).next()\n            }\n        }\n\n        document.addEventListener(\"click\", handler)\n\n        return () => {\n            document.removeEventListener(\"click\", handler)\n        }\n    }\n}\n\nexport function backdrop(options: BackdropTraitOptions) {\n    return new BackdropTrait(options)\n}\n"]}
37
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmFja2Ryb3AuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9mbG9hdGluZy9zcmMvZmxvYXRpbmcvdHJhaXRzL2JhY2tkcm9wLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLHFCQUFxQixDQUFBO0FBRXRELE9BQU8sRUFBRSxVQUFVLEVBQWMsTUFBTSxNQUFNLENBQUE7QUFFN0MsT0FBTyxFQUFFLG1CQUFtQixFQUFnQixNQUFNLGtCQUFrQixDQUFBO0FBS3BFLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxhQUFhLENBQUE7QUFNM0MsTUFBTSxPQUFPLGFBQWE7SUFHdEIsWUFBcUIsT0FBNkI7UUFBN0IsWUFBTyxHQUFQLE9BQU8sQ0FBc0I7UUFGekMsU0FBSSxHQUFHLFVBQVUsQ0FBQTtJQUUyQixDQUFDO0lBRXRELE9BQU8sQ0FBQyxXQUE2QjtRQUNqQyxPQUFPLElBQUksVUFBVSxDQUFDLENBQUMsSUFBNkIsRUFBRSxFQUFFO1lBQ3BELE1BQU0sZ0JBQWdCLEdBQUcsV0FBVyxDQUFDLFNBQVMsQ0FBQyxRQUFRLENBQUMsR0FBRyxDQUFDLGdCQUFnQixDQUFDLENBQUE7WUFDN0UsTUFBTSxPQUFPLEdBQUcsRUFBRSxHQUFHLElBQUksQ0FBQyxPQUFPLEVBQUUsQ0FBQTtZQUNuQyxNQUFNLFFBQVEsR0FBRyxXQUFXLENBQUMsUUFBUSxDQUFDLFdBQVcsQ0FBQyxXQUFXLENBQUMsU0FBUyxFQUFFLE9BQU8sQ0FBQyxDQUFBO1lBRWpGLFdBQVcsQ0FBQyxTQUFTLENBQUMsYUFBYSxDQUFDLFlBQVksQ0FBQyw0QkFBNEIsRUFBRSxNQUFNLENBQUMsQ0FBQTtZQUN0RixRQUFRLENBQUMsYUFBYSxDQUFDLFlBQVksQ0FBQyx3QkFBd0IsRUFBRSxXQUFXLENBQUMsR0FBRyxDQUFDLENBQUE7WUFFOUUsUUFBUSxDQUFDLEtBQUssQ0FBQyxFQUFFLENBQUMsU0FBUyxFQUFFLEdBQUcsRUFBRSxDQUM5QixtQkFBbUIsQ0FBQztnQkFDaEIsT0FBTyxFQUFFLGdCQUFnQjtnQkFDekIsT0FBTyxFQUFFLFFBQVEsQ0FBQyxhQUFhO2dCQUMvQixTQUFTLEVBQUUsYUFBYSxDQUFDLElBQUk7YUFDaEMsQ0FBQyxDQUNMLENBQUE7WUFDRCxRQUFRLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQyxXQUFXLEVBQUUsR0FBRyxFQUFFLENBQ2hDLG1CQUFtQixDQUFDO2dCQUNoQixPQUFPLEVBQUUsZ0JBQWdCO2dCQUN6QixPQUFPLEVBQUUsUUFBUSxDQUFDLGFBQWE7Z0JBQy9CLFNBQVMsRUFBRSxhQUFhLENBQUMsSUFBSTthQUNoQyxDQUFDLENBQ0wsQ0FBQTtZQUVELFFBQVEsQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDLFVBQVUsRUFBRSxHQUFHLEVBQUUsQ0FBQyxJQUFJLENBQUMsUUFBUSxFQUFFLENBQUMsQ0FBQTtZQUVwRCxXQUFXLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQyxXQUFXLEVBQUUsR0FBRyxFQUFFLENBQUMsUUFBUSxDQUFDLE9BQU8sRUFBRSxDQUFDLENBQUE7WUFFM0QsSUFBSSxDQUFDLEdBQUcsQ0FBQyxRQUFRLENBQUMsSUFBSSxFQUFFLENBQUMsU0FBUyxFQUFFLENBQUMsQ0FBQTtZQUNyQyxJQUFJLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFBO1FBQ3ZCLENBQUMsQ0FBQyxDQUFBO0lBQ04sQ0FBQztDQUNKO0FBRUQsTUFBTSxVQUFVLFFBQVEsQ0FBQyxPQUE2QjtJQUNsRCxPQUFPLElBQUksYUFBYSxDQUFDLE9BQU8sQ0FBQyxDQUFBO0FBQ3JDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBbmltYXRpb25CdWlsZGVyIH0gZnJvbSBcIkBhbmd1bGFyL2FuaW1hdGlvbnNcIlxuXG5pbXBvcnQgeyBPYnNlcnZhYmxlLCBTdWJzY3JpYmVyIH0gZnJvbSBcInJ4anNcIlxuXG5pbXBvcnQgeyBhbmltYXRpb25PYnNlcnZhYmxlLCBDb3Zlck9wdGlvbnMgfSBmcm9tIFwiQG5ndXRpbC9ncmFwaGljc1wiXG5cbmltcG9ydCB7IEJhY2tkcm9wUmVmIH0gZnJvbSBcIi4uLy4uL2xheWVyXCJcbmltcG9ydCB7IHR5cGUgRmxvYXRpbmdSZWYgfSBmcm9tIFwiLi4vZmxvYXRpbmctcmVmXCJcbmltcG9ydCB7IEZsb2F0aW5nVHJhaXQgfSBmcm9tIFwiLi9fYmFzZVwiXG5pbXBvcnQgeyBGYWRlQW5pbWF0aW9uIH0gZnJvbSBcIi4vYW5pbWF0aW9uXCJcblxuaW50ZXJmYWNlIENvbW1vbk9wdGlvbnMge31cblxuZXhwb3J0IHR5cGUgQmFja2Ryb3BUcmFpdE9wdGlvbnMgPSBDb3Zlck9wdGlvbnMgJiBDb21tb25PcHRpb25zXG5cbmV4cG9ydCBjbGFzcyBCYWNrZHJvcFRyYWl0IGltcGxlbWVudHMgRmxvYXRpbmdUcmFpdDxCYWNrZHJvcFJlZj4ge1xuICAgIHJlYWRvbmx5IG5hbWUgPSBcImJhY2tkcm9wXCJcblxuICAgIGNvbnN0cnVjdG9yKHJlYWRvbmx5IG9wdGlvbnM6IEJhY2tkcm9wVHJhaXRPcHRpb25zKSB7fVxuXG4gICAgY29ubmVjdChmbG9hdGluZ1JlZjogRmxvYXRpbmdSZWY8YW55Pik6IE9ic2VydmFibGU8QmFja2Ryb3BSZWY+IHtcbiAgICAgICAgcmV0dXJuIG5ldyBPYnNlcnZhYmxlKChkZXN0OiBTdWJzY3JpYmVyPEJhY2tkcm9wUmVmPikgPT4ge1xuICAgICAgICAgICAgY29uc3QgYW5pbWF0aW9uQnVpbGRlciA9IGZsb2F0aW5nUmVmLmNvbnRhaW5lci5pbmplY3Rvci5nZXQoQW5pbWF0aW9uQnVpbGRlcilcbiAgICAgICAgICAgIGNvbnN0IG9wdGlvbnMgPSB7IC4uLnRoaXMub3B0aW9ucyB9XG4gICAgICAgICAgICBjb25zdCBiYWNrZHJvcCA9IGZsb2F0aW5nUmVmLmxheWVyU3ZjLm5ld0JhY2tkcm9wKGZsb2F0aW5nUmVmLmNvbnRhaW5lciwgb3B0aW9ucylcblxuICAgICAgICAgICAgZmxvYXRpbmdSZWYuY29udGFpbmVyLm5hdGl2ZUVsZW1lbnQuc2V0QXR0cmlidXRlKFwiZGF0YS1mbG9hdGluZy1oYXMtYmFja2Ryb3BcIiwgXCJ0cnVlXCIpXG4gICAgICAgICAgICBiYWNrZHJvcC5uYXRpdmVFbGVtZW50LnNldEF0dHJpYnV0ZShcImRhdGEtZmxvYXRpbmctYmFja2Ryb3BcIiwgZmxvYXRpbmdSZWYudWlkKVxuXG4gICAgICAgICAgICBiYWNrZHJvcC5zdGF0ZS5vbihcInNob3dpbmdcIiwgKCkgPT5cbiAgICAgICAgICAgICAgICBhbmltYXRpb25PYnNlcnZhYmxlKHtcbiAgICAgICAgICAgICAgICAgICAgYnVpbGRlcjogYW5pbWF0aW9uQnVpbGRlcixcbiAgICAgICAgICAgICAgICAgICAgZWxlbWVudDogYmFja2Ryb3AubmF0aXZlRWxlbWVudCxcbiAgICAgICAgICAgICAgICAgICAgYW5pbWF0aW9uOiBGYWRlQW5pbWF0aW9uLnNob3dcbiAgICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgKVxuICAgICAgICAgICAgYmFja2Ryb3Auc3RhdGUub24oXCJkaXNwb3NpbmdcIiwgKCkgPT5cbiAgICAgICAgICAgICAgICBhbmltYXRpb25PYnNlcnZhYmxlKHtcbiAgICAgICAgICAgICAgICAgICAgYnVpbGRlcjogYW5pbWF0aW9uQnVpbGRlcixcbiAgICAgICAgICAgICAgICAgICAgZWxlbWVudDogYmFja2Ryb3AubmF0aXZlRWxlbWVudCxcbiAgICAgICAgICAgICAgICAgICAgYW5pbWF0aW9uOiBGYWRlQW5pbWF0aW9uLmhpZGVcbiAgICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgKVxuXG4gICAgICAgICAgICBiYWNrZHJvcC5zdGF0ZS5vbihcImRpc3Bvc2VkXCIsICgpID0+IGRlc3QuY29tcGxldGUoKSlcblxuICAgICAgICAgICAgZmxvYXRpbmdSZWYuc3RhdGUub24oXCJkaXNwb3NpbmdcIiwgKCkgPT4gYmFja2Ryb3AuZGlzcG9zZSgpKVxuXG4gICAgICAgICAgICBkZXN0LmFkZChiYWNrZHJvcC5zaG93KCkuc3Vic2NyaWJlKCkpXG4gICAgICAgICAgICBkZXN0Lm5leHQoYmFja2Ryb3ApXG4gICAgICAgIH0pXG4gICAgfVxufVxuXG5leHBvcnQgZnVuY3Rpb24gYmFja2Ryb3Aob3B0aW9uczogQmFja2Ryb3BUcmFpdE9wdGlvbnMpIHtcbiAgICByZXR1cm4gbmV3IEJhY2tkcm9wVHJhaXQob3B0aW9ucylcbn1cbiJdfQ==
@@ -0,0 +1,86 @@
1
+ import { distinctUntilChanged, exhaustMap, filter, finalize, from, fromEvent, map, of, race } from "rxjs";
2
+ import { KeystrokeService } from "@ngutil/aria";
3
+ import { coerceElement } from "@ngutil/common";
4
+ class CloseTriggerTrait {
5
+ constructor(options = {}) {
6
+ this.options = options;
7
+ this.name = "close-trigger";
8
+ }
9
+ connect(floatingRef) {
10
+ const { keystroke, clickOutside, trigger } = this.options;
11
+ const container = floatingRef.container.nativeElement;
12
+ const triggers = [];
13
+ const selfUid = Number(floatingRef.uid);
14
+ if (keystroke) {
15
+ const ks = floatingRef.container.injector.get(KeystrokeService);
16
+ triggers.push(ks.watch(container, { key: "Escape", state: "up" }).pipe(map(() => {
17
+ return { source: "keystroke" };
18
+ })));
19
+ // TODO: angular auxiliary route
20
+ }
21
+ if (clickOutside) {
22
+ const allowedElements = typeof clickOutside === "boolean" ? [] : clickOutside.allowedElements?.map(coerceElement) || [];
23
+ triggers.push(fromEvent(document, "click", { capture: true, passive: true }).pipe(filter(event => {
24
+ if (!(event.target instanceof HTMLElement)) {
25
+ return false;
26
+ }
27
+ const target = event.target;
28
+ for (const allowed of allowedElements) {
29
+ if (target === allowed || allowed.contains(target)) {
30
+ return false;
31
+ }
32
+ }
33
+ const floatingUid = getFloatingUid(target, "data-floating", "floating");
34
+ const backdropUid = getFloatingUid(target, "data-floating-backdrop", "floatingBackdrop");
35
+ const otherBackdropUid = floatingUid != null && floatingUid !== selfUid
36
+ ? getFloatingUid(document.querySelector(`[data-floating-backdrop="${floatingUid}"]`), "data-floating-backdrop", "floatingBackdrop")
37
+ : undefined;
38
+ // console.log({ floatingUid, backdropUid, otherBackdropUid, self: selfUid })
39
+ if (floatingUid == null && backdropUid == null) {
40
+ return true;
41
+ }
42
+ else {
43
+ return (
44
+ // click on self backdrop
45
+ (backdropUid != null && backdropUid === selfUid) ||
46
+ // click on other floating element, whitout backdrop
47
+ (floatingUid != null && otherBackdropUid == null && floatingUid !== selfUid) ||
48
+ // click on other floating element that opened erlier
49
+ (floatingUid != null && floatingUid < selfUid));
50
+ }
51
+ }), map(() => {
52
+ return { source: "click" };
53
+ })));
54
+ }
55
+ if (trigger) {
56
+ triggers.push(from(trigger).pipe(map(() => {
57
+ return { source: "trigger" };
58
+ })));
59
+ }
60
+ if (triggers.length === 0) {
61
+ return of();
62
+ }
63
+ else {
64
+ return race(...triggers).pipe(finalize(() => console.log("CLOSED", floatingRef.uid)), exhaustMap(event => floatingRef.close().pipe(map(() => event), distinctUntilChanged())));
65
+ }
66
+ }
67
+ }
68
+ export function closeTrigger(options = { clickOutside: true, keystroke: true }) {
69
+ return new CloseTriggerTrait(options);
70
+ }
71
+ function getFloatingUid(el, attr, dataset) {
72
+ if (el == null) {
73
+ return undefined;
74
+ }
75
+ if (el.matches(`[${attr}]`)) {
76
+ return Number(el.dataset[dataset]) || undefined;
77
+ }
78
+ else {
79
+ const parent = el.closest(`[${attr}]`);
80
+ if (parent) {
81
+ return Number(parent.dataset[dataset]) || undefined;
82
+ }
83
+ }
84
+ return undefined;
85
+ }
86
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"close-trigger.js","sourceRoot":"","sources":["../../../../../../packages/floating/src/floating/traits/close-trigger.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    distinctUntilChanged,\n    exhaustMap,\n    filter,\n    finalize,\n    from,\n    fromEvent,\n    map,\n    Observable,\n    ObservableInput,\n    of,\n    race\n} from \"rxjs\"\n\nimport { KeystrokeService } from \"@ngutil/aria\"\nimport { coerceElement, ElementInput } from \"@ngutil/common\"\n\nimport { FloatingRef } from \"../floating-ref\"\nimport { FloatingTrait } from \"./_base\"\n\nexport interface ClickOutsideOptions {\n    /**\n     * If clicks happened on this element, that not trigger the close event\n     */\n    allowedElements?: ElementInput[]\n}\n\nexport interface CloseTriggerOptions {\n    /**\n     * Close floating when click outside of floating element\n     */\n    clickOutside?: boolean | ClickOutsideOptions\n\n    /**\n     * Only works when the floating element is focused\n     */\n    keystroke?: boolean\n\n    /**\n     * Close when emitting any value\n     */\n    trigger?: ObservableInput<any>\n}\n\nexport interface CloseTriggerEvent {\n    source: \"click\" | \"keystroke\" | \"backbutton\" | \"trigger\"\n}\n\nclass CloseTriggerTrait implements FloatingTrait {\n    readonly name = \"close-trigger\"\n\n    constructor(readonly options: CloseTriggerOptions = {}) {}\n\n    connect(floatingRef: FloatingRef): Observable<CloseTriggerEvent | void> {\n        const { keystroke, clickOutside, trigger } = this.options\n\n        const container = floatingRef.container.nativeElement\n        const triggers = []\n        const selfUid = Number(floatingRef.uid)\n\n        if (keystroke) {\n            const ks = floatingRef.container.injector.get(KeystrokeService)\n\n            triggers.push(\n                ks.watch(container, { key: \"Escape\", state: \"up\" }).pipe(\n                    map(() => {\n                        return { source: \"keystroke\" as const }\n                    })\n                )\n            )\n\n            // TODO: angular auxiliary route\n        }\n\n        if (clickOutside) {\n            const allowedElements =\n                typeof clickOutside === \"boolean\" ? [] : clickOutside.allowedElements?.map(coerceElement) || []\n\n            triggers.push(\n                fromEvent(document, \"click\", { capture: true, passive: true }).pipe(\n                    filter(event => {\n                        if (!(event.target instanceof HTMLElement)) {\n                            return false\n                        }\n                        const target = event.target\n\n                        for (const allowed of allowedElements) {\n                            if (target === allowed || allowed.contains(target)) {\n                                return false\n                            }\n                        }\n\n                        const floatingUid = getFloatingUid(target, \"data-floating\", \"floating\")\n                        const backdropUid = getFloatingUid(target, \"data-floating-backdrop\", \"floatingBackdrop\")\n                        const otherBackdropUid =\n                            floatingUid != null && floatingUid !== selfUid\n                                ? getFloatingUid(\n                                      document.querySelector(`[data-floating-backdrop=\"${floatingUid}\"]`),\n                                      \"data-floating-backdrop\",\n                                      \"floatingBackdrop\"\n                                  )\n                                : undefined\n\n                        // console.log({ floatingUid, backdropUid, otherBackdropUid, self: selfUid })\n\n                        if (floatingUid == null && backdropUid == null) {\n                            return true\n                        } else {\n                            return (\n                                // click on self backdrop\n                                (backdropUid != null && backdropUid === selfUid) ||\n                                // click on other floating element, whitout backdrop\n                                (floatingUid != null && otherBackdropUid == null && floatingUid !== selfUid) ||\n                                // click on other floating element that opened erlier\n                                (floatingUid != null && floatingUid < selfUid)\n                            )\n                        }\n                    }),\n                    map(() => {\n                        return { source: \"click\" as const }\n                    })\n                )\n            )\n        }\n\n        if (trigger) {\n            triggers.push(\n                from(trigger).pipe(\n                    map(() => {\n                        return { source: \"trigger\" as const }\n                    })\n                )\n            )\n        }\n\n        if (triggers.length === 0) {\n            return of()\n        } else {\n            return race(...triggers).pipe(\n                finalize(() => console.log(\"CLOSED\", floatingRef.uid)),\n                exhaustMap(event =>\n                    floatingRef.close().pipe(\n                        map(() => event),\n                        distinctUntilChanged()\n                    )\n                )\n            )\n        }\n    }\n}\n\nexport function closeTrigger(options: CloseTriggerOptions = { clickOutside: true, keystroke: true }) {\n    return new CloseTriggerTrait(options)\n}\n\nfunction getFloatingUid(el: HTMLElement | null, attr: string, dataset: string): number | undefined {\n    if (el == null) {\n        return undefined\n    }\n\n    if (el.matches(`[${attr}]`)) {\n        return Number(el.dataset[dataset]) || undefined\n    } else {\n        const parent = el.closest(`[${attr}]`) as HTMLElement\n        if (parent) {\n            return Number(parent.dataset[dataset]) || undefined\n        }\n    }\n    return undefined\n}\n"]}
@@ -24,7 +24,7 @@ export class DimensionConstraintTrait {
24
24
  refDim: refDim,
25
25
  position: floatingRef.watchTrait("position")
26
26
  }).subscribe(({ refDim, position }) => {
27
- const floating = position.computed?.floating;
27
+ const floating = position.computed?.content;
28
28
  if (!floating) {
29
29
  return;
30
30
  }
@@ -33,7 +33,7 @@ export class DimensionConstraintTrait {
33
33
  }
34
34
  else {
35
35
  return floatingRef.watchTrait("position").subscribe(position => {
36
- const floating = position.computed?.floating;
36
+ const floating = position.computed?.content;
37
37
  if (!floating) {
38
38
  return;
39
39
  }
@@ -41,7 +41,7 @@ export class DimensionConstraintTrait {
41
41
  dst.next(floating[this.#map.computedRef][this.#map.dimension]);
42
42
  }
43
43
  else {
44
- dst.next(clamp(this.value, floating?.min[this.#map.dimension] || 0, floating?.max[this.#map.dimension] || Infinity));
44
+ dst.next(clamp(this.value, floating.min[this.#map.dimension] || 0, floating.max[this.#map.dimension] || Infinity));
45
45
  }
46
46
  });
47
47
  }
@@ -63,4 +63,4 @@ export function minWidth(value) {
63
63
  export function minHeight(value) {
64
64
  return new DimensionConstraintTrait("minHeight", value);
65
65
  }
66
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dim-contraint.js","sourceRoot":"","sources":["../../../../../../packages/floating/src/floating/traits/dim-contraint.ts"],"names":[],"mappings":"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","sourcesContent":["import { combineLatest, distinctUntilChanged, map, Observable, Subscriber, takeUntil, tap } from \"rxjs\"\n\nimport { clamp } from \"lodash\"\n\nimport { ElementInput, isElementInput } from \"@ngutil/common\"\nimport { Dimension, DimensionWatcher } from \"@ngutil/style\"\n\nimport { FloatingRef } from \"../floating-ref\"\nimport { FloatingTrait } from \"./_base\"\nimport type { FloatingPosition } from \"./position\"\n\nexport type DimensionConstraintInput = ElementInput | number\n\ninterface DimMapEntry {\n    computedRef: \"min\" | \"max\"\n    dimension: keyof Dimension\n}\n\nconst DIM_MAP: { [key: string]: DimMapEntry } = {\n    maxWidth: { computedRef: \"max\", dimension: \"width\" },\n    maxHeight: { computedRef: \"max\", dimension: \"height\" },\n    minWidth: { computedRef: \"min\", dimension: \"width\" },\n    minHeight: { computedRef: \"min\", dimension: \"height\" }\n}\n\nexport class DimensionConstraintTrait implements FloatingTrait<number> {\n    readonly name: string\n    readonly #map: DimMapEntry\n    constructor(\n        name: keyof typeof DIM_MAP,\n        readonly value: DimensionConstraintInput\n    ) {\n        this.name = name as string\n        this.#map = DIM_MAP[name]\n    }\n\n    connect(floatingRef: FloatingRef<any>): Observable<number> {\n        return new Observable((dst: Subscriber<number>) => {\n            if (isElementInput(this.value)) {\n                const watcher = floatingRef.container.injector.get(DimensionWatcher)\n                const refDim = watcher.watch(this.value, \"border-box\").pipe(map(value => value[this.#map.dimension]))\n                return combineLatest({\n                    refDim: refDim,\n                    position: floatingRef.watchTrait<FloatingPosition>(\"position\")\n                }).subscribe(({ refDim, position }) => {\n                    const floating = position.computed?.floating\n                    if (!floating) {\n                        return\n                    }\n                    dst.next(\n                        clamp(\n                            refDim,\n                            floating.min[this.#map.dimension] || 0,\n                            floating.max[this.#map.dimension] || Infinity\n                        )\n                    )\n                })\n            } else {\n                return floatingRef.watchTrait<FloatingPosition>(\"position\").subscribe(position => {\n                    const floating = position.computed?.floating\n                    if (!floating) {\n                        return\n                    }\n                    if (isNaN(this.value as number)) {\n                        dst.next(floating[this.#map.computedRef][this.#map.dimension])\n                    } else {\n                        dst.next(\n                            clamp(\n                                this.value as number,\n                                floating?.min[this.#map.dimension] || 0,\n                                floating?.max[this.#map.dimension] || Infinity\n                            )\n                        )\n                    }\n                })\n            }\n        }).pipe(\n            takeUntil(floatingRef.state.onExecute(\"disposing\")),\n            distinctUntilChanged(),\n            tap(value => {\n                const floatingEl = floatingRef.container.nativeElement\n                floatingEl.style[this.name as any] = `${value}px`\n            })\n        )\n    }\n}\n\nexport function maxWidth(value: DimensionConstraintInput) {\n    return new DimensionConstraintTrait(\"maxWidth\", value)\n}\n\nexport function maxHeight(value: DimensionConstraintInput) {\n    return new DimensionConstraintTrait(\"maxHeight\", value)\n}\n\nexport function minWidth(value: DimensionConstraintInput) {\n    return new DimensionConstraintTrait(\"minWidth\", value)\n}\n\nexport function minHeight(value: DimensionConstraintInput) {\n    return new DimensionConstraintTrait(\"minHeight\", value)\n}\n"]}
66
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dim-contraint.js","sourceRoot":"","sources":["../../../../../../packages/floating/src/floating/traits/dim-contraint.ts"],"names":[],"mappings":"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","sourcesContent":["import { combineLatest, distinctUntilChanged, map, Observable, Subscriber, takeUntil, tap } from \"rxjs\"\n\nimport { clamp } from \"lodash\"\n\nimport { ElementInput, isElementInput } from \"@ngutil/common\"\nimport { Dimension, DimensionWatcher } from \"@ngutil/style\"\n\nimport { FloatingRef } from \"../floating-ref\"\nimport { FloatingTrait } from \"./_base\"\nimport type { FloatingPosition } from \"./position\"\n\nexport type DimensionConstraintInput = ElementInput | number\n\ninterface DimMapEntry {\n    computedRef: \"min\" | \"max\"\n    dimension: keyof Dimension\n}\n\nconst DIM_MAP: { [key: string]: DimMapEntry } = {\n    maxWidth: { computedRef: \"max\", dimension: \"width\" },\n    maxHeight: { computedRef: \"max\", dimension: \"height\" },\n    minWidth: { computedRef: \"min\", dimension: \"width\" },\n    minHeight: { computedRef: \"min\", dimension: \"height\" }\n}\n\nexport class DimensionConstraintTrait implements FloatingTrait<number> {\n    readonly name: string\n    readonly #map: DimMapEntry\n    constructor(\n        name: keyof typeof DIM_MAP,\n        readonly value: DimensionConstraintInput\n    ) {\n        this.name = name as string\n        this.#map = DIM_MAP[name]\n    }\n\n    connect(floatingRef: FloatingRef<any>): Observable<number> {\n        return new Observable((dst: Subscriber<number>) => {\n            if (isElementInput(this.value)) {\n                const watcher = floatingRef.container.injector.get(DimensionWatcher)\n                const refDim = watcher.watch(this.value, \"border-box\").pipe(map(value => value[this.#map.dimension]))\n                return combineLatest({\n                    refDim: refDim,\n                    position: floatingRef.watchTrait<FloatingPosition>(\"position\")\n                }).subscribe(({ refDim, position }) => {\n                    const floating = position.computed?.content\n                    if (!floating) {\n                        return\n                    }\n                    dst.next(\n                        clamp(\n                            refDim,\n                            floating.min[this.#map.dimension] || 0,\n                            floating.max[this.#map.dimension] || Infinity\n                        )\n                    )\n                })\n            } else {\n                return floatingRef.watchTrait<FloatingPosition>(\"position\").subscribe(position => {\n                    const floating = position.computed?.content\n                    if (!floating) {\n                        return\n                    }\n                    if (isNaN(this.value as number)) {\n                        dst.next(floating[this.#map.computedRef][this.#map.dimension])\n                    } else {\n                        dst.next(\n                            clamp(\n                                this.value as number,\n                                floating!.min[this.#map.dimension] || 0,\n                                floating!.max[this.#map.dimension] || Infinity\n                            )\n                        )\n                    }\n                })\n            }\n        }).pipe(\n            takeUntil(floatingRef.state.onExecute(\"disposing\")),\n            distinctUntilChanged(),\n            tap(value => {\n                const floatingEl = floatingRef.container.nativeElement\n                floatingEl.style[this.name as any] = `${value}px`\n            })\n        )\n    }\n}\n\nexport function maxWidth(value: DimensionConstraintInput) {\n    return new DimensionConstraintTrait(\"maxWidth\", value)\n}\n\nexport function maxHeight(value: DimensionConstraintInput) {\n    return new DimensionConstraintTrait(\"maxHeight\", value)\n}\n\nexport function minWidth(value: DimensionConstraintInput) {\n    return new DimensionConstraintTrait(\"minWidth\", value)\n}\n\nexport function minHeight(value: DimensionConstraintInput) {\n    return new DimensionConstraintTrait(\"minHeight\", value)\n}\n"]}
@@ -1,12 +1,12 @@
1
1
  export * from "./_base";
2
2
  export * from "./animation";
3
3
  export * from "./backdrop";
4
+ export * from "./close-trigger";
4
5
  export * from "./dim-contraint";
5
6
  export * from "./focus";
6
- export * from "./keystroke";
7
7
  export * from "./modal";
8
8
  export * from "./position-calc";
9
9
  export * from "./position";
10
10
  export * from "./style";
11
11
  export * from "./attribute";
12
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9mbG9hdGluZy9zcmMvZmxvYXRpbmcvdHJhaXRzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsU0FBUyxDQUFBO0FBQ3ZCLGNBQWMsYUFBYSxDQUFBO0FBQzNCLGNBQWMsWUFBWSxDQUFBO0FBQzFCLGNBQWMsaUJBQWlCLENBQUE7QUFDL0IsY0FBYyxTQUFTLENBQUE7QUFDdkIsY0FBYyxhQUFhLENBQUE7QUFDM0IsY0FBYyxTQUFTLENBQUE7QUFDdkIsY0FBYyxpQkFBaUIsQ0FBQTtBQUMvQixjQUFjLFlBQVksQ0FBQTtBQUMxQixjQUFjLFNBQVMsQ0FBQTtBQUN2QixjQUFjLGFBQWEsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gXCIuL19iYXNlXCJcbmV4cG9ydCAqIGZyb20gXCIuL2FuaW1hdGlvblwiXG5leHBvcnQgKiBmcm9tIFwiLi9iYWNrZHJvcFwiXG5leHBvcnQgKiBmcm9tIFwiLi9kaW0tY29udHJhaW50XCJcbmV4cG9ydCAqIGZyb20gXCIuL2ZvY3VzXCJcbmV4cG9ydCAqIGZyb20gXCIuL2tleXN0cm9rZVwiXG5leHBvcnQgKiBmcm9tIFwiLi9tb2RhbFwiXG5leHBvcnQgKiBmcm9tIFwiLi9wb3NpdGlvbi1jYWxjXCJcbmV4cG9ydCAqIGZyb20gXCIuL3Bvc2l0aW9uXCJcbmV4cG9ydCAqIGZyb20gXCIuL3N0eWxlXCJcbmV4cG9ydCAqIGZyb20gXCIuL2F0dHJpYnV0ZVwiXG4iXX0=
12
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9mbG9hdGluZy9zcmMvZmxvYXRpbmcvdHJhaXRzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsU0FBUyxDQUFBO0FBQ3ZCLGNBQWMsYUFBYSxDQUFBO0FBQzNCLGNBQWMsWUFBWSxDQUFBO0FBQzFCLGNBQWMsaUJBQWlCLENBQUE7QUFDL0IsY0FBYyxpQkFBaUIsQ0FBQTtBQUMvQixjQUFjLFNBQVMsQ0FBQTtBQUN2QixjQUFjLFNBQVMsQ0FBQTtBQUN2QixjQUFjLGlCQUFpQixDQUFBO0FBQy9CLGNBQWMsWUFBWSxDQUFBO0FBQzFCLGNBQWMsU0FBUyxDQUFBO0FBQ3ZCLGNBQWMsYUFBYSxDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSBcIi4vX2Jhc2VcIlxuZXhwb3J0ICogZnJvbSBcIi4vYW5pbWF0aW9uXCJcbmV4cG9ydCAqIGZyb20gXCIuL2JhY2tkcm9wXCJcbmV4cG9ydCAqIGZyb20gXCIuL2Nsb3NlLXRyaWdnZXJcIlxuZXhwb3J0ICogZnJvbSBcIi4vZGltLWNvbnRyYWludFwiXG5leHBvcnQgKiBmcm9tIFwiLi9mb2N1c1wiXG5leHBvcnQgKiBmcm9tIFwiLi9tb2RhbFwiXG5leHBvcnQgKiBmcm9tIFwiLi9wb3NpdGlvbi1jYWxjXCJcbmV4cG9ydCAqIGZyb20gXCIuL3Bvc2l0aW9uXCJcbmV4cG9ydCAqIGZyb20gXCIuL3N0eWxlXCJcbmV4cG9ydCAqIGZyb20gXCIuL2F0dHJpYnV0ZVwiXG4iXX0=
@@ -1,9 +1,9 @@
1
1
  import { fallAnimation } from "./animation";
2
2
  import { backdrop } from "./backdrop";
3
+ import { closeTrigger } from "./close-trigger";
3
4
  import { focus } from "./focus";
4
- import { keystroke } from "./keystroke";
5
5
  import { position } from "./position";
6
- export function modal(options = {}) {
6
+ export function modal() {
7
7
  return [
8
8
  position({
9
9
  anchor: {
@@ -15,10 +15,10 @@ export function modal(options = {}) {
15
15
  padding: "16px"
16
16
  }
17
17
  }),
18
- backdrop({ type: "solid", color: "rgba(0, 0, 0, .3)", closeOnClick: options.closeOnBackdropClick }),
18
+ backdrop({ type: "solid", color: "rgba(0, 0, 0, .3)" }),
19
19
  focus({ trap: true }),
20
- keystroke(),
20
+ closeTrigger(),
21
21
  fallAnimation()
22
22
  ];
23
23
  }
24
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibW9kYWwuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9mbG9hdGluZy9zcmMvZmxvYXRpbmcvdHJhaXRzL21vZGFsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxhQUFhLENBQUE7QUFDM0MsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLFlBQVksQ0FBQTtBQUNyQyxPQUFPLEVBQUUsS0FBSyxFQUFFLE1BQU0sU0FBUyxDQUFBO0FBQy9CLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxhQUFhLENBQUE7QUFDdkMsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLFlBQVksQ0FBQTtBQU1yQyxNQUFNLFVBQVUsS0FBSyxDQUFDLFVBQXdCLEVBQUU7SUFDNUMsT0FBTztRQUNILFFBQVEsQ0FBQztZQUNMLE1BQU0sRUFBRTtnQkFDSixHQUFHLEVBQUUsVUFBVTtnQkFDZixLQUFLLEVBQUUsZUFBZTthQUN6QjtZQUNELFNBQVMsRUFBRTtnQkFDUCxHQUFHLEVBQUUsVUFBVTtnQkFDZixPQUFPLEVBQUUsTUFBTTthQUNsQjtTQUNKLENBQUM7UUFDRixRQUFRLENBQUMsRUFBRSxJQUFJLEVBQUUsT0FBTyxFQUFFLEtBQUssRUFBRSxtQkFBbUIsRUFBRSxZQUFZLEVBQUUsT0FBTyxDQUFDLG9CQUFvQixFQUFFLENBQUM7UUFDbkcsS0FBSyxDQUFDLEVBQUUsSUFBSSxFQUFFLElBQUksRUFBRSxDQUFDO1FBQ3JCLFNBQVMsRUFBRTtRQUNYLGFBQWEsRUFBRTtLQUNsQixDQUFBO0FBQ0wsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGZhbGxBbmltYXRpb24gfSBmcm9tIFwiLi9hbmltYXRpb25cIlxuaW1wb3J0IHsgYmFja2Ryb3AgfSBmcm9tIFwiLi9iYWNrZHJvcFwiXG5pbXBvcnQgeyBmb2N1cyB9IGZyb20gXCIuL2ZvY3VzXCJcbmltcG9ydCB7IGtleXN0cm9rZSB9IGZyb20gXCIuL2tleXN0cm9rZVwiXG5pbXBvcnQgeyBwb3NpdGlvbiB9IGZyb20gXCIuL3Bvc2l0aW9uXCJcblxuZXhwb3J0IGludGVyZmFjZSBNb2RhbE9wdGlvbnMge1xuICAgIGNsb3NlT25CYWNrZHJvcENsaWNrPzogYm9vbGVhblxufVxuXG5leHBvcnQgZnVuY3Rpb24gbW9kYWwob3B0aW9uczogTW9kYWxPcHRpb25zID0ge30pIHtcbiAgICByZXR1cm4gW1xuICAgICAgICBwb3NpdGlvbih7XG4gICAgICAgICAgICBhbmNob3I6IHtcbiAgICAgICAgICAgICAgICByZWY6IFwidmlld3BvcnRcIixcbiAgICAgICAgICAgICAgICBhbGlnbjogXCJjZW50ZXIgbWlkZGxlXCJcbiAgICAgICAgICAgIH0sXG4gICAgICAgICAgICBwbGFjZW1lbnQ6IHtcbiAgICAgICAgICAgICAgICByZWY6IFwidmlld3BvcnRcIixcbiAgICAgICAgICAgICAgICBwYWRkaW5nOiBcIjE2cHhcIlxuICAgICAgICAgICAgfVxuICAgICAgICB9KSxcbiAgICAgICAgYmFja2Ryb3AoeyB0eXBlOiBcInNvbGlkXCIsIGNvbG9yOiBcInJnYmEoMCwgMCwgMCwgLjMpXCIsIGNsb3NlT25DbGljazogb3B0aW9ucy5jbG9zZU9uQmFja2Ryb3BDbGljayB9KSxcbiAgICAgICAgZm9jdXMoeyB0cmFwOiB0cnVlIH0pLFxuICAgICAgICBrZXlzdHJva2UoKSxcbiAgICAgICAgZmFsbEFuaW1hdGlvbigpXG4gICAgXVxufVxuIl19
24
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibW9kYWwuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9mbG9hdGluZy9zcmMvZmxvYXRpbmcvdHJhaXRzL21vZGFsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxhQUFhLENBQUE7QUFDM0MsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLFlBQVksQ0FBQTtBQUNyQyxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUE7QUFDOUMsT0FBTyxFQUFFLEtBQUssRUFBRSxNQUFNLFNBQVMsQ0FBQTtBQUMvQixPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sWUFBWSxDQUFBO0FBRXJDLE1BQU0sVUFBVSxLQUFLO0lBQ2pCLE9BQU87UUFDSCxRQUFRLENBQUM7WUFDTCxNQUFNLEVBQUU7Z0JBQ0osR0FBRyxFQUFFLFVBQVU7Z0JBQ2YsS0FBSyxFQUFFLGVBQWU7YUFDekI7WUFDRCxTQUFTLEVBQUU7Z0JBQ1AsR0FBRyxFQUFFLFVBQVU7Z0JBQ2YsT0FBTyxFQUFFLE1BQU07YUFDbEI7U0FDSixDQUFDO1FBQ0YsUUFBUSxDQUFDLEVBQUUsSUFBSSxFQUFFLE9BQU8sRUFBRSxLQUFLLEVBQUUsbUJBQW1CLEVBQUUsQ0FBQztRQUN2RCxLQUFLLENBQUMsRUFBRSxJQUFJLEVBQUUsSUFBSSxFQUFFLENBQUM7UUFDckIsWUFBWSxFQUFFO1FBQ2QsYUFBYSxFQUFFO0tBQ2xCLENBQUE7QUFDTCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgZmFsbEFuaW1hdGlvbiB9IGZyb20gXCIuL2FuaW1hdGlvblwiXG5pbXBvcnQgeyBiYWNrZHJvcCB9IGZyb20gXCIuL2JhY2tkcm9wXCJcbmltcG9ydCB7IGNsb3NlVHJpZ2dlciB9IGZyb20gXCIuL2Nsb3NlLXRyaWdnZXJcIlxuaW1wb3J0IHsgZm9jdXMgfSBmcm9tIFwiLi9mb2N1c1wiXG5pbXBvcnQgeyBwb3NpdGlvbiB9IGZyb20gXCIuL3Bvc2l0aW9uXCJcblxuZXhwb3J0IGZ1bmN0aW9uIG1vZGFsKCkge1xuICAgIHJldHVybiBbXG4gICAgICAgIHBvc2l0aW9uKHtcbiAgICAgICAgICAgIGFuY2hvcjoge1xuICAgICAgICAgICAgICAgIHJlZjogXCJ2aWV3cG9ydFwiLFxuICAgICAgICAgICAgICAgIGFsaWduOiBcImNlbnRlciBtaWRkbGVcIlxuICAgICAgICAgICAgfSxcbiAgICAgICAgICAgIHBsYWNlbWVudDoge1xuICAgICAgICAgICAgICAgIHJlZjogXCJ2aWV3cG9ydFwiLFxuICAgICAgICAgICAgICAgIHBhZGRpbmc6IFwiMTZweFwiXG4gICAgICAgICAgICB9XG4gICAgICAgIH0pLFxuICAgICAgICBiYWNrZHJvcCh7IHR5cGU6IFwic29saWRcIiwgY29sb3I6IFwicmdiYSgwLCAwLCAwLCAuMylcIiB9KSxcbiAgICAgICAgZm9jdXMoeyB0cmFwOiB0cnVlIH0pLFxuICAgICAgICBjbG9zZVRyaWdnZXIoKSxcbiAgICAgICAgZmFsbEFuaW1hdGlvbigpXG4gICAgXVxufVxuIl19