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;
|