@legendapp/state 2.2.0-next.7 → 2.2.0-next.71
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +4 -2
- package/babel.js.map +1 -1
- package/config/enable$get.d.ts +8 -0
- package/config/enable$get.js +24 -0
- package/config/enable$get.js.map +1 -0
- package/config/enable$get.mjs +21 -0
- package/config/enable$get.mjs.map +1 -0
- package/config/enableReactComponents.js.map +1 -1
- package/config/enableReactComponents.mjs.map +1 -1
- package/config/enableReactNativeComponents.js.map +1 -1
- package/config/enableReactNativeComponents.mjs.map +1 -1
- package/config/enableReactTracking.d.ts +0 -9
- package/config/enableReactTracking.js.map +1 -1
- package/config/enableReactTracking.mjs.map +1 -1
- package/config/enableReactUse.d.ts +1 -1
- package/config/enableReactUse.js +1 -0
- package/config/enableReactUse.js.map +1 -1
- package/config/enableReactUse.mjs +1 -0
- package/config/enableReactUse.mjs.map +1 -1
- package/config/enable_peek.d.ts +8 -0
- package/config/{enableDirectPeek.js → enable_peek.js} +6 -3
- package/config/enable_peek.js.map +1 -0
- package/config/{enableDirectPeek.mjs → enable_peek.mjs} +5 -3
- package/config/enable_peek.mjs.map +1 -0
- package/helpers/fetch.d.ts +4 -3
- package/helpers/fetch.js.map +1 -1
- package/helpers/fetch.mjs.map +1 -1
- package/helpers/pageHash.js.map +1 -1
- package/helpers/pageHash.mjs.map +1 -1
- package/helpers/pageHashParams.js.map +1 -1
- package/helpers/pageHashParams.mjs.map +1 -1
- package/helpers/time.d.ts +2 -2
- package/helpers/time.js.map +1 -1
- package/helpers/time.mjs.map +1 -1
- package/history.js +2 -2
- package/history.js.map +1 -1
- package/history.mjs +3 -3
- package/history.mjs.map +1 -1
- package/index.d.ts +30 -9
- package/index.js +877 -661
- package/index.js.map +1 -1
- package/index.mjs +874 -658
- package/index.mjs.map +1 -1
- package/package.json +22 -25
- package/persist-plugins/async-storage.d.ts +3 -3
- package/persist-plugins/async-storage.js +8 -7
- package/persist-plugins/async-storage.js.map +1 -1
- package/persist-plugins/async-storage.mjs +9 -8
- package/persist-plugins/async-storage.mjs.map +1 -1
- package/persist-plugins/fetch.js.map +1 -1
- package/persist-plugins/fetch.mjs.map +1 -1
- package/persist-plugins/firebase.d.ts +1 -1
- package/persist-plugins/firebase.js +12 -11
- package/persist-plugins/firebase.js.map +1 -1
- package/persist-plugins/firebase.mjs +13 -12
- package/persist-plugins/firebase.mjs.map +1 -1
- package/persist-plugins/indexeddb.d.ts +10 -10
- package/persist-plugins/indexeddb.js +2 -2
- package/persist-plugins/indexeddb.js.map +1 -1
- package/persist-plugins/indexeddb.mjs +2 -2
- package/persist-plugins/indexeddb.mjs.map +1 -1
- package/persist-plugins/local-storage.d.ts +3 -4
- package/persist-plugins/local-storage.js +19 -7
- package/persist-plugins/local-storage.js.map +1 -1
- package/persist-plugins/local-storage.mjs +20 -9
- package/persist-plugins/local-storage.mjs.map +1 -1
- package/persist-plugins/mmkv.d.ts +8 -8
- package/persist-plugins/mmkv.js +5 -4
- package/persist-plugins/mmkv.js.map +1 -1
- package/persist-plugins/mmkv.mjs +6 -5
- package/persist-plugins/mmkv.mjs.map +1 -1
- package/persist-plugins/query.js.map +1 -1
- package/persist-plugins/query.mjs.map +1 -1
- package/persist.d.ts +2 -14
- package/persist.js +1250 -268
- package/persist.js.map +1 -1
- package/persist.mjs +1250 -269
- package/persist.mjs.map +1 -1
- package/react-hooks/createObservableHook.js +1 -1
- package/react-hooks/createObservableHook.js.map +1 -1
- package/react-hooks/createObservableHook.mjs +1 -1
- package/react-hooks/createObservableHook.mjs.map +1 -1
- package/react-hooks/useFetch.d.ts +4 -3
- package/react-hooks/useFetch.js.map +1 -1
- package/react-hooks/useFetch.mjs.map +1 -1
- package/react-hooks/useHover.js.map +1 -1
- package/react-hooks/useHover.mjs.map +1 -1
- package/react-hooks/useMeasure.js.map +1 -1
- package/react-hooks/useMeasure.mjs.map +1 -1
- package/react-hooks/useObservableNextRouter.js.map +1 -1
- package/react-hooks/useObservableNextRouter.mjs.map +1 -1
- package/react-hooks/useObservableQuery.js.map +1 -1
- package/react-hooks/useObservableQuery.mjs.map +1 -1
- package/react-hooks/usePersistedObservable.d.ts +5 -3
- package/react-hooks/usePersistedObservable.js +5 -2
- package/react-hooks/usePersistedObservable.js.map +1 -1
- package/react-hooks/usePersistedObservable.mjs +5 -2
- package/react-hooks/usePersistedObservable.mjs.map +1 -1
- package/react.js +61 -75
- package/react.js.map +1 -1
- package/react.mjs +61 -75
- package/react.mjs.map +1 -1
- package/src/ObservableObject.ts +1184 -0
- package/src/ObservablePrimitive.ts +62 -0
- package/src/babel/index.ts +70 -0
- package/src/batching.ts +372 -0
- package/src/computed.ts +16 -0
- package/src/config/enable$get.ts +30 -0
- package/src/config/enableReactComponents.ts +26 -0
- package/src/config/enableReactNativeComponents.ts +102 -0
- package/src/config/enableReactTracking.ts +60 -0
- package/src/config/enableReactUse.ts +23 -0
- package/src/config/enable_peek.ts +31 -0
- package/src/config.ts +47 -0
- package/src/createObservable.ts +46 -0
- package/src/event.ts +26 -0
- package/src/globals.ts +224 -0
- package/src/helpers/fetch.ts +26 -0
- package/src/helpers/pageHash.ts +41 -0
- package/src/helpers/pageHashParams.ts +55 -0
- package/src/helpers/time.ts +30 -0
- package/src/helpers.ts +221 -0
- package/src/history/trackHistory.ts +29 -0
- package/src/is.ts +56 -0
- package/src/linked.ts +6 -0
- package/src/observable.ts +32 -0
- package/src/observableInterfaces.ts +165 -0
- package/src/observableTypes.ts +221 -0
- package/src/observe.ts +89 -0
- package/src/onChange.ts +136 -0
- package/src/persist/configureObservablePersistence.ts +7 -0
- package/src/persist/fieldTransformer.ts +149 -0
- package/src/persist/observablePersistRemoteFunctionsAdapter.ts +39 -0
- package/src/persist/persistObservable.ts +1029 -0
- package/src/persist-plugins/async-storage.ts +102 -0
- package/src/persist-plugins/fetch.ts +33 -0
- package/src/persist-plugins/firebase.ts +1050 -0
- package/src/persist-plugins/indexeddb.ts +433 -0
- package/src/persist-plugins/local-storage.ts +90 -0
- package/src/persist-plugins/mmkv.ts +90 -0
- package/src/persist-plugins/query.ts +133 -0
- package/src/persistTypes.ts +226 -0
- package/src/proxy.ts +28 -0
- package/src/react/Computed.tsx +7 -0
- package/src/react/For.tsx +116 -0
- package/src/react/Memo.tsx +4 -0
- package/src/react/Reactive.tsx +53 -0
- package/src/react/Show.tsx +33 -0
- package/src/react/Switch.tsx +43 -0
- package/src/react/react-globals.ts +3 -0
- package/src/react/{reactInterfaces.d.ts → reactInterfaces.ts} +15 -7
- package/src/react/reactive-observer.tsx +210 -0
- package/src/react/useComputed.ts +36 -0
- package/src/react/useEffectOnce.ts +41 -0
- package/src/react/useIsMounted.ts +16 -0
- package/src/react/useMount.ts +15 -0
- package/src/react/useObservable.ts +24 -0
- package/src/react/useObservableReducer.ts +52 -0
- package/src/react/useObservableState.ts +30 -0
- package/src/react/useObserve.ts +54 -0
- package/src/react/useObserveEffect.ts +40 -0
- package/src/react/usePauseProvider.tsx +13 -0
- package/src/react/useSelector.ts +167 -0
- package/src/react/useUnmount.ts +8 -0
- package/src/react/useWhen.ts +9 -0
- package/src/react-hooks/createObservableHook.ts +53 -0
- package/src/react-hooks/useFetch.ts +16 -0
- package/src/react-hooks/useHover.ts +40 -0
- package/src/react-hooks/useMeasure.ts +48 -0
- package/src/react-hooks/useObservableNextRouter.ts +137 -0
- package/src/react-hooks/useObservableQuery.ts +205 -0
- package/src/react-hooks/usePersistedObservable.ts +24 -0
- package/src/retry.ts +69 -0
- package/src/setupTracking.ts +26 -0
- package/src/sync/activateSyncedNode.ts +146 -0
- package/src/sync/configureObservableSync.ts +7 -0
- package/src/sync/syncHelpers.ts +15 -0
- package/src/sync/syncObservable.ts +989 -0
- package/src/sync/syncObservableAdapter.ts +30 -0
- package/src/sync/synced.ts +20 -0
- package/src/sync-plugins/fetch.ts +42 -0
- package/src/syncTypes.ts +163 -0
- package/src/trace/traceHelpers.ts +11 -0
- package/src/trace/useTraceListeners.ts +34 -0
- package/src/trace/useTraceUpdates.ts +24 -0
- package/src/trace/useVerifyNotTracking.ts +33 -0
- package/src/trace/useVerifyOneRender.ts +10 -0
- package/src/trackSelector.ts +52 -0
- package/src/tracking.ts +43 -0
- package/src/types/babel.d.ts +12 -0
- package/src/when.ts +70 -0
- package/sync-plugins/fetch.d.ts +11 -0
- package/sync-plugins/fetch.js +24 -0
- package/sync-plugins/fetch.js.map +1 -0
- package/sync-plugins/fetch.mjs +22 -0
- package/sync-plugins/fetch.mjs.map +1 -0
- package/sync.d.ts +8 -0
- package/sync.js +919 -0
- package/sync.js.map +1 -0
- package/sync.mjs +912 -0
- package/sync.mjs.map +1 -0
- package/trace.js +13 -10
- package/trace.js.map +1 -1
- package/trace.mjs +11 -8
- package/trace.mjs.map +1 -1
- package/types/babel.d.ts +3 -3
- package/config/enableDirectAccess.d.ts +0 -7
- package/config/enableDirectAccess.js +0 -25
- package/config/enableDirectAccess.js.map +0 -1
- package/config/enableDirectAccess.mjs +0 -23
- package/config/enableDirectAccess.mjs.map +0 -1
- package/config/enableDirectPeek.d.ts +0 -7
- package/config/enableDirectPeek.js.map +0 -1
- package/config/enableDirectPeek.mjs.map +0 -1
- package/config/enableReactDirectRender.d.ts +0 -2
- package/config/enableReactDirectRender.js +0 -78
- package/config/enableReactDirectRender.js.map +0 -1
- package/config/enableReactDirectRender.mjs +0 -75
- package/config/enableReactDirectRender.mjs.map +0 -1
- package/src/ObservableObject.d.ts +0 -14
- package/src/ObservablePrimitive.d.ts +0 -7
- package/src/babel/index.d.ts +0 -17
- package/src/batching.d.ts +0 -6
- package/src/computed.d.ts +0 -4
- package/src/config/enableDirectAccess.d.ts +0 -7
- package/src/config/enableDirectPeek.d.ts +0 -7
- package/src/config/enableReactComponents.d.ts +0 -7
- package/src/config/enableReactDirectRender.d.ts +0 -2
- package/src/config/enableReactNativeComponents.d.ts +0 -20
- package/src/config/enableReactTracking.d.ts +0 -15
- package/src/config/enableReactUse.d.ts +0 -7
- package/src/config.d.ts +0 -8
- package/src/createObservable.d.ts +0 -2
- package/src/event.d.ts +0 -2
- package/src/globals.d.ts +0 -32
- package/src/helpers/fetch.d.ts +0 -6
- package/src/helpers/pageHash.d.ts +0 -7
- package/src/helpers/pageHashParams.d.ts +0 -7
- package/src/helpers/time.d.ts +0 -3
- package/src/helpers.d.ts +0 -13
- package/src/history/trackHistory.d.ts +0 -4
- package/src/is.d.ts +0 -10
- package/src/observable.d.ts +0 -16
- package/src/observableInterfaces.d.ts +0 -456
- package/src/observe.d.ts +0 -6
- package/src/onChange.d.ts +0 -7
- package/src/persist/configureObservablePersistence.d.ts +0 -3
- package/src/persist/fieldTransformer.d.ts +0 -8
- package/src/persist/observablePersistRemoteFunctionsAdapter.d.ts +0 -2
- package/src/persist/persistActivateNode.d.ts +0 -1
- package/src/persist/persistHelpers.d.ts +0 -1
- package/src/persist/persistObservable.d.ts +0 -25
- package/src/persist-plugins/async-storage.d.ts +0 -14
- package/src/persist-plugins/fetch.d.ts +0 -10
- package/src/persist-plugins/firebase.d.ts +0 -51
- package/src/persist-plugins/indexeddb.d.ts +0 -25
- package/src/persist-plugins/local-storage.d.ts +0 -21
- package/src/persist-plugins/mmkv.d.ts +0 -14
- package/src/persist-plugins/query.d.ts +0 -18
- package/src/proxy.d.ts +0 -5
- package/src/react/Computed.d.ts +0 -5
- package/src/react/For.d.ts +0 -15
- package/src/react/Memo.d.ts +0 -3
- package/src/react/Reactive.d.ts +0 -9
- package/src/react/Show.d.ts +0 -18
- package/src/react/Switch.d.ts +0 -14
- package/src/react/react-globals.d.ts +0 -3
- package/src/react/reactive-observer.d.ts +0 -14
- package/src/react/useComputed.d.ts +0 -5
- package/src/react/useEffectOnce.d.ts +0 -1
- package/src/react/useIsMounted.d.ts +0 -2
- package/src/react/useMount.d.ts +0 -2
- package/src/react/useObservable.d.ts +0 -9
- package/src/react/useObservableReducer.d.ts +0 -7
- package/src/react/useObservableState.d.ts +0 -2
- package/src/react/useObserve.d.ts +0 -4
- package/src/react/useObserveEffect.d.ts +0 -4
- package/src/react/usePauseProvider.d.ts +0 -8
- package/src/react/useSelector.d.ts +0 -3
- package/src/react/useUnmount.d.ts +0 -2
- package/src/react/useWhen.d.ts +0 -3
- package/src/react-hooks/createObservableHook.d.ts +0 -2
- package/src/react-hooks/useFetch.d.ts +0 -6
- package/src/react-hooks/useHover.d.ts +0 -3
- package/src/react-hooks/useMeasure.d.ts +0 -6
- package/src/react-hooks/useObservableNextRouter.d.ts +0 -33
- package/src/react-hooks/useObservableQuery.d.ts +0 -6
- package/src/react-hooks/usePersistedObservable.d.ts +0 -11
- package/src/retry.d.ts +0 -9
- package/src/setupTracking.d.ts +0 -2
- package/src/trace/traceHelpers.d.ts +0 -2
- package/src/trace/useTraceListeners.d.ts +0 -1
- package/src/trace/useTraceUpdates.d.ts +0 -1
- package/src/trace/useVerifyNotTracking.d.ts +0 -1
- package/src/trace/useVerifyOneRender.d.ts +0 -1
- package/src/trackSelector.d.ts +0 -7
- package/src/tracking.d.ts +0 -13
- package/src/when.d.ts +0 -3
|
@@ -0,0 +1,221 @@
|
|
|
1
|
+
import type { GetOptions, ListenerFn, TrackingType } from './observableInterfaces';
|
|
2
|
+
|
|
3
|
+
type Primitive = string | number | boolean | symbol | bigint | undefined | null | Date;
|
|
4
|
+
type ArrayOverrideFnNames =
|
|
5
|
+
| 'find'
|
|
6
|
+
| 'every'
|
|
7
|
+
| 'some'
|
|
8
|
+
| 'filter'
|
|
9
|
+
| 'reduce'
|
|
10
|
+
| 'reduceRight'
|
|
11
|
+
| 'forEach'
|
|
12
|
+
| 'map'
|
|
13
|
+
| 'sort';
|
|
14
|
+
|
|
15
|
+
type RemoveIndex<T> = {
|
|
16
|
+
[K in keyof T as string extends K ? never : number extends K ? never : K]: T[K];
|
|
17
|
+
};
|
|
18
|
+
|
|
19
|
+
type BuiltIns = String | Boolean | Number | Date | Error | RegExp | Array<any> | Function | Promise<any>;
|
|
20
|
+
|
|
21
|
+
type IsUserDefinedObject<T> =
|
|
22
|
+
// Only objects that are not function or arrays or instances of BuiltIns.
|
|
23
|
+
T extends Function | BuiltIns | any[] ? false : T extends object ? true : false;
|
|
24
|
+
|
|
25
|
+
export type RemoveObservables<T> = T extends ImmutableObservableBase<infer t>
|
|
26
|
+
? t
|
|
27
|
+
: IsUserDefinedObject<T> extends true
|
|
28
|
+
? {
|
|
29
|
+
[K in keyof T]: RemoveObservables<T[K]>;
|
|
30
|
+
}
|
|
31
|
+
: T;
|
|
32
|
+
|
|
33
|
+
interface ObservableArray<T, U>
|
|
34
|
+
extends ObservablePrimitive<T>,
|
|
35
|
+
Pick<Array<Observable<U>>, ArrayOverrideFnNames>,
|
|
36
|
+
Omit<RemoveIndex<Array<U>>, ArrayOverrideFnNames> {}
|
|
37
|
+
|
|
38
|
+
interface ObservableObjectFns<T> {
|
|
39
|
+
assign(value: Partial<T>): Observable<T>;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
interface ObservableObjectFunctions<T = Record<string, any>> extends ObservablePrimitive<T>, ObservableObjectFns<T> {}
|
|
43
|
+
|
|
44
|
+
type ObservableMap<T extends Map<any, any> | WeakMap<any, any>> = Omit<T, 'get' | 'size'> &
|
|
45
|
+
Omit<ObservablePrimitive<T>, 'get' | 'size'> & {
|
|
46
|
+
get(key: Parameters<T['get']>[0]): Observable<Parameters<T['set']>[1]>;
|
|
47
|
+
get(): T;
|
|
48
|
+
size: ImmutableObservableBase<number>;
|
|
49
|
+
};
|
|
50
|
+
|
|
51
|
+
type ObservableSet<T extends Set<any> | WeakSet<any>> = Omit<T, 'size'> &
|
|
52
|
+
Omit<ObservablePrimitive<T>, 'size'> & { size: ImmutableObservableBase<number> };
|
|
53
|
+
|
|
54
|
+
interface ObservableBoolean extends ObservablePrimitive<boolean> {
|
|
55
|
+
toggle(): void;
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
interface ObservablePrimitive<T> extends ImmutableObservableBase<T>, MutableObservableBase<T> {}
|
|
59
|
+
type ObservableAny = Partial<ObservableObjectFns<any>> & ObservablePrimitive<any> & Record<string, any>;
|
|
60
|
+
|
|
61
|
+
interface ImmutableObservableSimple<T> {
|
|
62
|
+
peek(): T;
|
|
63
|
+
get(trackingType?: any): any;
|
|
64
|
+
onChange(cb: ListenerFn<any>, options?: any): () => void;
|
|
65
|
+
}
|
|
66
|
+
interface ImmutableObservableBase<T> extends ImmutableObservableSimple<T> {
|
|
67
|
+
peek(): RemoveObservables<T>;
|
|
68
|
+
peek(): T; // This is just to match the Simple base type
|
|
69
|
+
get(trackingType?: TrackingType | GetOptions): RemoveObservables<T>;
|
|
70
|
+
onChange(
|
|
71
|
+
cb: ListenerFn<T>,
|
|
72
|
+
options?: { trackingType?: TrackingType; initial?: boolean; immediate?: boolean; noArgs?: boolean },
|
|
73
|
+
): () => void;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
interface MutableObservableSimple {
|
|
77
|
+
set(value: any): void;
|
|
78
|
+
delete(): void;
|
|
79
|
+
}
|
|
80
|
+
interface MutableObservableBase<T> extends MutableObservableSimple {
|
|
81
|
+
set(value: (prev: RemoveObservables<T>) => RemoveObservables<T>): void;
|
|
82
|
+
set(value: RecursiveValueOrFunction<T>): void;
|
|
83
|
+
set(value: Promise<RemoveObservables<T>>): void;
|
|
84
|
+
set(value: RemoveObservables<T>): void;
|
|
85
|
+
set(value: Observable<RemoveObservables<T>>): void;
|
|
86
|
+
delete(): void;
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
type UndefinedIf<T, U> = U extends true ? T | undefined : T;
|
|
90
|
+
|
|
91
|
+
type IsNullable<T> = undefined extends T ? true : null extends T ? true : false;
|
|
92
|
+
|
|
93
|
+
type NonObservable = Function | Observable;
|
|
94
|
+
type NonObservableKeys<T> = {
|
|
95
|
+
[K in keyof T]-?: IsStrictAny<T[K]> extends true
|
|
96
|
+
? never
|
|
97
|
+
: T[K] extends undefined | null
|
|
98
|
+
? never
|
|
99
|
+
: NonNullable<T[K]> extends NonObservable
|
|
100
|
+
? K
|
|
101
|
+
: never;
|
|
102
|
+
}[keyof T];
|
|
103
|
+
type ObservableProps<T> = NonObservableKeys<NonNullable<T>> extends never
|
|
104
|
+
? T
|
|
105
|
+
: RestoreNullability<T, Omit<NonNullable<T>, NonObservableKeys<NonNullable<T>>>>;
|
|
106
|
+
|
|
107
|
+
type NonObservableProps<T> = RestoreNullability<
|
|
108
|
+
T,
|
|
109
|
+
NullablePropsIf<Pick<NonNullable<T>, NonObservableKeys<NonNullable<T>>>, IsNullable<T>>
|
|
110
|
+
>;
|
|
111
|
+
type NullablePropsIf<T, U> = {
|
|
112
|
+
[K in keyof T]: UndefinedIf<T[K], U>;
|
|
113
|
+
};
|
|
114
|
+
|
|
115
|
+
type RestoreNullability<Source, Target> = IsNullable<Source> extends true
|
|
116
|
+
? Target | Extract<Source, null | undefined>
|
|
117
|
+
: Target;
|
|
118
|
+
|
|
119
|
+
type ObservableChildren<T, Nullable = IsNullable<T>> = {
|
|
120
|
+
[K in keyof T]-?: Observable<UndefinedIf<T[K], Nullable>>;
|
|
121
|
+
};
|
|
122
|
+
type ObservableFunctionChildren<T> = {
|
|
123
|
+
[K in keyof T]-?: T[K] extends Observable
|
|
124
|
+
? T[K]
|
|
125
|
+
: T[K] extends (key: infer Key extends string) => Promise<infer t> | infer t
|
|
126
|
+
? HasOneStringParam<T[K]> extends true
|
|
127
|
+
? Observable<Record<Key, t>> & T[K]
|
|
128
|
+
: t extends void
|
|
129
|
+
? T[K]
|
|
130
|
+
: t extends Observable
|
|
131
|
+
? t
|
|
132
|
+
: Observable<t> & (() => t)
|
|
133
|
+
: T[K];
|
|
134
|
+
};
|
|
135
|
+
|
|
136
|
+
type IsStrictAny<T> = 0 extends 1 & T ? true : false;
|
|
137
|
+
|
|
138
|
+
export interface ObservableState {
|
|
139
|
+
isLoaded: boolean;
|
|
140
|
+
error?: Error;
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
type ObservableObject<T> = ObservableObjectFunctions<ObservableProps<T> & NonObservableProps<T>> &
|
|
144
|
+
ObservableChildren<ObservableProps<T>> &
|
|
145
|
+
ObservableFunctionChildren<NonObservableProps<T>>;
|
|
146
|
+
|
|
147
|
+
type ObservableFunction<T> = T extends () => infer t ? t | (() => t) : T;
|
|
148
|
+
|
|
149
|
+
// Check if the function type T has one lookup parameter
|
|
150
|
+
type HasOneStringParam<T> = T extends (...args: infer P) => any
|
|
151
|
+
? P extends { length: 1 }
|
|
152
|
+
? P[0] extends string | ObservablePrimitive<string>
|
|
153
|
+
? true
|
|
154
|
+
: false
|
|
155
|
+
: false
|
|
156
|
+
: false;
|
|
157
|
+
|
|
158
|
+
// : [T] extends [(key: infer K extends string) => infer t]
|
|
159
|
+
// ? // ? HasParams<T> extends true ? Observable<Record<K, t>>
|
|
160
|
+
type ObservableNode<T, NT = NonNullable<T>> = [NT] extends [never] // means that T is ONLY undefined or null
|
|
161
|
+
? ObservablePrimitive<T>
|
|
162
|
+
: IsStrictAny<T> extends true
|
|
163
|
+
? ObservableAny
|
|
164
|
+
: [T] extends [Promise<infer t>]
|
|
165
|
+
? ObservableNode<t>
|
|
166
|
+
: [T] extends [(key: infer K extends string) => infer t]
|
|
167
|
+
? [t] extends [ImmutableObservableBase<any>]
|
|
168
|
+
? HasOneStringParam<T> extends true
|
|
169
|
+
? Observable<Record<K, t>>
|
|
170
|
+
: t
|
|
171
|
+
: HasOneStringParam<T> extends true
|
|
172
|
+
? Observable<Record<K, t>> & T
|
|
173
|
+
: Observable<ObservableFunction<t>>
|
|
174
|
+
: [NT] extends [ImmutableObservableBase<any>]
|
|
175
|
+
? NT
|
|
176
|
+
: [NT] extends [Primitive]
|
|
177
|
+
? [NT] extends [boolean]
|
|
178
|
+
? ObservableBoolean
|
|
179
|
+
: ObservablePrimitive<T>
|
|
180
|
+
: NT extends Map<any, any> | WeakMap<any, any>
|
|
181
|
+
? ObservableMap<NT>
|
|
182
|
+
: NT extends Set<infer U>
|
|
183
|
+
? ObservableSet<Set<UndefinedIf<U, IsNullable<T>>>>
|
|
184
|
+
: NT extends WeakSet<any>
|
|
185
|
+
? ObservableSet<NT> // TODO what to do here with nullable? WeakKey is type object | symbol
|
|
186
|
+
: NT extends Array<infer U>
|
|
187
|
+
? ObservableArray<T, U> & ObservableChildren<T>
|
|
188
|
+
: ObservableObject<T> & {};
|
|
189
|
+
|
|
190
|
+
// Note: The {} makes intellisense display observables as Observable instead of all the subtypes
|
|
191
|
+
type Observable<T = any> = ObservableNode<T> & {};
|
|
192
|
+
|
|
193
|
+
type ObservableParam<T = any> = ImmutableObservableSimple<T> & MutableObservableSimple;
|
|
194
|
+
|
|
195
|
+
// Allow input types to have functions in them
|
|
196
|
+
type ValueOrFunction<T> = T extends Function ? T : T | ImmutableObservableBase<T> | Promise<T> | (() => T | Promise<T>);
|
|
197
|
+
type ValueOrFunctionKeys<T> = {
|
|
198
|
+
[K in keyof T]: RecursiveValueOrFunction<T[K]>;
|
|
199
|
+
};
|
|
200
|
+
|
|
201
|
+
type RecursiveValueOrFunction<T> = T extends Function
|
|
202
|
+
? T
|
|
203
|
+
: T extends object
|
|
204
|
+
?
|
|
205
|
+
| ((key: string) => any)
|
|
206
|
+
| Promise<ValueOrFunctionKeys<T>>
|
|
207
|
+
| ValueOrFunctionKeys<T>
|
|
208
|
+
| ImmutableObservableBase<T>
|
|
209
|
+
| (() => T | Promise<T> | ValueOrFunctionKeys<T> | Promise<ValueOrFunctionKeys<T>> | Observable<T>)
|
|
210
|
+
: ValueOrFunction<T>;
|
|
211
|
+
|
|
212
|
+
export type {
|
|
213
|
+
// TODO: how to make these internal somehow?
|
|
214
|
+
ImmutableObservableBase,
|
|
215
|
+
Observable,
|
|
216
|
+
ObservableBoolean,
|
|
217
|
+
ObservableObject,
|
|
218
|
+
ObservablePrimitive,
|
|
219
|
+
ObservableParam,
|
|
220
|
+
RecursiveValueOrFunction,
|
|
221
|
+
};
|
package/src/observe.ts
ADDED
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
import { beginBatch, endBatch } from './batching';
|
|
2
|
+
import { isEvent } from './globals';
|
|
3
|
+
import { isFunction } from './is';
|
|
4
|
+
import { ObserveEvent, ObserveEventCallback, Selector } from './observableInterfaces';
|
|
5
|
+
import { trackSelector } from './trackSelector';
|
|
6
|
+
|
|
7
|
+
export interface ObserveOptions {
|
|
8
|
+
immediate?: boolean; // Ignore batching and run immediately
|
|
9
|
+
/* @internal */
|
|
10
|
+
fromComputed?: boolean;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export function observe<T>(run: (e: ObserveEvent<T>) => T | void, options?: ObserveOptions): () => void;
|
|
14
|
+
export function observe<T>(
|
|
15
|
+
selector: Selector<T> | ((e: ObserveEvent<T>) => any),
|
|
16
|
+
reaction?: (e: ObserveEventCallback<T>) => any,
|
|
17
|
+
options?: ObserveOptions,
|
|
18
|
+
): () => void;
|
|
19
|
+
export function observe<T>(
|
|
20
|
+
selectorOrRun: Selector<T> | ((e: ObserveEvent<T>) => any),
|
|
21
|
+
reactionOrOptions?: ((e: ObserveEventCallback<T>) => any) | ObserveOptions,
|
|
22
|
+
options?: ObserveOptions,
|
|
23
|
+
) {
|
|
24
|
+
let reaction: (e: ObserveEventCallback<T>) => any;
|
|
25
|
+
if (isFunction(reactionOrOptions)) {
|
|
26
|
+
reaction = reactionOrOptions;
|
|
27
|
+
} else {
|
|
28
|
+
options = reactionOrOptions;
|
|
29
|
+
}
|
|
30
|
+
let dispose: (() => void) | undefined;
|
|
31
|
+
const e: ObserveEventCallback<T> = { num: 0 } as ObserveEventCallback<T>;
|
|
32
|
+
// Wrap it in a function so it doesn't pass all the arguments to run()
|
|
33
|
+
const update = function () {
|
|
34
|
+
if (e.onCleanup) {
|
|
35
|
+
e.onCleanup();
|
|
36
|
+
e.onCleanup = undefined;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
// Run in a batch so changes don't happen until we're done tracking here
|
|
40
|
+
beginBatch();
|
|
41
|
+
|
|
42
|
+
// Run the function/selector
|
|
43
|
+
delete e.value;
|
|
44
|
+
|
|
45
|
+
// Dispose listeners from previous run
|
|
46
|
+
dispose?.();
|
|
47
|
+
|
|
48
|
+
const { dispose: _dispose, value, nodes } = trackSelector(selectorOrRun, update, e, options);
|
|
49
|
+
dispose = _dispose;
|
|
50
|
+
|
|
51
|
+
e.value = value;
|
|
52
|
+
e.nodes = nodes;
|
|
53
|
+
e.refresh = update;
|
|
54
|
+
|
|
55
|
+
if (e.onCleanupReaction) {
|
|
56
|
+
e.onCleanupReaction();
|
|
57
|
+
e.onCleanupReaction = undefined;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
endBatch();
|
|
61
|
+
|
|
62
|
+
// Call the reaction if there is one and the value changed
|
|
63
|
+
if (
|
|
64
|
+
reaction &&
|
|
65
|
+
(options?.fromComputed ||
|
|
66
|
+
((e.num > 0 || !isEvent(selectorOrRun as any)) &&
|
|
67
|
+
(e.previous !== e.value || typeof e.value === 'object')))
|
|
68
|
+
) {
|
|
69
|
+
reaction(e);
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
// Update the previous value
|
|
73
|
+
e.previous = e.value;
|
|
74
|
+
|
|
75
|
+
// Increment the counter
|
|
76
|
+
e.num++;
|
|
77
|
+
};
|
|
78
|
+
|
|
79
|
+
update();
|
|
80
|
+
|
|
81
|
+
// Return function calling dispose because dispose may be changed in update()
|
|
82
|
+
return () => {
|
|
83
|
+
e.onCleanup?.();
|
|
84
|
+
e.onCleanup = undefined;
|
|
85
|
+
e.onCleanupReaction?.();
|
|
86
|
+
e.onCleanupReaction = undefined;
|
|
87
|
+
dispose?.();
|
|
88
|
+
};
|
|
89
|
+
}
|
package/src/onChange.ts
ADDED
|
@@ -0,0 +1,136 @@
|
|
|
1
|
+
import { isArray } from './is';
|
|
2
|
+
import { getNodeValue } from './globals';
|
|
3
|
+
import type { ListenerFn, ListenerParams, NodeValue, NodeValueListener, TrackingType } from './observableInterfaces';
|
|
4
|
+
|
|
5
|
+
export function onChange(
|
|
6
|
+
node: NodeValue,
|
|
7
|
+
callback: ListenerFn,
|
|
8
|
+
options: { trackingType?: TrackingType; initial?: boolean; immediate?: boolean; noArgs?: boolean } = {},
|
|
9
|
+
fromLinks?: Set<NodeValue>,
|
|
10
|
+
): () => void {
|
|
11
|
+
const { initial, immediate, noArgs } = options;
|
|
12
|
+
const { trackingType } = options;
|
|
13
|
+
|
|
14
|
+
let listeners = immediate ? node.listenersImmediate : node.listeners;
|
|
15
|
+
if (!listeners) {
|
|
16
|
+
listeners = new Set();
|
|
17
|
+
if (immediate) {
|
|
18
|
+
node.listenersImmediate = listeners;
|
|
19
|
+
} else {
|
|
20
|
+
node.listeners = listeners;
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
const listener: NodeValueListener = {
|
|
25
|
+
listener: callback,
|
|
26
|
+
track: trackingType,
|
|
27
|
+
noArgs,
|
|
28
|
+
};
|
|
29
|
+
|
|
30
|
+
listeners.add(listener);
|
|
31
|
+
|
|
32
|
+
if (initial) {
|
|
33
|
+
const value = getNodeValue(node);
|
|
34
|
+
callback({
|
|
35
|
+
value,
|
|
36
|
+
loading: true,
|
|
37
|
+
remote: false,
|
|
38
|
+
changes: [
|
|
39
|
+
{
|
|
40
|
+
path: [],
|
|
41
|
+
pathTypes: [],
|
|
42
|
+
prevAtPath: value,
|
|
43
|
+
valueAtPath: value,
|
|
44
|
+
},
|
|
45
|
+
],
|
|
46
|
+
getPrevious: () => undefined,
|
|
47
|
+
});
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
let extraDisposes: (() => void)[];
|
|
51
|
+
|
|
52
|
+
function addLinkedNodeListeners(childNode: NodeValue, cb: ListenerFn = callback, from?: NodeValue) {
|
|
53
|
+
// Don't add listeners for the same node more than once
|
|
54
|
+
if (!fromLinks?.has(childNode)) {
|
|
55
|
+
fromLinks ||= new Set();
|
|
56
|
+
fromLinks.add(from || node);
|
|
57
|
+
cb ||= callback;
|
|
58
|
+
const childOptions: Parameters<typeof onChange>[2] = {
|
|
59
|
+
trackingType: true,
|
|
60
|
+
...options,
|
|
61
|
+
};
|
|
62
|
+
// onChange for the linked node
|
|
63
|
+
extraDisposes = [...(extraDisposes || []), onChange(childNode, cb as ListenerFn, childOptions, fromLinks)];
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
// Add listeners for linked to nodes
|
|
68
|
+
if (node.linkedToNode) {
|
|
69
|
+
addLinkedNodeListeners(node.linkedToNode);
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
// Add listeners for linked from nodes
|
|
73
|
+
node.linkedFromNodes?.forEach((linkedFromNode) => addLinkedNodeListeners(linkedFromNode));
|
|
74
|
+
|
|
75
|
+
// Go up through the parents and add listeners for linked from nodes
|
|
76
|
+
let parent = node.parent;
|
|
77
|
+
let pathParent: string[] = [node!.key!];
|
|
78
|
+
while (parent) {
|
|
79
|
+
if (parent.linkedFromNodes) {
|
|
80
|
+
for (const linkedFromNode of parent.linkedFromNodes) {
|
|
81
|
+
if (!fromLinks?.has(linkedFromNode)) {
|
|
82
|
+
const cb = createCb(linkedFromNode, pathParent, callback);
|
|
83
|
+
addLinkedNodeListeners(linkedFromNode, cb, parent);
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
pathParent = [parent!.key!, ...pathParent];
|
|
88
|
+
parent = parent.parent;
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
return () => {
|
|
92
|
+
listeners!.delete(listener);
|
|
93
|
+
extraDisposes?.forEach((fn) => fn());
|
|
94
|
+
};
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
function createCb(linkedFromNode: NodeValue, path: string[], callback: ListenerFn) {
|
|
98
|
+
// Create a callback for a path that calls it with the current value at the path
|
|
99
|
+
let { valueAtPath: prevAtPath } = getValueAtPath(getNodeValue(linkedFromNode), path);
|
|
100
|
+
|
|
101
|
+
return function ({ value: valueA, loading, remote }: ListenerParams<any>) {
|
|
102
|
+
const { valueAtPath, pathTypes } = getValueAtPath(valueA, path);
|
|
103
|
+
if (valueAtPath !== prevAtPath) {
|
|
104
|
+
callback({
|
|
105
|
+
value: valueAtPath,
|
|
106
|
+
loading,
|
|
107
|
+
remote,
|
|
108
|
+
changes: [
|
|
109
|
+
{
|
|
110
|
+
path,
|
|
111
|
+
pathTypes,
|
|
112
|
+
prevAtPath,
|
|
113
|
+
valueAtPath,
|
|
114
|
+
},
|
|
115
|
+
],
|
|
116
|
+
getPrevious: () => prevAtPath,
|
|
117
|
+
});
|
|
118
|
+
}
|
|
119
|
+
prevAtPath = valueAtPath;
|
|
120
|
+
};
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
function getValueAtPath(
|
|
124
|
+
obj: Record<string, any>,
|
|
125
|
+
path: string[],
|
|
126
|
+
): { valueAtPath: any; pathTypes: ('object' | 'array')[] } {
|
|
127
|
+
let o: Record<string, any> = obj;
|
|
128
|
+
const pathTypes: ('object' | 'array')[] = [];
|
|
129
|
+
for (let i = 0; o && i < path.length; i++) {
|
|
130
|
+
pathTypes.push(isArray(o) ? 'array' : 'object');
|
|
131
|
+
const p = path[i];
|
|
132
|
+
o = (o as any)[p];
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
return { valueAtPath: o, pathTypes };
|
|
136
|
+
}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import type { ObservablePersistenceConfig } from '@legendapp/state';
|
|
2
|
+
|
|
3
|
+
export const observablePersistConfiguration: ObservablePersistenceConfig = {};
|
|
4
|
+
|
|
5
|
+
export function configureObservablePersistence(options?: ObservablePersistenceConfig) {
|
|
6
|
+
Object.assign(observablePersistConfiguration, options);
|
|
7
|
+
}
|
|
@@ -0,0 +1,149 @@
|
|
|
1
|
+
import {
|
|
2
|
+
constructObjectWithPath,
|
|
3
|
+
deconstructObjectWithPath,
|
|
4
|
+
FieldTransforms,
|
|
5
|
+
isArray,
|
|
6
|
+
isObject,
|
|
7
|
+
isString,
|
|
8
|
+
TypeAtPath,
|
|
9
|
+
internal,
|
|
10
|
+
} from '@legendapp/state';
|
|
11
|
+
|
|
12
|
+
const { initializePathType, symbolDelete } = internal;
|
|
13
|
+
|
|
14
|
+
let validateMap: (map: Record<string, any>) => void;
|
|
15
|
+
|
|
16
|
+
export function transformPath(path: string[], pathTypes: TypeAtPath[], map: Record<string, any>): string[] {
|
|
17
|
+
const data: Record<string, any> = {};
|
|
18
|
+
let d: Record<string, any> | null = data;
|
|
19
|
+
for (let i = 0; i < path.length; i++) {
|
|
20
|
+
d = d![path[i]] = i === path.length - 1 ? null : initializePathType(pathTypes[i]);
|
|
21
|
+
}
|
|
22
|
+
let value = transformObject(data, map);
|
|
23
|
+
const pathOut = [];
|
|
24
|
+
for (let i = 0; i < path.length; i++) {
|
|
25
|
+
const key = Object.keys(value)[0];
|
|
26
|
+
pathOut.push(key);
|
|
27
|
+
value = value[key];
|
|
28
|
+
}
|
|
29
|
+
return pathOut;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
export function transformObject(dataIn: Record<string, any>, map: Record<string, any>) {
|
|
33
|
+
if (process.env.NODE_ENV === 'development') {
|
|
34
|
+
validateMap(map);
|
|
35
|
+
}
|
|
36
|
+
let ret = dataIn;
|
|
37
|
+
if (dataIn) {
|
|
38
|
+
if ((dataIn as unknown) === symbolDelete) return dataIn;
|
|
39
|
+
if (isString(dataIn)) {
|
|
40
|
+
return map[dataIn];
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
ret = {};
|
|
44
|
+
|
|
45
|
+
const dict = Object.keys(map).length === 1 && map['_dict'];
|
|
46
|
+
|
|
47
|
+
for (const key in dataIn) {
|
|
48
|
+
let v = dataIn[key];
|
|
49
|
+
|
|
50
|
+
if (dict) {
|
|
51
|
+
ret[key] = transformObject(v, dict);
|
|
52
|
+
} else {
|
|
53
|
+
const mapped = map[key];
|
|
54
|
+
if (mapped === undefined) {
|
|
55
|
+
// Don't transform dateModified if user doesn't want it
|
|
56
|
+
if (key !== '@') {
|
|
57
|
+
ret[key] = v;
|
|
58
|
+
if (process.env.NODE_ENV === 'development' || process.env.NODE_ENV === 'test') {
|
|
59
|
+
console.error('A fatal field transformation error has occurred', key, dataIn, map);
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
} else if (mapped !== null) {
|
|
63
|
+
if (v !== undefined && v !== null) {
|
|
64
|
+
if (map[key + '_val']) {
|
|
65
|
+
const mapChild = map[key + '_val'];
|
|
66
|
+
if (isArray(v)) {
|
|
67
|
+
v = v.map((vChild) => mapChild[vChild]);
|
|
68
|
+
} else {
|
|
69
|
+
v = mapChild[v];
|
|
70
|
+
}
|
|
71
|
+
} else if (map[key + '_arr'] && isArray(v)) {
|
|
72
|
+
const mapChild = map[key + '_arr'];
|
|
73
|
+
v = v.map((vChild) => transformObject(vChild, mapChild));
|
|
74
|
+
} else if (isObject(v)) {
|
|
75
|
+
if (map[key + '_obj']) {
|
|
76
|
+
v = transformObject(v, map[key + '_obj']);
|
|
77
|
+
} else if (map[key + '_dict']) {
|
|
78
|
+
const mapChild = map[key + '_dict'];
|
|
79
|
+
const out: Record<string, any> = {};
|
|
80
|
+
for (const keyChild in v) {
|
|
81
|
+
out[keyChild] = transformObject(v[keyChild], mapChild);
|
|
82
|
+
}
|
|
83
|
+
v = out;
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
ret[mapped] = v;
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
return ret;
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
export function transformObjectWithPath(
|
|
97
|
+
obj: object,
|
|
98
|
+
path: string[],
|
|
99
|
+
pathTypes: TypeAtPath[],
|
|
100
|
+
fieldTransforms: FieldTransforms<any>,
|
|
101
|
+
) {
|
|
102
|
+
const constructed = constructObjectWithPath(path, pathTypes, obj);
|
|
103
|
+
const transformed = transformObject(constructed, fieldTransforms);
|
|
104
|
+
const transformedPath = transformPath(path as string[], pathTypes, fieldTransforms);
|
|
105
|
+
return { path: transformedPath, obj: deconstructObjectWithPath(transformedPath, pathTypes, transformed) };
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
const invertedMaps = new WeakMap();
|
|
109
|
+
|
|
110
|
+
export function invertFieldMap(obj: Record<string, any>) {
|
|
111
|
+
const existing = invertedMaps.get(obj);
|
|
112
|
+
if (existing) return existing;
|
|
113
|
+
|
|
114
|
+
const target: Record<string, any> = {} as any;
|
|
115
|
+
|
|
116
|
+
for (const key in obj) {
|
|
117
|
+
const val = obj[key];
|
|
118
|
+
if (key === '_dict') {
|
|
119
|
+
target[key] = invertFieldMap(val);
|
|
120
|
+
} else if (key.endsWith('_obj') || key.endsWith('_dict') || key.endsWith('_arr') || key.endsWith('_val')) {
|
|
121
|
+
const keyMapped = obj[key.replace(/_obj|_dict|_arr|_val$/, '')];
|
|
122
|
+
const suffix = key.match(/_obj|_dict|_arr|_val$/)![0];
|
|
123
|
+
target[keyMapped + suffix] = invertFieldMap(val);
|
|
124
|
+
} else if (typeof val === 'string') {
|
|
125
|
+
target[val] = key;
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
invertedMaps.set(obj, target);
|
|
129
|
+
|
|
130
|
+
return target;
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
if (process.env.NODE_ENV === 'development') {
|
|
134
|
+
validateMap = function (record: Record<string, any>) {
|
|
135
|
+
const values = Object.values(record).filter((value) => {
|
|
136
|
+
if (isObject(value)) {
|
|
137
|
+
validateMap(value);
|
|
138
|
+
} else {
|
|
139
|
+
return isString(value);
|
|
140
|
+
}
|
|
141
|
+
});
|
|
142
|
+
|
|
143
|
+
const uniques = Array.from(new Set(values));
|
|
144
|
+
if (values.length !== uniques.length) {
|
|
145
|
+
console.error('Field transform map has duplicate values', record, values.length, uniques.length);
|
|
146
|
+
}
|
|
147
|
+
return record;
|
|
148
|
+
};
|
|
149
|
+
}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
import {
|
|
2
|
+
isPromise,
|
|
3
|
+
type ObservablePersistRemoteClass,
|
|
4
|
+
type ObservablePersistRemoteFunctions,
|
|
5
|
+
type ObservablePersistRemoteGetParams,
|
|
6
|
+
} from '@legendapp/state';
|
|
7
|
+
|
|
8
|
+
export function observablePersistRemoteFunctionsAdapter<T = {}>({
|
|
9
|
+
get,
|
|
10
|
+
set,
|
|
11
|
+
}: ObservablePersistRemoteFunctions<T>): ObservablePersistRemoteClass {
|
|
12
|
+
const ret: ObservablePersistRemoteClass = {};
|
|
13
|
+
|
|
14
|
+
if (get) {
|
|
15
|
+
ret.get = (async (params: ObservablePersistRemoteGetParams<T>) => {
|
|
16
|
+
try {
|
|
17
|
+
let value = get(params);
|
|
18
|
+
if (isPromise(value)) {
|
|
19
|
+
value = await value;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
params.onChange({
|
|
23
|
+
value,
|
|
24
|
+
dateModified: params.dateModified,
|
|
25
|
+
lastSync: params.lastSync,
|
|
26
|
+
mode: params.mode,
|
|
27
|
+
});
|
|
28
|
+
params.onGet();
|
|
29
|
+
// eslint-disable-next-line no-empty
|
|
30
|
+
} catch {}
|
|
31
|
+
}) as ObservablePersistRemoteClass['get'];
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
if (set) {
|
|
35
|
+
ret.set = set as ObservablePersistRemoteClass['set'];
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
return ret;
|
|
39
|
+
}
|