angular-three 1.9.2 → 1.9.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -63,13 +63,10 @@ export class NgtRxStore extends RxState {
63
63
  this.cache = {};
64
64
  // set a dummy property so that initial this.get() won't return undefined
65
65
  this.set({ __ngt_dummy__: '__ngt_dummy__' });
66
- // call initialize that might be setup by derived Stores
67
- this.initialize();
68
66
  // override set so our consumers don't have to handle undefined for state that already have default values
69
67
  const originalSet = this.set.bind(this);
70
68
  Object.defineProperty(this, 'set', {
71
69
  get: () => {
72
- // Parameters type does not do well with overloads (RxState#set). So we use any[] here
73
70
  return (...args) => {
74
71
  const firstArg = args[0];
75
72
  if (is.obj(firstArg)) {
@@ -79,11 +76,22 @@ export class NgtRxStore extends RxState {
79
76
  }, {});
80
77
  return originalSet(modArgs);
81
78
  }
82
- // @ts-expect-error not sure why ...args here doesn't pass tuple check
83
79
  return originalSet(...args);
84
80
  };
85
81
  },
86
82
  });
83
+ // override get to return {} if get() returns undefined
84
+ const originalGet = this.get.bind(this);
85
+ Object.defineProperty(this, 'get', {
86
+ get: () => {
87
+ return (...args) => {
88
+ const state = originalGet(...args);
89
+ return state || {};
90
+ };
91
+ },
92
+ });
93
+ // call initialize that might be setup by derived Stores
94
+ this.initialize();
87
95
  }
88
96
  initialize() {
89
97
  return;
@@ -117,4 +125,4 @@ NgtRxStore.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version:
117
125
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.1.3", ngImport: i0, type: NgtRxStore, decorators: [{
118
126
  type: Injectable
119
127
  }], ctorParameters: function () { return []; } });
120
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rx-store.js","sourceRoot":"","sources":["../../../../../../libs/angular-three/src/lib/stores/rx-store.ts"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectorRef, Injectable, OnDestroy } from '@angular/core';\nimport { RxState } from '@rx-angular/state';\nimport { combineLatest, distinctUntilChanged, MonoTypeOperatorFunction, Observable, startWith, tap } from 'rxjs';\nimport type { NgtAnyRecord } from '../types';\nimport { is } from '../utils/is';\n\nexport const startWithUndefined = <T>(): MonoTypeOperatorFunction<T> => startWith<T>(undefined! as T);\n\ntype EffectFn<TValue> = (\n    value: TValue\n) => void | undefined | ((cleanUpParams: { prev: TValue | undefined; complete: boolean; error: boolean }) => void);\n\n/**\n * An extended `tap` operator that accepts an `effectFn` which:\n * - runs on every `next` notification from `source$`\n * - can optionally return a `cleanUp` function that\n * invokes from the 2nd `next` notification onward and on `unsubscribe` (destroyed)\n *\n *\n * @example\n * ```typescript\n * source$.pipe(\n *  tapEffect((sourceValue) = {\n *    const cb = () => {\n *      doStuff(sourceValue);\n *    };\n *    addListener('event', cb);\n *\n *    return () => {\n *      removeListener('event', cb);\n *    }\n *  })\n * )\n * ```\n */\nexport function tapEffect<TValue>(effectFn: EffectFn<TValue>): MonoTypeOperatorFunction<TValue> {\n    let cleanupFn: (cleanUpParams: { prev: TValue | undefined; complete: boolean; error: boolean }) => void = () => {};\n    let firstRun = false;\n    let prev: TValue | undefined = undefined;\n\n    const teardown = (error: boolean) => {\n        return () => {\n            if (cleanupFn) {\n                cleanupFn({ prev, complete: true, error });\n            }\n        };\n    };\n\n    return tap<TValue>({\n        next: (value: TValue) => {\n            if (cleanupFn && firstRun) {\n                cleanupFn({ prev, complete: false, error: false });\n            }\n\n            const cleanUpOrVoid = effectFn(value);\n            if (cleanUpOrVoid) {\n                cleanupFn = cleanUpOrVoid;\n            }\n\n            prev = value;\n\n            if (!firstRun) {\n                firstRun = true;\n            }\n        },\n        complete: teardown(false),\n        unsubscribe: teardown(false),\n        error: teardown(true),\n    });\n}\n\n@Injectable()\nexport class NgtRxStore<TState extends object = any, TRxState extends object = TState & Record<string, any>>\n    extends RxState<TRxState>\n    implements OnDestroy\n{\n    constructor() {\n        super();\n        // set a dummy property so that initial this.get() won't return undefined\n        this.set({ __ngt_dummy__: '__ngt_dummy__' } as TRxState);\n        // call initialize that might be setup by derived Stores\n        this.initialize();\n        // override set so our consumers don't have to handle undefined for state that already have default values\n        const originalSet = this.set.bind(this);\n        Object.defineProperty(this, 'set', {\n            get: () => {\n                // Parameters type does not do well with overloads (RxState#set). So we use any[] here\n                return (...args: any[]) => {\n                    const firstArg = args[0];\n                    if (is.obj(firstArg)) {\n                        const modArgs = Object.entries(firstArg).reduce((modded, [key, value]) => {\n                            modded[key] = value === undefined ? this.get(key as keyof TRxState) : value;\n                            return modded;\n                        }, {} as NgtAnyRecord);\n                        return originalSet(modArgs as Partial<TRxState>);\n                    }\n                    // @ts-expect-error not sure why ...args here doesn't pass tuple check\n                    return originalSet(...args);\n                };\n            },\n        });\n    }\n\n    protected initialize() {\n        return;\n    }\n\n    effect<S>(obs: Observable<S>, sideEffectFn: EffectFn<S>): void {\n        return this.hold(obs.pipe(tapEffect(sideEffectFn)));\n    }\n\n    triggerChangeDetection(cdr: ChangeDetectorRef, keys: Array<keyof TRxState> = []) {\n        let $: Observable<any> = this.$;\n\n        if (keys.length) {\n            $ = combineLatest(keys.map((key) => this.select(key).pipe(startWith(this.get(key) ?? undefined))));\n        }\n\n        this.hold($, () => {\n            requestAnimationFrame(() => void cdr.detectChanges());\n        });\n    }\n\n    private cache: Record<string, Observable<any>> = {};\n    key$<K extends keyof TRxState>(key: K): Observable<TRxState[K]>;\n    key$<K1 extends keyof TRxState, K2 extends keyof TRxState[K1]>(key1: K1, key2: K2): Observable<TRxState[K1][K2]>;\n    key$<K1 extends keyof TRxState, K2 extends keyof TRxState[K1], K3 extends keyof TRxState[K1][K2]>(\n        key1: K1,\n        key2: K2,\n        key3: K3\n    ): Observable<TRxState[K1][K2][K3]>;\n    key$(...keys: string[]) {\n        const key = keys.join('.');\n        if (!this.cache[key]) {\n            this.cache[key] = this.select(...(keys as [any])).pipe(\n                startWith(this.get(...(keys as [any])) ?? undefined),\n                distinctUntilChanged()\n            );\n        }\n        return this.cache[key];\n    }\n\n    override ngOnDestroy() {\n        this.cache = {};\n        super.ngOnDestroy();\n    }\n}\n"]}
128
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rx-store.js","sourceRoot":"","sources":["../../../../../../libs/angular-three/src/lib/stores/rx-store.ts"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectorRef, Injectable, OnDestroy } from '@angular/core';\nimport { RxState } from '@rx-angular/state';\nimport { combineLatest, distinctUntilChanged, MonoTypeOperatorFunction, Observable, startWith, tap } from 'rxjs';\nimport type { NgtAnyRecord } from '../types';\nimport { is } from '../utils/is';\n\nexport const startWithUndefined = <T>(): MonoTypeOperatorFunction<T> => startWith<T>(undefined! as T);\n\ntype EffectFn<TValue> = (\n    value: TValue\n) => void | undefined | ((cleanUpParams: { prev: TValue | undefined; complete: boolean; error: boolean }) => void);\n\n/**\n * An extended `tap` operator that accepts an `effectFn` which:\n * - runs on every `next` notification from `source$`\n * - can optionally return a `cleanUp` function that\n * invokes from the 2nd `next` notification onward and on `unsubscribe` (destroyed)\n *\n *\n * @example\n * ```typescript\n * source$.pipe(\n *  tapEffect((sourceValue) = {\n *    const cb = () => {\n *      doStuff(sourceValue);\n *    };\n *    addListener('event', cb);\n *\n *    return () => {\n *      removeListener('event', cb);\n *    }\n *  })\n * )\n * ```\n */\nexport function tapEffect<TValue>(effectFn: EffectFn<TValue>): MonoTypeOperatorFunction<TValue> {\n    let cleanupFn: (cleanUpParams: { prev: TValue | undefined; complete: boolean; error: boolean }) => void = () => {};\n    let firstRun = false;\n    let prev: TValue | undefined = undefined;\n\n    const teardown = (error: boolean) => {\n        return () => {\n            if (cleanupFn) {\n                cleanupFn({ prev, complete: true, error });\n            }\n        };\n    };\n\n    return tap<TValue>({\n        next: (value: TValue) => {\n            if (cleanupFn && firstRun) {\n                cleanupFn({ prev, complete: false, error: false });\n            }\n\n            const cleanUpOrVoid = effectFn(value);\n            if (cleanUpOrVoid) {\n                cleanupFn = cleanUpOrVoid;\n            }\n\n            prev = value;\n\n            if (!firstRun) {\n                firstRun = true;\n            }\n        },\n        complete: teardown(false),\n        unsubscribe: teardown(false),\n        error: teardown(true),\n    });\n}\n\n@Injectable()\nexport class NgtRxStore<TState extends object = any, TRxState extends object = TState & Record<string, any>>\n    extends RxState<TRxState>\n    implements OnDestroy\n{\n    constructor() {\n        super();\n        // set a dummy property so that initial this.get() won't return undefined\n        this.set({ __ngt_dummy__: '__ngt_dummy__' } as TRxState);\n        // override set so our consumers don't have to handle undefined for state that already have default values\n        const originalSet = this.set.bind(this);\n        Object.defineProperty(this, 'set', {\n            get: () => {\n                return (...args: Parameters<RxState<TRxState>['set']>) => {\n                    const firstArg = args[0];\n                    if (is.obj(firstArg)) {\n                        const modArgs = Object.entries(firstArg).reduce((modded, [key, value]) => {\n                            modded[key] = value === undefined ? this.get(key as keyof TRxState) : value;\n                            return modded;\n                        }, {} as NgtAnyRecord);\n                        return originalSet(modArgs as Partial<TRxState>);\n                    }\n                    return originalSet(...args);\n                };\n            },\n        });\n\n        // override get to return {} if get() returns undefined\n        const originalGet = this.get.bind(this);\n        Object.defineProperty(this, 'get', {\n            get: () => {\n                return (...args: Parameters<RxState<TRxState>['get']>) => {\n                    const state = originalGet(...args);\n                    return state || {};\n                };\n            },\n        });\n\n        // call initialize that might be setup by derived Stores\n        this.initialize();\n    }\n\n    protected initialize() {\n        return;\n    }\n\n    effect<S>(obs: Observable<S>, sideEffectFn: EffectFn<S>): void {\n        return this.hold(obs.pipe(tapEffect(sideEffectFn)));\n    }\n\n    triggerChangeDetection(cdr: ChangeDetectorRef, keys: Array<keyof TRxState> = []) {\n        let $: Observable<any> = this.$;\n\n        if (keys.length) {\n            $ = combineLatest(keys.map((key) => this.select(key).pipe(startWith(this.get(key) ?? undefined))));\n        }\n\n        this.hold($, () => {\n            requestAnimationFrame(() => void cdr.detectChanges());\n        });\n    }\n\n    private cache: Record<string, Observable<any>> = {};\n    key$<K extends keyof TRxState>(key: K): Observable<TRxState[K]>;\n    key$<K1 extends keyof TRxState, K2 extends keyof TRxState[K1]>(key1: K1, key2: K2): Observable<TRxState[K1][K2]>;\n    key$<K1 extends keyof TRxState, K2 extends keyof TRxState[K1], K3 extends keyof TRxState[K1][K2]>(\n        key1: K1,\n        key2: K2,\n        key3: K3\n    ): Observable<TRxState[K1][K2][K3]>;\n    key$(...keys: string[]) {\n        const key = keys.join('.');\n        if (!this.cache[key]) {\n            this.cache[key] = this.select(...(keys as [any])).pipe(\n                startWith(this.get(...(keys as [any])) ?? undefined),\n                distinctUntilChanged()\n            );\n        }\n        return this.cache[key];\n    }\n\n    override ngOnDestroy() {\n        this.cache = {};\n        super.ngOnDestroy();\n    }\n}\n"]}
@@ -511,13 +511,10 @@ class NgtRxStore extends RxState {
511
511
  this.cache = {};
512
512
  // set a dummy property so that initial this.get() won't return undefined
513
513
  this.set({ __ngt_dummy__: '__ngt_dummy__' });
514
- // call initialize that might be setup by derived Stores
515
- this.initialize();
516
514
  // override set so our consumers don't have to handle undefined for state that already have default values
517
515
  const originalSet = this.set.bind(this);
518
516
  Object.defineProperty(this, 'set', {
519
517
  get: () => {
520
- // Parameters type does not do well with overloads (RxState#set). So we use any[] here
521
518
  return (...args) => {
522
519
  const firstArg = args[0];
523
520
  if (is.obj(firstArg)) {
@@ -527,11 +524,22 @@ class NgtRxStore extends RxState {
527
524
  }, {});
528
525
  return originalSet(modArgs);
529
526
  }
530
- // @ts-expect-error not sure why ...args here doesn't pass tuple check
531
527
  return originalSet(...args);
532
528
  };
533
529
  },
534
530
  });
531
+ // override get to return {} if get() returns undefined
532
+ const originalGet = this.get.bind(this);
533
+ Object.defineProperty(this, 'get', {
534
+ get: () => {
535
+ return (...args) => {
536
+ const state = originalGet(...args);
537
+ return state || {};
538
+ };
539
+ },
540
+ });
541
+ // call initialize that might be setup by derived Stores
542
+ this.initialize();
535
543
  }
536
544
  initialize() {
537
545
  return;