@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.
Files changed (298) hide show
  1. package/README.md +4 -2
  2. package/babel.js.map +1 -1
  3. package/config/enable$get.d.ts +8 -0
  4. package/config/enable$get.js +24 -0
  5. package/config/enable$get.js.map +1 -0
  6. package/config/enable$get.mjs +21 -0
  7. package/config/enable$get.mjs.map +1 -0
  8. package/config/enableReactComponents.js.map +1 -1
  9. package/config/enableReactComponents.mjs.map +1 -1
  10. package/config/enableReactNativeComponents.js.map +1 -1
  11. package/config/enableReactNativeComponents.mjs.map +1 -1
  12. package/config/enableReactTracking.d.ts +0 -9
  13. package/config/enableReactTracking.js.map +1 -1
  14. package/config/enableReactTracking.mjs.map +1 -1
  15. package/config/enableReactUse.d.ts +1 -1
  16. package/config/enableReactUse.js +1 -0
  17. package/config/enableReactUse.js.map +1 -1
  18. package/config/enableReactUse.mjs +1 -0
  19. package/config/enableReactUse.mjs.map +1 -1
  20. package/config/enable_peek.d.ts +8 -0
  21. package/config/{enableDirectPeek.js → enable_peek.js} +6 -3
  22. package/config/enable_peek.js.map +1 -0
  23. package/config/{enableDirectPeek.mjs → enable_peek.mjs} +5 -3
  24. package/config/enable_peek.mjs.map +1 -0
  25. package/helpers/fetch.d.ts +4 -3
  26. package/helpers/fetch.js.map +1 -1
  27. package/helpers/fetch.mjs.map +1 -1
  28. package/helpers/pageHash.js.map +1 -1
  29. package/helpers/pageHash.mjs.map +1 -1
  30. package/helpers/pageHashParams.js.map +1 -1
  31. package/helpers/pageHashParams.mjs.map +1 -1
  32. package/helpers/time.d.ts +2 -2
  33. package/helpers/time.js.map +1 -1
  34. package/helpers/time.mjs.map +1 -1
  35. package/history.js +2 -2
  36. package/history.js.map +1 -1
  37. package/history.mjs +3 -3
  38. package/history.mjs.map +1 -1
  39. package/index.d.ts +30 -9
  40. package/index.js +877 -661
  41. package/index.js.map +1 -1
  42. package/index.mjs +874 -658
  43. package/index.mjs.map +1 -1
  44. package/package.json +22 -25
  45. package/persist-plugins/async-storage.d.ts +3 -3
  46. package/persist-plugins/async-storage.js +8 -7
  47. package/persist-plugins/async-storage.js.map +1 -1
  48. package/persist-plugins/async-storage.mjs +9 -8
  49. package/persist-plugins/async-storage.mjs.map +1 -1
  50. package/persist-plugins/fetch.js.map +1 -1
  51. package/persist-plugins/fetch.mjs.map +1 -1
  52. package/persist-plugins/firebase.d.ts +1 -1
  53. package/persist-plugins/firebase.js +12 -11
  54. package/persist-plugins/firebase.js.map +1 -1
  55. package/persist-plugins/firebase.mjs +13 -12
  56. package/persist-plugins/firebase.mjs.map +1 -1
  57. package/persist-plugins/indexeddb.d.ts +10 -10
  58. package/persist-plugins/indexeddb.js +2 -2
  59. package/persist-plugins/indexeddb.js.map +1 -1
  60. package/persist-plugins/indexeddb.mjs +2 -2
  61. package/persist-plugins/indexeddb.mjs.map +1 -1
  62. package/persist-plugins/local-storage.d.ts +3 -4
  63. package/persist-plugins/local-storage.js +19 -7
  64. package/persist-plugins/local-storage.js.map +1 -1
  65. package/persist-plugins/local-storage.mjs +20 -9
  66. package/persist-plugins/local-storage.mjs.map +1 -1
  67. package/persist-plugins/mmkv.d.ts +8 -8
  68. package/persist-plugins/mmkv.js +5 -4
  69. package/persist-plugins/mmkv.js.map +1 -1
  70. package/persist-plugins/mmkv.mjs +6 -5
  71. package/persist-plugins/mmkv.mjs.map +1 -1
  72. package/persist-plugins/query.js.map +1 -1
  73. package/persist-plugins/query.mjs.map +1 -1
  74. package/persist.d.ts +2 -14
  75. package/persist.js +1250 -268
  76. package/persist.js.map +1 -1
  77. package/persist.mjs +1250 -269
  78. package/persist.mjs.map +1 -1
  79. package/react-hooks/createObservableHook.js +1 -1
  80. package/react-hooks/createObservableHook.js.map +1 -1
  81. package/react-hooks/createObservableHook.mjs +1 -1
  82. package/react-hooks/createObservableHook.mjs.map +1 -1
  83. package/react-hooks/useFetch.d.ts +4 -3
  84. package/react-hooks/useFetch.js.map +1 -1
  85. package/react-hooks/useFetch.mjs.map +1 -1
  86. package/react-hooks/useHover.js.map +1 -1
  87. package/react-hooks/useHover.mjs.map +1 -1
  88. package/react-hooks/useMeasure.js.map +1 -1
  89. package/react-hooks/useMeasure.mjs.map +1 -1
  90. package/react-hooks/useObservableNextRouter.js.map +1 -1
  91. package/react-hooks/useObservableNextRouter.mjs.map +1 -1
  92. package/react-hooks/useObservableQuery.js.map +1 -1
  93. package/react-hooks/useObservableQuery.mjs.map +1 -1
  94. package/react-hooks/usePersistedObservable.d.ts +5 -3
  95. package/react-hooks/usePersistedObservable.js +5 -2
  96. package/react-hooks/usePersistedObservable.js.map +1 -1
  97. package/react-hooks/usePersistedObservable.mjs +5 -2
  98. package/react-hooks/usePersistedObservable.mjs.map +1 -1
  99. package/react.js +61 -75
  100. package/react.js.map +1 -1
  101. package/react.mjs +61 -75
  102. package/react.mjs.map +1 -1
  103. package/src/ObservableObject.ts +1184 -0
  104. package/src/ObservablePrimitive.ts +62 -0
  105. package/src/babel/index.ts +70 -0
  106. package/src/batching.ts +372 -0
  107. package/src/computed.ts +16 -0
  108. package/src/config/enable$get.ts +30 -0
  109. package/src/config/enableReactComponents.ts +26 -0
  110. package/src/config/enableReactNativeComponents.ts +102 -0
  111. package/src/config/enableReactTracking.ts +60 -0
  112. package/src/config/enableReactUse.ts +23 -0
  113. package/src/config/enable_peek.ts +31 -0
  114. package/src/config.ts +47 -0
  115. package/src/createObservable.ts +46 -0
  116. package/src/event.ts +26 -0
  117. package/src/globals.ts +224 -0
  118. package/src/helpers/fetch.ts +26 -0
  119. package/src/helpers/pageHash.ts +41 -0
  120. package/src/helpers/pageHashParams.ts +55 -0
  121. package/src/helpers/time.ts +30 -0
  122. package/src/helpers.ts +221 -0
  123. package/src/history/trackHistory.ts +29 -0
  124. package/src/is.ts +56 -0
  125. package/src/linked.ts +6 -0
  126. package/src/observable.ts +32 -0
  127. package/src/observableInterfaces.ts +165 -0
  128. package/src/observableTypes.ts +221 -0
  129. package/src/observe.ts +89 -0
  130. package/src/onChange.ts +136 -0
  131. package/src/persist/configureObservablePersistence.ts +7 -0
  132. package/src/persist/fieldTransformer.ts +149 -0
  133. package/src/persist/observablePersistRemoteFunctionsAdapter.ts +39 -0
  134. package/src/persist/persistObservable.ts +1029 -0
  135. package/src/persist-plugins/async-storage.ts +102 -0
  136. package/src/persist-plugins/fetch.ts +33 -0
  137. package/src/persist-plugins/firebase.ts +1050 -0
  138. package/src/persist-plugins/indexeddb.ts +433 -0
  139. package/src/persist-plugins/local-storage.ts +90 -0
  140. package/src/persist-plugins/mmkv.ts +90 -0
  141. package/src/persist-plugins/query.ts +133 -0
  142. package/src/persistTypes.ts +226 -0
  143. package/src/proxy.ts +28 -0
  144. package/src/react/Computed.tsx +7 -0
  145. package/src/react/For.tsx +116 -0
  146. package/src/react/Memo.tsx +4 -0
  147. package/src/react/Reactive.tsx +53 -0
  148. package/src/react/Show.tsx +33 -0
  149. package/src/react/Switch.tsx +43 -0
  150. package/src/react/react-globals.ts +3 -0
  151. package/src/react/{reactInterfaces.d.ts → reactInterfaces.ts} +15 -7
  152. package/src/react/reactive-observer.tsx +210 -0
  153. package/src/react/useComputed.ts +36 -0
  154. package/src/react/useEffectOnce.ts +41 -0
  155. package/src/react/useIsMounted.ts +16 -0
  156. package/src/react/useMount.ts +15 -0
  157. package/src/react/useObservable.ts +24 -0
  158. package/src/react/useObservableReducer.ts +52 -0
  159. package/src/react/useObservableState.ts +30 -0
  160. package/src/react/useObserve.ts +54 -0
  161. package/src/react/useObserveEffect.ts +40 -0
  162. package/src/react/usePauseProvider.tsx +13 -0
  163. package/src/react/useSelector.ts +167 -0
  164. package/src/react/useUnmount.ts +8 -0
  165. package/src/react/useWhen.ts +9 -0
  166. package/src/react-hooks/createObservableHook.ts +53 -0
  167. package/src/react-hooks/useFetch.ts +16 -0
  168. package/src/react-hooks/useHover.ts +40 -0
  169. package/src/react-hooks/useMeasure.ts +48 -0
  170. package/src/react-hooks/useObservableNextRouter.ts +137 -0
  171. package/src/react-hooks/useObservableQuery.ts +205 -0
  172. package/src/react-hooks/usePersistedObservable.ts +24 -0
  173. package/src/retry.ts +69 -0
  174. package/src/setupTracking.ts +26 -0
  175. package/src/sync/activateSyncedNode.ts +146 -0
  176. package/src/sync/configureObservableSync.ts +7 -0
  177. package/src/sync/syncHelpers.ts +15 -0
  178. package/src/sync/syncObservable.ts +989 -0
  179. package/src/sync/syncObservableAdapter.ts +30 -0
  180. package/src/sync/synced.ts +20 -0
  181. package/src/sync-plugins/fetch.ts +42 -0
  182. package/src/syncTypes.ts +163 -0
  183. package/src/trace/traceHelpers.ts +11 -0
  184. package/src/trace/useTraceListeners.ts +34 -0
  185. package/src/trace/useTraceUpdates.ts +24 -0
  186. package/src/trace/useVerifyNotTracking.ts +33 -0
  187. package/src/trace/useVerifyOneRender.ts +10 -0
  188. package/src/trackSelector.ts +52 -0
  189. package/src/tracking.ts +43 -0
  190. package/src/types/babel.d.ts +12 -0
  191. package/src/when.ts +70 -0
  192. package/sync-plugins/fetch.d.ts +11 -0
  193. package/sync-plugins/fetch.js +24 -0
  194. package/sync-plugins/fetch.js.map +1 -0
  195. package/sync-plugins/fetch.mjs +22 -0
  196. package/sync-plugins/fetch.mjs.map +1 -0
  197. package/sync.d.ts +8 -0
  198. package/sync.js +919 -0
  199. package/sync.js.map +1 -0
  200. package/sync.mjs +912 -0
  201. package/sync.mjs.map +1 -0
  202. package/trace.js +13 -10
  203. package/trace.js.map +1 -1
  204. package/trace.mjs +11 -8
  205. package/trace.mjs.map +1 -1
  206. package/types/babel.d.ts +3 -3
  207. package/config/enableDirectAccess.d.ts +0 -7
  208. package/config/enableDirectAccess.js +0 -25
  209. package/config/enableDirectAccess.js.map +0 -1
  210. package/config/enableDirectAccess.mjs +0 -23
  211. package/config/enableDirectAccess.mjs.map +0 -1
  212. package/config/enableDirectPeek.d.ts +0 -7
  213. package/config/enableDirectPeek.js.map +0 -1
  214. package/config/enableDirectPeek.mjs.map +0 -1
  215. package/config/enableReactDirectRender.d.ts +0 -2
  216. package/config/enableReactDirectRender.js +0 -78
  217. package/config/enableReactDirectRender.js.map +0 -1
  218. package/config/enableReactDirectRender.mjs +0 -75
  219. package/config/enableReactDirectRender.mjs.map +0 -1
  220. package/src/ObservableObject.d.ts +0 -14
  221. package/src/ObservablePrimitive.d.ts +0 -7
  222. package/src/babel/index.d.ts +0 -17
  223. package/src/batching.d.ts +0 -6
  224. package/src/computed.d.ts +0 -4
  225. package/src/config/enableDirectAccess.d.ts +0 -7
  226. package/src/config/enableDirectPeek.d.ts +0 -7
  227. package/src/config/enableReactComponents.d.ts +0 -7
  228. package/src/config/enableReactDirectRender.d.ts +0 -2
  229. package/src/config/enableReactNativeComponents.d.ts +0 -20
  230. package/src/config/enableReactTracking.d.ts +0 -15
  231. package/src/config/enableReactUse.d.ts +0 -7
  232. package/src/config.d.ts +0 -8
  233. package/src/createObservable.d.ts +0 -2
  234. package/src/event.d.ts +0 -2
  235. package/src/globals.d.ts +0 -32
  236. package/src/helpers/fetch.d.ts +0 -6
  237. package/src/helpers/pageHash.d.ts +0 -7
  238. package/src/helpers/pageHashParams.d.ts +0 -7
  239. package/src/helpers/time.d.ts +0 -3
  240. package/src/helpers.d.ts +0 -13
  241. package/src/history/trackHistory.d.ts +0 -4
  242. package/src/is.d.ts +0 -10
  243. package/src/observable.d.ts +0 -16
  244. package/src/observableInterfaces.d.ts +0 -456
  245. package/src/observe.d.ts +0 -6
  246. package/src/onChange.d.ts +0 -7
  247. package/src/persist/configureObservablePersistence.d.ts +0 -3
  248. package/src/persist/fieldTransformer.d.ts +0 -8
  249. package/src/persist/observablePersistRemoteFunctionsAdapter.d.ts +0 -2
  250. package/src/persist/persistActivateNode.d.ts +0 -1
  251. package/src/persist/persistHelpers.d.ts +0 -1
  252. package/src/persist/persistObservable.d.ts +0 -25
  253. package/src/persist-plugins/async-storage.d.ts +0 -14
  254. package/src/persist-plugins/fetch.d.ts +0 -10
  255. package/src/persist-plugins/firebase.d.ts +0 -51
  256. package/src/persist-plugins/indexeddb.d.ts +0 -25
  257. package/src/persist-plugins/local-storage.d.ts +0 -21
  258. package/src/persist-plugins/mmkv.d.ts +0 -14
  259. package/src/persist-plugins/query.d.ts +0 -18
  260. package/src/proxy.d.ts +0 -5
  261. package/src/react/Computed.d.ts +0 -5
  262. package/src/react/For.d.ts +0 -15
  263. package/src/react/Memo.d.ts +0 -3
  264. package/src/react/Reactive.d.ts +0 -9
  265. package/src/react/Show.d.ts +0 -18
  266. package/src/react/Switch.d.ts +0 -14
  267. package/src/react/react-globals.d.ts +0 -3
  268. package/src/react/reactive-observer.d.ts +0 -14
  269. package/src/react/useComputed.d.ts +0 -5
  270. package/src/react/useEffectOnce.d.ts +0 -1
  271. package/src/react/useIsMounted.d.ts +0 -2
  272. package/src/react/useMount.d.ts +0 -2
  273. package/src/react/useObservable.d.ts +0 -9
  274. package/src/react/useObservableReducer.d.ts +0 -7
  275. package/src/react/useObservableState.d.ts +0 -2
  276. package/src/react/useObserve.d.ts +0 -4
  277. package/src/react/useObserveEffect.d.ts +0 -4
  278. package/src/react/usePauseProvider.d.ts +0 -8
  279. package/src/react/useSelector.d.ts +0 -3
  280. package/src/react/useUnmount.d.ts +0 -2
  281. package/src/react/useWhen.d.ts +0 -3
  282. package/src/react-hooks/createObservableHook.d.ts +0 -2
  283. package/src/react-hooks/useFetch.d.ts +0 -6
  284. package/src/react-hooks/useHover.d.ts +0 -3
  285. package/src/react-hooks/useMeasure.d.ts +0 -6
  286. package/src/react-hooks/useObservableNextRouter.d.ts +0 -33
  287. package/src/react-hooks/useObservableQuery.d.ts +0 -6
  288. package/src/react-hooks/usePersistedObservable.d.ts +0 -11
  289. package/src/retry.d.ts +0 -9
  290. package/src/setupTracking.d.ts +0 -2
  291. package/src/trace/traceHelpers.d.ts +0 -2
  292. package/src/trace/useTraceListeners.d.ts +0 -1
  293. package/src/trace/useTraceUpdates.d.ts +0 -1
  294. package/src/trace/useVerifyNotTracking.d.ts +0 -1
  295. package/src/trace/useVerifyOneRender.d.ts +0 -1
  296. package/src/trackSelector.d.ts +0 -7
  297. package/src/tracking.d.ts +0 -13
  298. 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
+ }
@@ -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
+ }