@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
package/src/helpers.ts ADDED
@@ -0,0 +1,221 @@
1
+ import { beginBatch, endBatch } from './batching';
2
+ import { getNode, globalState, isObservable, setNodeValue, symbolDelete, symbolOpaque } from './globals';
3
+ import { isArray, isEmpty, isFunction, isObject } from './is';
4
+ import type { ObserveEvent, OpaqueObject, Selector, TypeAtPath } from './observableInterfaces';
5
+ import type { Observable, ObservableParam } from './observableTypes';
6
+
7
+ export function computeSelector<T>(selector: Selector<T>, e?: ObserveEvent<T>, retainObservable?: boolean) {
8
+ let c = selector as any;
9
+ if (!isObservable(c) && isFunction(c)) {
10
+ c = e ? c(e) : c();
11
+ }
12
+
13
+ return isObservable(c) && !retainObservable ? c.get() : c;
14
+ }
15
+
16
+ export function getObservableIndex(obs: ObservableParam): number {
17
+ const node = getNode(obs);
18
+ const n = +node.key! as number;
19
+ return n - n < 1 ? +n : -1;
20
+ }
21
+
22
+ export function opaqueObject<T extends object>(value: T): OpaqueObject<T> {
23
+ if (value) {
24
+ (value as OpaqueObject<T>)[symbolOpaque] = true;
25
+ }
26
+ return value as OpaqueObject<T>;
27
+ }
28
+
29
+ export function getValueAtPath(obj: Record<string, any>, path: string[]): any {
30
+ let o: Record<string, any> = obj;
31
+ for (let i = 0; o && i < path.length; i++) {
32
+ const p = path[i];
33
+ o = o[p];
34
+ }
35
+
36
+ return o;
37
+ }
38
+
39
+ export function setAtPath<T extends object>(
40
+ obj: T,
41
+ path: string[],
42
+ pathTypes: TypeAtPath[],
43
+ value: any,
44
+ mode?: 'set' | 'merge',
45
+ fullObj?: T,
46
+ restore?: (path: string[], value: any) => void,
47
+ ) {
48
+ let o: Record<string, any> = obj;
49
+ let oFull: Record<string, any> | undefined = fullObj;
50
+ let p: string | undefined = undefined;
51
+ if (path.length > 0) {
52
+ for (let i = 0; i < path.length; i++) {
53
+ p = path[i];
54
+ if (o[p] === symbolDelete) {
55
+ // If this was previously deleted, restore it
56
+ if (oFull) {
57
+ o[p] = oFull[p];
58
+ restore?.(path.slice(0, i + 1), o[p]);
59
+ }
60
+ return obj;
61
+ } else if (o[p] === undefined || o[p] === null) {
62
+ o[p] = initializePathType(pathTypes[i]);
63
+ }
64
+ if (i < path.length - 1) {
65
+ o = o[p];
66
+ if (oFull) {
67
+ oFull = oFull[p];
68
+ }
69
+ }
70
+ }
71
+ }
72
+
73
+ // Don't set if the value is the same. This prevents creating a new key
74
+ // when setting undefined on an object without this key
75
+ if (p === undefined) {
76
+ if (mode === 'merge') {
77
+ obj = _mergeIntoObservable(obj, value);
78
+ } else {
79
+ obj = value;
80
+ }
81
+ } else {
82
+ if (mode === 'merge') {
83
+ o[p] = _mergeIntoObservable(o[p], value);
84
+ } else {
85
+ o[p] = value;
86
+ }
87
+ }
88
+
89
+ return obj;
90
+ }
91
+ export function setInObservableAtPath(
92
+ obs: ObservableParam,
93
+ path: string[],
94
+ pathTypes: TypeAtPath[],
95
+ value: any,
96
+ mode: 'assign' | 'set' | 'merge',
97
+ ) {
98
+ let o: Record<string, any> = obs;
99
+ let v = value;
100
+ for (let i = 0; i < path.length; i++) {
101
+ const p = path[i];
102
+ if (!o.peek()[p]) {
103
+ o[p].set(initializePathType(pathTypes[i]));
104
+ }
105
+ o = o[p];
106
+ v = v[p];
107
+ }
108
+
109
+ if (v === symbolDelete) {
110
+ (o as Observable).delete();
111
+ }
112
+ // Assign if possible, or set otherwise
113
+ else if (mode === 'assign' && (o as Observable).assign && isObject(o.peek())) {
114
+ (o as Observable<{}>).assign(v);
115
+ } else if (mode === 'merge') {
116
+ mergeIntoObservable(o, v);
117
+ } else {
118
+ o.set(v);
119
+ }
120
+ }
121
+ export function mergeIntoObservable<T extends ObservableParam<Record<string, any>> | object>(
122
+ target: T,
123
+ ...sources: any[]
124
+ ): T {
125
+ beginBatch();
126
+ globalState.isMerging = true;
127
+ for (let i = 0; i < sources.length; i++) {
128
+ target = _mergeIntoObservable(target, sources[i]);
129
+ }
130
+ globalState.isMerging = false;
131
+ endBatch();
132
+ return target;
133
+ }
134
+ function _mergeIntoObservable<T extends ObservableParam<Record<string, any>> | object>(target: T, source: any): T {
135
+ if (isObservable(source)) {
136
+ source = source.peek();
137
+ }
138
+ const needsSet = isObservable(target);
139
+ const targetValue = needsSet ? target.peek() : target;
140
+
141
+ const isTargetArr = isArray(targetValue);
142
+ const isTargetObj = !isTargetArr && isObject(targetValue);
143
+
144
+ if ((isTargetObj && isObject(source) && !isEmpty(targetValue)) || (isTargetArr && targetValue.length > 0)) {
145
+ const keys: string[] = Object.keys(source);
146
+
147
+ for (let i = 0; i < keys.length; i++) {
148
+ const key = keys[i];
149
+ const sourceValue = (source as Record<string, any>)[key];
150
+ if (sourceValue === symbolDelete) {
151
+ needsSet && (target as any)[key]?.delete
152
+ ? (target as any)[key].delete()
153
+ : delete (target as Record<string, any>)[key];
154
+ } else {
155
+ const isObj = isObject(sourceValue);
156
+ const isArr = !isObj && isArray(sourceValue);
157
+ const targetChild = (target as Record<string, any>)[key];
158
+ if ((isObj || isArr) && targetChild && (needsSet || !isEmpty(targetChild))) {
159
+ if (!needsSet && (!targetChild || (isObj ? !isObject(targetChild) : !isArray(targetChild)))) {
160
+ (target as Record<string, any>)[key] = sourceValue;
161
+ } else {
162
+ _mergeIntoObservable(targetChild, sourceValue);
163
+ }
164
+ } else {
165
+ needsSet
166
+ ? targetChild.set(sourceValue)
167
+ : (((target as Record<string, any>)[key] as any) = sourceValue);
168
+ }
169
+ }
170
+ }
171
+ } else if (source !== undefined) {
172
+ needsSet ? target.set(source) : ((target as any) = source);
173
+ }
174
+
175
+ return target;
176
+ }
177
+ export function constructObjectWithPath(path: string[], pathTypes: TypeAtPath[], value: any): object {
178
+ let out;
179
+ if (path.length > 0) {
180
+ let o: Record<string, any> = (out = {});
181
+ for (let i = 0; i < path.length; i++) {
182
+ const p = path[i];
183
+ o[p] = i === path.length - 1 ? value : initializePathType(pathTypes[i]);
184
+ o = o[p];
185
+ }
186
+ } else {
187
+ out = value;
188
+ }
189
+
190
+ return out;
191
+ }
192
+ export function deconstructObjectWithPath(path: string[], pathTypes: TypeAtPath[], value: any): object {
193
+ let o = value;
194
+ for (let i = 0; i < path.length; i++) {
195
+ const p = path[i];
196
+ o = o ? o[p] : initializePathType(pathTypes[i]);
197
+ }
198
+
199
+ return o;
200
+ }
201
+ export function isObservableValueReady(value: any) {
202
+ return !!value && ((!isObject(value) && !isArray(value)) || !isEmpty(value));
203
+ }
204
+
205
+ export function setSilently(obs: ObservableParam, newValue: any) {
206
+ const node = getNode(obs);
207
+ return setNodeValue(node, newValue).newValue;
208
+ }
209
+
210
+ export function initializePathType(pathType: TypeAtPath): any {
211
+ switch (pathType) {
212
+ case 'array':
213
+ return [];
214
+ case 'object':
215
+ return {};
216
+ case 'map':
217
+ return new Map();
218
+ case 'set':
219
+ return new Set();
220
+ }
221
+ }
@@ -0,0 +1,29 @@
1
+ import { ObservableParam, constructObjectWithPath, mergeIntoObservable, observable } from '@legendapp/state';
2
+
3
+ // This type is purely for documentation.
4
+ type TimestampAsString = string;
5
+
6
+ export function trackHistory<T>(
7
+ obs: ObservableParam<T>,
8
+ targetObservable?: ObservableParam<Record<TimestampAsString, Partial<T>>>,
9
+ ): ObservableParam<Record<TimestampAsString, any>> {
10
+ const history = targetObservable ?? observable<Record<TimestampAsString, Partial<T>>>();
11
+
12
+ obs.onChange(({ loading, remote, changes }) => {
13
+ // Don't save history if this is a remote change.
14
+ // History will be saved remotely by the client making the local change.
15
+ if (!loading && !remote) {
16
+ const time: TimestampAsString = Date.now().toString();
17
+
18
+ // Save to history observable by date, with the previous value
19
+ for (let i = 0; i < changes.length; i++) {
20
+ const { path, prevAtPath, pathTypes } = changes[i];
21
+
22
+ const obj = constructObjectWithPath(path, pathTypes, prevAtPath);
23
+ mergeIntoObservable((history as any)[time], obj);
24
+ }
25
+ }
26
+ });
27
+
28
+ return history as any;
29
+ }
package/src/is.ts ADDED
@@ -0,0 +1,56 @@
1
+ import type { ChildNodeValue, NodeValue } from './observableInterfaces';
2
+
3
+ export const hasOwnProperty = Object.prototype.hasOwnProperty;
4
+
5
+ export function isArray(obj: unknown): obj is Array<any> {
6
+ return Array.isArray(obj);
7
+ }
8
+ export function isString(obj: unknown): obj is string {
9
+ return typeof obj === 'string';
10
+ }
11
+ export function isObject(obj: unknown): obj is Record<any, any> {
12
+ return !!obj && typeof obj === 'object' && !(obj instanceof Date) && !isArray(obj);
13
+ }
14
+ export function isFunction(obj: unknown): obj is Function {
15
+ return typeof obj === 'function';
16
+ }
17
+ export function isPrimitive(arg: unknown): arg is string | number | bigint | boolean | symbol {
18
+ const type = typeof arg;
19
+ return arg !== undefined && (isDate(arg) || (type !== 'object' && type !== 'function'));
20
+ }
21
+ export function isDate(obj: unknown): obj is Date {
22
+ return obj instanceof Date;
23
+ }
24
+ export function isSymbol(obj: unknown): obj is symbol {
25
+ return typeof obj === 'symbol';
26
+ }
27
+ export function isBoolean(obj: unknown): obj is boolean {
28
+ return typeof obj === 'boolean';
29
+ }
30
+ export function isPromise<T>(obj: unknown): obj is Promise<T> {
31
+ return obj instanceof Promise;
32
+ }
33
+ export function isEmpty(obj: object): boolean {
34
+ // Looping and returning false on the first property is faster than Object.keys(obj).length === 0
35
+ // https://jsbench.me/qfkqv692c8
36
+ if (!obj) return false;
37
+ if (isArray(obj)) return obj.length === 0;
38
+ for (const key in obj) {
39
+ if (hasOwnProperty.call(obj, key)) {
40
+ return false;
41
+ }
42
+ }
43
+ return true;
44
+ }
45
+ export function isNullOrUndefined(value: any): value is undefined | null {
46
+ return value === undefined || value === null;
47
+ }
48
+ const setPrimitives = new Set(['boolean', 'string', 'number']);
49
+ /** @internal */
50
+ export function isActualPrimitive(arg: unknown): arg is boolean | string | number {
51
+ return setPrimitives.has(typeof arg);
52
+ }
53
+ /** @internal */
54
+ export function isChildNodeValue(node: NodeValue): node is ChildNodeValue {
55
+ return !!node.parent;
56
+ }
package/src/linked.ts ADDED
@@ -0,0 +1,6 @@
1
+ import { symbolLinked } from './globals';
2
+ import { Linked, LinkedOptions } from './observableInterfaces';
3
+
4
+ export function linked<T>(params: LinkedOptions<T>): Linked<T> {
5
+ return (() => ({ [symbolLinked]: params })) as any;
6
+ }
@@ -0,0 +1,32 @@
1
+ import { extractPromise, getProxy, peekInternal } from './ObservableObject';
2
+ import { ObservablePrimitiveClass } from './ObservablePrimitive';
3
+ import { createObservable } from './createObservable';
4
+ import { getNode } from './globals';
5
+ import type { Observable, ObservableParam, ObservablePrimitive, RecursiveValueOrFunction } from './observableTypes';
6
+ import type { ObservableSyncState } from './syncTypes';
7
+
8
+ export function observable<T>(): Observable<T | undefined>;
9
+ export function observable<T>(
10
+ value: Promise<RecursiveValueOrFunction<T>> | (() => RecursiveValueOrFunction<T>) | RecursiveValueOrFunction<T>,
11
+ ): Observable<T>;
12
+ export function observable<T>(value: T): Observable<T>;
13
+ export function observable<T>(value?: T): Observable<any> {
14
+ return createObservable(value, false, extractPromise, getProxy, ObservablePrimitiveClass) as any;
15
+ }
16
+
17
+ export function observablePrimitive<T>(value: Promise<T>): ObservablePrimitive<T>;
18
+ export function observablePrimitive<T>(value?: T): ObservablePrimitive<T>;
19
+ export function observablePrimitive<T>(value?: T | Promise<T>): ObservablePrimitive<T> {
20
+ return createObservable(value, true, extractPromise, getProxy, ObservablePrimitiveClass) as any;
21
+ }
22
+
23
+ export function syncState(obs: ObservableParam) {
24
+ const node = getNode(obs);
25
+ if (!node.state) {
26
+ peekInternal(node);
27
+ }
28
+ if (!node.state) {
29
+ node.state = observable({} as ObservableSyncState);
30
+ }
31
+ return node.state!;
32
+ }
@@ -0,0 +1,165 @@
1
+ import type { symbolOpaque } from './globals';
2
+ import type { Observable, ObservableParam } from './observableTypes';
3
+ import { GetMode, ObservableSyncState, SyncedOptions } from './syncTypes';
4
+
5
+ export type TrackingType = undefined | true | symbol; // true === shallow
6
+
7
+ export interface GetOptions {
8
+ shallow: boolean;
9
+ }
10
+
11
+ export type OpaqueObject<T> = T & { [symbolOpaque]: true };
12
+
13
+ export interface ListenerParams<T = any> {
14
+ value: T;
15
+ getPrevious: () => T;
16
+ changes: Change[];
17
+ remote: boolean;
18
+ /** @internal */
19
+ loading: boolean;
20
+ }
21
+
22
+ export type ListenerFn<T = any> = (params: ListenerParams<T>) => void;
23
+
24
+ export interface ObservableEvent {
25
+ fire(): void;
26
+ on(cb?: () => void): ObservableListenerDispose;
27
+ get(): void;
28
+ }
29
+
30
+ export type TypeAtPath = 'object' | 'array' | 'map' | 'set';
31
+
32
+ export interface Change {
33
+ path: string[];
34
+ pathTypes: TypeAtPath[];
35
+ valueAtPath: any;
36
+ prevAtPath: any;
37
+ }
38
+
39
+ export type RecordValue<T> = T extends Record<string, infer t> ? t : never;
40
+ export type ArrayValue<T> = T extends Array<infer t> ? t : never;
41
+ export type ObservableValue<T> = T extends Observable<infer t> ? t : never;
42
+
43
+ export type Selector<T> = ObservableParam<T> | ObservableEvent | (() => T) | T;
44
+
45
+ export type ClassConstructor<I, Args extends any[] = any[]> = new (...args: Args) => I;
46
+ export type ObservableListenerDispose = () => void;
47
+
48
+ export interface ObservableRoot {
49
+ _: any;
50
+ set?: (value: any) => void;
51
+ }
52
+
53
+ export type Primitive = boolean | string | number | Date;
54
+ export type NotPrimitive<T> = T extends Primitive ? never : T;
55
+
56
+ export interface NodeValueListener {
57
+ track: TrackingType;
58
+ noArgs?: boolean;
59
+ listener: ListenerFn;
60
+ }
61
+ export interface TrackingState {
62
+ nodes?: Map<NodeValue, TrackingNode>;
63
+ traceListeners?: (nodes: Map<NodeValue, TrackingNode>) => void;
64
+ traceUpdates?: (fn: Function) => Function;
65
+ }
66
+
67
+ interface BaseNodeValue {
68
+ children?: Map<string, ChildNodeValue>;
69
+ proxy?: object;
70
+ root: ObservableRoot;
71
+ listeners?: Set<NodeValueListener>;
72
+ listenersImmediate?: Set<NodeValueListener>;
73
+ isEvent?: boolean;
74
+ linkedToNode?: NodeValue;
75
+ linkedToNodeDispose?: () => void;
76
+ activatedObserveDispose?: () => void;
77
+ linkedFromNodes?: Set<NodeValue>;
78
+ isSetting?: number;
79
+ isAssigning?: number;
80
+ isComputing?: boolean;
81
+ parentOther?: NodeValue;
82
+ functions?: Map<string, Function | Observable<any>>;
83
+ lazy?: boolean;
84
+ lazyFn?: Function;
85
+ needsExtract?: boolean;
86
+ state?: Observable<ObservableSyncState>;
87
+ activated?: boolean;
88
+ activationState?: SyncedOptions & { onError?: () => void; persistedRetry?: boolean };
89
+ dirtyFn?: () => void;
90
+ }
91
+
92
+ export interface RootNodeValue extends BaseNodeValue {
93
+ parent?: undefined;
94
+ key?: undefined;
95
+ }
96
+
97
+ export interface ChildNodeValue extends BaseNodeValue {
98
+ parent: NodeValue;
99
+ key: string;
100
+ }
101
+
102
+ export type NodeValue = RootNodeValue | ChildNodeValue;
103
+
104
+ export interface TrackingNode {
105
+ node: NodeValue;
106
+ track: TrackingType;
107
+ num: number;
108
+ }
109
+ export interface ObserveEvent<T> {
110
+ num: number;
111
+ previous?: T | undefined;
112
+ cancel?: boolean;
113
+ onCleanup?: () => void;
114
+ }
115
+ export interface ObserveEventCallback<T> {
116
+ num: number;
117
+ previous?: T | undefined;
118
+ value?: T;
119
+ cancel: boolean;
120
+ nodes: Map<NodeValue, TrackingNode> | undefined;
121
+ refresh: () => void;
122
+ onCleanup?: () => void;
123
+ onCleanupReaction?: () => void;
124
+ }
125
+
126
+ export type SetParams<T> = ListenerParams<T extends Promise<infer t> ? t : T>;
127
+
128
+ export interface LinkedOptions<T = any> {
129
+ get?: () => Promise<T> | T;
130
+ set?: (params: SetParams<T>) => void | Promise<any>;
131
+ waitFor?: Selector<any>;
132
+ waitForSet?:
133
+ | ((params: { value: T; changes: Change[] }) => any)
134
+ | Promise<any>
135
+ | ObservableParam<any>
136
+ | ObservableEvent;
137
+ initial?: T extends Promise<infer t> ? t : T;
138
+ }
139
+
140
+ export interface UpdateFnParams {
141
+ value: unknown;
142
+ mode?: GetMode;
143
+ lastSync?: number | undefined;
144
+ }
145
+ export type UpdateFn = (params: UpdateFnParams) => void;
146
+ export interface RetryOptions {
147
+ infinite?: boolean;
148
+ times?: number;
149
+ delay?: number;
150
+ backoff?: 'constant' | 'exponential';
151
+ maxDelay?: number;
152
+ }
153
+
154
+ export interface SubscribeOptions {
155
+ node: NodeValue;
156
+ update: UpdateFn;
157
+ refresh: () => void;
158
+ }
159
+ export interface CacheReturnValue {
160
+ lastSync: number;
161
+ value: any;
162
+ }
163
+
164
+ export type Synced<T> = T;
165
+ export type Linked<T> = T;