@legendapp/state 3.0.0-alpha.0 → 3.0.0-alpha.2

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 (323) hide show
  1. package/.DS_Store +0 -0
  2. package/CHANGELOG.md +1 -827
  3. package/LICENSE +1 -21
  4. package/README.md +1 -141
  5. package/as/arrayAsRecord.d.mts +5 -0
  6. package/as/arrayAsRecord.d.ts +5 -0
  7. package/as/arrayAsRecord.js +28 -0
  8. package/as/arrayAsRecord.mjs +26 -0
  9. package/as/arrayAsSet.d.mts +5 -0
  10. package/as/arrayAsSet.d.ts +5 -0
  11. package/as/arrayAsSet.js +13 -0
  12. package/as/arrayAsSet.mjs +11 -0
  13. package/as/arrayAsString.d.mts +5 -0
  14. package/as/arrayAsString.d.ts +5 -0
  15. package/as/arrayAsString.js +13 -0
  16. package/as/arrayAsString.mjs +11 -0
  17. package/as/numberAsString.d.mts +5 -0
  18. package/as/numberAsString.d.ts +5 -0
  19. package/as/numberAsString.js +13 -0
  20. package/as/numberAsString.mjs +11 -0
  21. package/as/recordAsArray.d.mts +5 -0
  22. package/as/recordAsArray.d.ts +5 -0
  23. package/as/recordAsArray.js +25 -0
  24. package/as/recordAsArray.mjs +23 -0
  25. package/as/recordAsString.d.mts +5 -0
  26. package/as/recordAsString.d.ts +5 -0
  27. package/as/recordAsString.js +13 -0
  28. package/as/recordAsString.mjs +11 -0
  29. package/as/setAsArray.d.mts +5 -0
  30. package/as/setAsArray.d.ts +5 -0
  31. package/as/setAsArray.js +13 -0
  32. package/as/setAsArray.mjs +11 -0
  33. package/as/setAsString.d.mts +5 -0
  34. package/as/setAsString.d.ts +5 -0
  35. package/as/setAsString.js +13 -0
  36. package/as/setAsString.mjs +11 -0
  37. package/as/stringAsArray.d.mts +5 -0
  38. package/as/stringAsArray.d.ts +5 -0
  39. package/as/stringAsArray.js +13 -0
  40. package/as/stringAsArray.mjs +11 -0
  41. package/as/stringAsNumber.d.mts +5 -0
  42. package/as/stringAsNumber.d.ts +5 -0
  43. package/as/stringAsNumber.js +16 -0
  44. package/as/stringAsNumber.mjs +14 -0
  45. package/as/stringAsRecord.d.mts +5 -0
  46. package/as/stringAsRecord.d.ts +5 -0
  47. package/as/stringAsRecord.js +15 -0
  48. package/as/stringAsRecord.mjs +13 -0
  49. package/as/stringAsSet.d.mts +5 -0
  50. package/as/stringAsSet.d.ts +5 -0
  51. package/as/stringAsSet.js +13 -0
  52. package/as/stringAsSet.mjs +11 -0
  53. package/babel.d.mts +21 -0
  54. package/babel.d.ts +21 -2
  55. package/babel.js +57 -53
  56. package/babel.mjs +65 -0
  57. package/config/enable$GetSet.js +13 -14
  58. package/config/enable$GetSet.mjs +13 -14
  59. package/config/enableReactComponents.d.mts +9 -0
  60. package/config/enableReactComponents.d.ts +4 -2
  61. package/config/enableReactComponents.js +13 -10
  62. package/config/enableReactComponents.mjs +13 -10
  63. package/config/enableReactNativeComponents.d.mts +22 -0
  64. package/config/enableReactNativeComponents.d.ts +6 -4
  65. package/config/enableReactNativeComponents.js +43 -47
  66. package/config/enableReactNativeComponents.mjs +43 -47
  67. package/config/enableReactTracking.d.mts +7 -0
  68. package/config/enableReactTracking.d.ts +3 -2
  69. package/config/enableReactTracking.js +33 -38
  70. package/config/enableReactTracking.mjs +33 -38
  71. package/config/enableReactUse.d.mts +10 -0
  72. package/config/enableReactUse.d.ts +4 -1
  73. package/config/enableReactUse.js +15 -14
  74. package/config/enableReactUse.mjs +15 -14
  75. package/config/{enable$GetSet.d.ts → enable_GetSet.d.mts} +4 -2
  76. package/config/enable_GetSet.d.ts +10 -0
  77. package/config/enable_PeekAssign.d.mts +10 -0
  78. package/config/enable_PeekAssign.d.ts +4 -2
  79. package/config/enable_PeekAssign.js +13 -14
  80. package/config/enable_PeekAssign.mjs +13 -14
  81. package/helpers/pageHash.d.mts +9 -0
  82. package/helpers/pageHash.d.ts +2 -0
  83. package/helpers/pageHash.js +25 -30
  84. package/helpers/pageHash.mjs +25 -30
  85. package/helpers/pageHashParams.d.mts +9 -0
  86. package/helpers/pageHashParams.d.ts +2 -0
  87. package/helpers/pageHashParams.js +34 -37
  88. package/helpers/pageHashParams.mjs +34 -37
  89. package/helpers/time.d.mts +6 -0
  90. package/helpers/time.d.ts +6 -3
  91. package/helpers/time.js +17 -17
  92. package/helpers/time.mjs +17 -17
  93. package/helpers/trackHistory.d.mts +6 -0
  94. package/helpers/trackHistory.d.ts +6 -0
  95. package/helpers/trackHistory.js +21 -0
  96. package/helpers/trackHistory.mjs +19 -0
  97. package/helpers/undoRedo.d.mts +37 -0
  98. package/helpers/undoRedo.d.ts +37 -0
  99. package/helpers/undoRedo.js +68 -0
  100. package/helpers/undoRedo.mjs +66 -0
  101. package/index.d.mts +404 -0
  102. package/index.d.ts +371 -28
  103. package/index.js +2003 -2164
  104. package/index.mjs +2003 -2164
  105. package/package.json +254 -185
  106. package/persist-plugins/async-storage.d.mts +18 -0
  107. package/persist-plugins/async-storage.d.ts +6 -3
  108. package/persist-plugins/async-storage.js +79 -86
  109. package/persist-plugins/async-storage.mjs +79 -86
  110. package/persist-plugins/indexeddb.d.mts +29 -0
  111. package/persist-plugins/indexeddb.d.ts +6 -3
  112. package/persist-plugins/indexeddb.js +331 -348
  113. package/persist-plugins/indexeddb.mjs +331 -348
  114. package/persist-plugins/local-storage.d.mts +23 -0
  115. package/persist-plugins/local-storage.d.ts +8 -5
  116. package/persist-plugins/local-storage.js +74 -76
  117. package/persist-plugins/local-storage.mjs +74 -76
  118. package/persist-plugins/mmkv.d.mts +18 -0
  119. package/persist-plugins/mmkv.d.ts +6 -3
  120. package/persist-plugins/mmkv.js +82 -86
  121. package/persist-plugins/mmkv.mjs +82 -86
  122. package/react-hooks/createObservableHook.d.mts +5 -0
  123. package/react-hooks/createObservableHook.d.ts +4 -1
  124. package/react-hooks/createObservableHook.js +29 -30
  125. package/react-hooks/createObservableHook.mjs +25 -30
  126. package/react-hooks/useHover.d.mts +5 -0
  127. package/react-hooks/useHover.d.ts +5 -3
  128. package/react-hooks/useHover.js +29 -29
  129. package/react-hooks/useHover.mjs +29 -29
  130. package/react-hooks/useMeasure.d.mts +9 -0
  131. package/react-hooks/useMeasure.d.ts +5 -2
  132. package/react-hooks/useMeasure.js +30 -32
  133. package/react-hooks/useMeasure.mjs +30 -32
  134. package/react-hooks/useObservableNextRouter.d.mts +35 -0
  135. package/react-hooks/useObservableNextRouter.d.ts +9 -7
  136. package/react-hooks/useObservableNextRouter.js +64 -77
  137. package/react-hooks/useObservableNextRouter.mjs +60 -77
  138. package/react.d.mts +157 -0
  139. package/react.d.ts +157 -21
  140. package/react.js +458 -749
  141. package/react.mjs +457 -752
  142. package/sync-plugins/crud.d.mts +54 -0
  143. package/sync-plugins/crud.d.ts +12 -10
  144. package/sync-plugins/crud.js +253 -270
  145. package/sync-plugins/crud.mjs +253 -270
  146. package/sync-plugins/fetch.d.mts +21 -0
  147. package/sync-plugins/fetch.d.ts +7 -4
  148. package/sync-plugins/fetch.js +50 -37
  149. package/sync-plugins/fetch.mjs +50 -37
  150. package/sync-plugins/keel.d.mts +108 -0
  151. package/sync-plugins/keel.d.ts +17 -15
  152. package/sync-plugins/keel.js +229 -462
  153. package/sync-plugins/keel.mjs +227 -464
  154. package/sync-plugins/supabase.d.mts +39 -0
  155. package/sync-plugins/supabase.d.ts +16 -14
  156. package/sync-plugins/supabase.js +128 -128
  157. package/sync-plugins/supabase.mjs +128 -128
  158. package/sync-plugins/tanstack-query.d.mts +14 -0
  159. package/sync-plugins/tanstack-query.d.ts +7 -4
  160. package/sync-plugins/tanstack-query.js +51 -57
  161. package/sync-plugins/tanstack-query.mjs +51 -57
  162. package/sync-plugins/tanstack-react-query.d.mts +8 -0
  163. package/sync-plugins/tanstack-react-query.d.ts +6 -1
  164. package/sync-plugins/tanstack-react-query.js +2 -2
  165. package/sync-plugins/tanstack-react-query.mjs +2 -2
  166. package/sync.d.mts +351 -0
  167. package/sync.d.ts +349 -9
  168. package/sync.js +909 -962
  169. package/sync.mjs +919 -972
  170. package/trace.d.mts +9 -0
  171. package/trace.d.ts +9 -4
  172. package/trace.js +72 -62
  173. package/trace.mjs +72 -62
  174. package/types/babel.d.ts +1 -12
  175. package/babel.js.map +0 -1
  176. package/config/enable$GetSet.js.map +0 -1
  177. package/config/enable$GetSet.mjs.map +0 -1
  178. package/config/enableReactComponents.js.map +0 -1
  179. package/config/enableReactComponents.mjs.map +0 -1
  180. package/config/enableReactNativeComponents.js.map +0 -1
  181. package/config/enableReactNativeComponents.mjs.map +0 -1
  182. package/config/enableReactTracking.js.map +0 -1
  183. package/config/enableReactTracking.mjs.map +0 -1
  184. package/config/enableReactUse.js.map +0 -1
  185. package/config/enableReactUse.mjs.map +0 -1
  186. package/config/enable_PeekAssign.js.map +0 -1
  187. package/config/enable_PeekAssign.mjs.map +0 -1
  188. package/helpers/pageHash.js.map +0 -1
  189. package/helpers/pageHash.mjs.map +0 -1
  190. package/helpers/pageHashParams.js.map +0 -1
  191. package/helpers/pageHashParams.mjs.map +0 -1
  192. package/helpers/time.js.map +0 -1
  193. package/helpers/time.mjs.map +0 -1
  194. package/history.d.ts +0 -1
  195. package/history.js +0 -24
  196. package/history.js.map +0 -1
  197. package/history.mjs +0 -22
  198. package/history.mjs.map +0 -1
  199. package/index.js.map +0 -1
  200. package/index.mjs.map +0 -1
  201. package/persist-plugins/async-storage.js.map +0 -1
  202. package/persist-plugins/async-storage.mjs.map +0 -1
  203. package/persist-plugins/indexeddb.js.map +0 -1
  204. package/persist-plugins/indexeddb.mjs.map +0 -1
  205. package/persist-plugins/local-storage.js.map +0 -1
  206. package/persist-plugins/local-storage.mjs.map +0 -1
  207. package/persist-plugins/mmkv.js.map +0 -1
  208. package/persist-plugins/mmkv.mjs.map +0 -1
  209. package/react-hooks/createObservableHook.js.map +0 -1
  210. package/react-hooks/createObservableHook.mjs.map +0 -1
  211. package/react-hooks/useHover.js.map +0 -1
  212. package/react-hooks/useHover.mjs.map +0 -1
  213. package/react-hooks/useMeasure.js.map +0 -1
  214. package/react-hooks/useMeasure.mjs.map +0 -1
  215. package/react-hooks/useObservableNextRouter.js.map +0 -1
  216. package/react-hooks/useObservableNextRouter.mjs.map +0 -1
  217. package/react.js.map +0 -1
  218. package/react.mjs.map +0 -1
  219. package/src/ObservableObject.ts +0 -1350
  220. package/src/ObservablePrimitive.ts +0 -62
  221. package/src/babel/index.ts +0 -83
  222. package/src/batching.ts +0 -357
  223. package/src/computed.ts +0 -18
  224. package/src/config/enable$GetSet.ts +0 -30
  225. package/src/config/enableReactComponents.ts +0 -26
  226. package/src/config/enableReactNativeComponents.ts +0 -102
  227. package/src/config/enableReactTracking.ts +0 -62
  228. package/src/config/enableReactUse.ts +0 -32
  229. package/src/config/enable_PeekAssign.ts +0 -31
  230. package/src/config.ts +0 -47
  231. package/src/createObservable.ts +0 -47
  232. package/src/event.ts +0 -26
  233. package/src/globals.ts +0 -235
  234. package/src/helpers/pageHash.ts +0 -41
  235. package/src/helpers/pageHashParams.ts +0 -55
  236. package/src/helpers/time.ts +0 -30
  237. package/src/helpers.ts +0 -231
  238. package/src/history/trackHistory.ts +0 -29
  239. package/src/history/undoRedo.ts +0 -111
  240. package/src/is.ts +0 -63
  241. package/src/linked.ts +0 -17
  242. package/src/observable.ts +0 -32
  243. package/src/observableInterfaces.ts +0 -151
  244. package/src/observableTypes.ts +0 -232
  245. package/src/observe.ts +0 -89
  246. package/src/old-plugins/firebase.ts +0 -1053
  247. package/src/onChange.ts +0 -146
  248. package/src/persist/configureObservablePersistence.ts +0 -7
  249. package/src/persist/fieldTransformer.ts +0 -149
  250. package/src/persist/observablePersistRemoteFunctionsAdapter.ts +0 -39
  251. package/src/persist/persistObservable.ts +0 -1034
  252. package/src/persist-plugins/async-storage.ts +0 -99
  253. package/src/persist-plugins/indexeddb.ts +0 -432
  254. package/src/persist-plugins/local-storage.ts +0 -86
  255. package/src/persist-plugins/mmkv.ts +0 -91
  256. package/src/proxy.ts +0 -28
  257. package/src/react/Computed.tsx +0 -8
  258. package/src/react/For.tsx +0 -116
  259. package/src/react/Memo.tsx +0 -4
  260. package/src/react/Reactive.tsx +0 -53
  261. package/src/react/Show.tsx +0 -33
  262. package/src/react/Switch.tsx +0 -43
  263. package/src/react/react-globals.ts +0 -3
  264. package/src/react/reactInterfaces.ts +0 -32
  265. package/src/react/reactive-observer.tsx +0 -210
  266. package/src/react/useComputed.ts +0 -36
  267. package/src/react/useEffectOnce.ts +0 -41
  268. package/src/react/useIsMounted.ts +0 -16
  269. package/src/react/useMount.ts +0 -15
  270. package/src/react/useObservable.ts +0 -24
  271. package/src/react/useObservableReducer.ts +0 -52
  272. package/src/react/useObservableState.ts +0 -30
  273. package/src/react/useObserve.ts +0 -54
  274. package/src/react/useObserveEffect.ts +0 -40
  275. package/src/react/usePauseProvider.tsx +0 -16
  276. package/src/react/useSelector.ts +0 -167
  277. package/src/react/useUnmount.ts +0 -8
  278. package/src/react/useWhen.ts +0 -9
  279. package/src/react-hooks/createObservableHook.ts +0 -53
  280. package/src/react-hooks/useHover.ts +0 -40
  281. package/src/react-hooks/useMeasure.ts +0 -48
  282. package/src/react-hooks/useObservableNextRouter.ts +0 -137
  283. package/src/retry.ts +0 -71
  284. package/src/setupTracking.ts +0 -26
  285. package/src/sync/activateSyncedNode.ts +0 -128
  286. package/src/sync/configureObservableSync.ts +0 -7
  287. package/src/sync/persistTypes.ts +0 -216
  288. package/src/sync/syncHelpers.ts +0 -180
  289. package/src/sync/syncObservable.ts +0 -1056
  290. package/src/sync/syncObservableAdapter.ts +0 -31
  291. package/src/sync/syncTypes.ts +0 -189
  292. package/src/sync/synced.ts +0 -21
  293. package/src/sync-plugins/crud.ts +0 -412
  294. package/src/sync-plugins/fetch.ts +0 -80
  295. package/src/sync-plugins/keel.ts +0 -495
  296. package/src/sync-plugins/supabase.ts +0 -249
  297. package/src/sync-plugins/tanstack-query.ts +0 -113
  298. package/src/sync-plugins/tanstack-react-query.ts +0 -12
  299. package/src/trace/traceHelpers.ts +0 -11
  300. package/src/trace/useTraceListeners.ts +0 -34
  301. package/src/trace/useTraceUpdates.ts +0 -24
  302. package/src/trace/useVerifyNotTracking.ts +0 -33
  303. package/src/trace/useVerifyOneRender.ts +0 -10
  304. package/src/trackSelector.ts +0 -52
  305. package/src/tracking.ts +0 -43
  306. package/src/types/babel.d.ts +0 -12
  307. package/src/when.ts +0 -75
  308. package/sync-plugins/crud.js.map +0 -1
  309. package/sync-plugins/crud.mjs.map +0 -1
  310. package/sync-plugins/fetch.js.map +0 -1
  311. package/sync-plugins/fetch.mjs.map +0 -1
  312. package/sync-plugins/keel.js.map +0 -1
  313. package/sync-plugins/keel.mjs.map +0 -1
  314. package/sync-plugins/supabase.js.map +0 -1
  315. package/sync-plugins/supabase.mjs.map +0 -1
  316. package/sync-plugins/tanstack-query.js.map +0 -1
  317. package/sync-plugins/tanstack-query.mjs.map +0 -1
  318. package/sync-plugins/tanstack-react-query.js.map +0 -1
  319. package/sync-plugins/tanstack-react-query.mjs.map +0 -1
  320. package/sync.js.map +0 -1
  321. package/sync.mjs.map +0 -1
  322. package/trace.js.map +0 -1
  323. package/trace.mjs.map +0 -1
@@ -1,99 +0,0 @@
1
- import type { Change } from '@legendapp/state';
2
- import { applyChanges, internal, isArray } from '@legendapp/state';
3
- import type {
4
- ObservablePersistPlugin,
5
- ObservablePersistenceConfigLocalGlobalOptions,
6
- PersistMetadata,
7
- } from '@legendapp/state/sync';
8
- import type { AsyncStorageStatic } from '@react-native-async-storage/async-storage';
9
-
10
- const MetadataSuffix = '__m';
11
-
12
- let AsyncStorage: AsyncStorageStatic;
13
-
14
- const { safeParse, safeStringify } = internal;
15
-
16
- export class ObservablePersistAsyncStorage implements ObservablePersistPlugin {
17
- private data: Record<string, any> = {};
18
-
19
- // Init
20
- public async initialize(config: ObservablePersistenceConfigLocalGlobalOptions) {
21
- let tables: readonly string[] = [];
22
- const storageConfig = config.asyncStorage;
23
- if (storageConfig) {
24
- AsyncStorage = storageConfig.AsyncStorage;
25
- const { preload } = storageConfig;
26
- try {
27
- if (preload === true) {
28
- // If preloadAllKeys, load all keys and preload tables on startup
29
- tables = await AsyncStorage.getAllKeys();
30
- } else if (isArray(preload)) {
31
- // If preloadKeys, preload load the tables on startup
32
- tables = preload;
33
- }
34
- if (tables) {
35
- const values = await AsyncStorage.multiGet(tables);
36
-
37
- values.forEach(([table, value]) => {
38
- this.data[table] = value ? safeParse(value) : undefined;
39
- });
40
- }
41
- } catch (e) {
42
- console.error('[legend-state] ObservablePersistAsyncStorage failed to initialize', e);
43
- }
44
- } else {
45
- console.error('[legend-state] Missing asyncStorage configuration');
46
- }
47
- }
48
- public loadTable(table: string): void | Promise<void> {
49
- if (this.data[table] === undefined) {
50
- try {
51
- return (async () => {
52
- const value = await AsyncStorage.getItem(table);
53
- this.data[table] = value ? safeParse(value) : undefined;
54
- })();
55
- } catch {
56
- console.error('[legend-state] ObservablePersistLocalAsyncStorage failed to parse', table);
57
- }
58
- }
59
- }
60
- // Gets
61
- public getTable(table: string, init: object) {
62
- return this.data[table] ?? init ?? {};
63
- }
64
- public getMetadata(table: string): PersistMetadata {
65
- return this.getTable(table + MetadataSuffix, {});
66
- }
67
- // Sets
68
- public set(table: string, changes: Change[]): Promise<void> {
69
- if (!this.data[table]) {
70
- this.data[table] = {};
71
- }
72
-
73
- this.data[table] = applyChanges(this.data[table], changes);
74
- return this.save(table);
75
- }
76
- public setMetadata(table: string, metadata: PersistMetadata) {
77
- return this.setValue(table + MetadataSuffix, metadata);
78
- }
79
- public async deleteTable(table: string) {
80
- return AsyncStorage.removeItem(table);
81
- }
82
- public deleteMetadata(table: string) {
83
- return this.deleteTable(table + MetadataSuffix);
84
- }
85
- // Private
86
- private async setValue(table: string, value: any) {
87
- this.data[table] = value;
88
- await this.save(table);
89
- }
90
- private async save(table: string) {
91
- const v = this.data[table];
92
-
93
- if (v !== undefined && v !== null) {
94
- return AsyncStorage.setItem(table, safeStringify(v));
95
- } else {
96
- return AsyncStorage.removeItem(table);
97
- }
98
- }
99
- }
@@ -1,432 +0,0 @@
1
- import type { Change, Observable } from '@legendapp/state';
2
- import type {
3
- ObservablePersistPluginOptions,
4
- ObservablePersistPlugin,
5
- PersistMetadata,
6
- PersistOptions,
7
- } from '@legendapp/state/sync';
8
- import { isPrimitive, isPromise, observable, setAtPath, when } from '@legendapp/state';
9
-
10
- const MetadataSuffix = '__legend_metadata';
11
-
12
- function requestToPromise(request: IDBRequest) {
13
- return new Promise<void>((resolve) => (request.onsuccess = () => resolve()));
14
- }
15
-
16
- export class ObservablePersistIndexedDB implements ObservablePersistPlugin {
17
- private tableData: Record<string, any> = {};
18
- private tableMetadata: Record<string, any> = {};
19
- private tablesAdjusted: Map<string, Observable<boolean>> = new Map();
20
- private db: IDBDatabase | undefined;
21
- private isSaveTaskQueued = false;
22
- private pendingSaves = new Map<
23
- PersistOptions,
24
- Record<string, { tableName: string; tablePrev?: any; items: Set<string> }>
25
- >();
26
- private promisesQueued: (() => void)[] = [];
27
-
28
- constructor() {
29
- this.doSave = this.doSave.bind(this);
30
- }
31
-
32
- public async initialize(config: ObservablePersistPluginOptions) {
33
- if (typeof indexedDB === 'undefined') return;
34
- if (process.env.NODE_ENV === 'development' && !config?.indexedDB) {
35
- console.error('[legend-state] Must configure ObservablePersistIndexedDB');
36
- }
37
-
38
- const { databaseName, version, tableNames } = config!.indexedDB!;
39
- const openRequest = indexedDB.open(databaseName, version);
40
-
41
- openRequest.onerror = () => {
42
- console.error('Error', openRequest.error);
43
- };
44
-
45
- openRequest.onupgradeneeded = () => {
46
- const db = openRequest.result;
47
- const { tableNames } = config!.indexedDB!;
48
- // Create a table for each name with "id" as the key
49
- tableNames.forEach((table) => {
50
- if (!db.objectStoreNames.contains(table)) {
51
- db.createObjectStore(table, {
52
- keyPath: 'id',
53
- });
54
- }
55
- });
56
- };
57
-
58
- return new Promise<void>((resolve) => {
59
- openRequest.onsuccess = async () => {
60
- this.db = openRequest.result;
61
-
62
- // Load each table
63
- const objectStoreNames = this.db.objectStoreNames;
64
- const tables = tableNames.filter((table) => objectStoreNames.contains(table));
65
- try {
66
- const transaction = this.db.transaction(tables, 'readonly');
67
-
68
- await Promise.all(tables.map((table) => this.initTable(table, transaction)));
69
- } catch (err) {
70
- console.error('[legend-state] Error loading IndexedDB', err);
71
- }
72
-
73
- resolve();
74
- };
75
- });
76
- }
77
- public loadTable(table: string, config: PersistOptions): void | Promise<void> {
78
- if (!this.tableData[table]) {
79
- const transaction = this.db!.transaction(table, 'readonly');
80
-
81
- return this.initTable(table, transaction).then(() => this.loadTable(table, config));
82
- }
83
-
84
- const prefix = config.indexedDB?.prefixID;
85
-
86
- if (prefix) {
87
- const tableName = prefix ? table + '/' + prefix : table;
88
- if (this.tablesAdjusted.has(tableName)) {
89
- const promise = when(this.tablesAdjusted.get(tableName)!);
90
- if (isPromise(promise)) {
91
- return promise as unknown as Promise<void>;
92
- }
93
- } else {
94
- const obsLoaded = observable(false);
95
- this.tablesAdjusted.set(tableName, obsLoaded);
96
- const data = this.getTable(table, {}, config);
97
- let hasPromise = false;
98
- let promises: Promise<any>[];
99
- if (data) {
100
- const keys = Object.keys(data);
101
- promises = keys.map(async (key) => {
102
- const value = data[key];
103
-
104
- if (isPromise(value)) {
105
- hasPromise = true;
106
- return value.then(() => {
107
- data[key] = value;
108
- });
109
- } else {
110
- data[key] = value;
111
- }
112
- });
113
- }
114
- if (hasPromise) {
115
- return Promise.all(promises!).then(() => {
116
- obsLoaded.set(true);
117
- });
118
- } else {
119
- obsLoaded.set(true);
120
- }
121
- }
122
- }
123
- }
124
- public getTable(table: string, init: object, config: PersistOptions) {
125
- const configIDB = config.indexedDB;
126
- const prefix = configIDB?.prefixID;
127
- const data = this.tableData[prefix ? table + '/' + prefix : table];
128
- if (data && configIDB?.itemID) {
129
- return data[configIDB.itemID];
130
- } else {
131
- return data;
132
- }
133
- }
134
- public getMetadata(table: string, config: PersistOptions) {
135
- const { tableName } = this.getMetadataTableName(table, config);
136
- return this.tableMetadata[tableName];
137
- }
138
- public async setMetadata(table: string, metadata: PersistMetadata, config: PersistOptions) {
139
- const { tableName, tableNameBase } = this.getMetadataTableName(table, config);
140
- // Assign new metadata into the table, and make sure it has the id
141
- this.tableMetadata[tableName] = Object.assign(metadata, {
142
- id: tableNameBase + MetadataSuffix,
143
- });
144
- this.tableMetadata[tableName] = metadata;
145
- const store = this.transactionStore(table);
146
- return store.put(metadata);
147
- }
148
- public async deleteMetadata(table: string, config: PersistOptions): Promise<void> {
149
- const { tableName, tableNameBase } = this.getMetadataTableName(table, config);
150
- delete this.tableMetadata[tableName];
151
- const store = this.transactionStore(table);
152
- const key = tableNameBase + MetadataSuffix;
153
- store.delete(key);
154
- }
155
- public async set(table: string, changes: Change[], config: PersistOptions) {
156
- if (typeof indexedDB === 'undefined') return;
157
-
158
- if (!this.pendingSaves.has(config)) {
159
- this.pendingSaves.set(config, {});
160
- }
161
- const pendingSaves = this.pendingSaves.get(config)!;
162
-
163
- const realTable = table;
164
- const prefixID = config.indexedDB?.prefixID;
165
- if (prefixID) {
166
- table += '/' + prefixID;
167
- }
168
- const prev = this.tableData[table];
169
-
170
- const itemID = config.indexedDB?.itemID;
171
-
172
- if (!pendingSaves[table]) {
173
- pendingSaves[table] = { tableName: realTable, items: new Set() };
174
- }
175
-
176
- const pendingTable = pendingSaves[table];
177
-
178
- // Combine changes into a minimal set of saves
179
- for (let i = 0; i < changes.length; i++) {
180
- // eslint-disable-next-line prefer-const
181
- let { path, valueAtPath, pathTypes } = changes[i];
182
- if (itemID) {
183
- path = [itemID].concat(path as string[]);
184
- pathTypes.splice(0, 0, 'object');
185
- }
186
- if (path.length > 0) {
187
- // If change is deep in an object save it to IDB by the first key
188
- const key = path[0] as string;
189
- if (!this.tableData[table]) {
190
- this.tableData[table] = {};
191
- }
192
- this.tableData[table] = setAtPath(this.tableData[table], path as string[], pathTypes, valueAtPath);
193
- pendingTable.items.add(key);
194
- } else {
195
- // Set the whole table
196
- this.tableData[table] = valueAtPath;
197
- pendingTable.tablePrev = prev;
198
- break;
199
- }
200
- }
201
-
202
- return new Promise<void>((resolve) => {
203
- this.promisesQueued.push(resolve);
204
-
205
- if (!this.isSaveTaskQueued) {
206
- this.isSaveTaskQueued = true;
207
- queueMicrotask(this.doSave);
208
- }
209
- });
210
- }
211
- private async doSave() {
212
- this.isSaveTaskQueued = false;
213
- const promisesQueued = this.promisesQueued;
214
- this.promisesQueued = [];
215
- const promises: Promise<IDBRequest>[] = [];
216
- let lastPut: IDBRequest | undefined;
217
- this.pendingSaves.forEach((pendingSaves, config) => {
218
- Object.keys(pendingSaves).forEach((table) => {
219
- const pendingTable = pendingSaves[table];
220
- const { tablePrev, items, tableName } = pendingTable;
221
- const store = this.transactionStore(tableName);
222
- const tableValue = this.tableData[table];
223
- if (tablePrev) {
224
- promises.push(this._setTable(table, tablePrev, tableValue, store, config));
225
- } else {
226
- items.forEach((key) => {
227
- lastPut = this._setItem(table, key, tableValue[key], store, config);
228
- });
229
- }
230
-
231
- // Clear pending saves
232
- items.clear();
233
- delete pendingTable.tablePrev;
234
- });
235
- });
236
- this.pendingSaves.clear();
237
-
238
- // setTable awaits multiple sets and deletes so we need to await that to get
239
- // the lastPut from it.
240
- if (promises.length) {
241
- const puts = await Promise.all(promises);
242
- lastPut = puts[puts.length - 1];
243
- }
244
-
245
- if (lastPut) {
246
- await requestToPromise(lastPut);
247
- }
248
-
249
- promisesQueued.forEach((resolve) => resolve());
250
- }
251
- public async deleteTable(table: string, config: PersistOptions): Promise<void> {
252
- const configIDB = config.indexedDB;
253
- const prefixID = configIDB?.prefixID;
254
- const tableName = prefixID ? table + '/' + prefixID : table;
255
- let data = this.tableData[tableName];
256
- const itemID = configIDB?.itemID;
257
- if (data && configIDB?.itemID) {
258
- const dataTemp = data[itemID!];
259
- delete data[itemID!];
260
- data = dataTemp;
261
- } else {
262
- delete this.tableData[tableName];
263
- delete this.tableData[tableName + '_transformed'];
264
- }
265
-
266
- if (typeof indexedDB === 'undefined') return;
267
-
268
- this.deleteMetadata(table, config);
269
-
270
- if (data) {
271
- const store = this.transactionStore(table);
272
- let result: Promise<any>;
273
- if (!prefixID && !itemID) {
274
- result = requestToPromise(store.clear());
275
- } else {
276
- const keys = Object.keys(data);
277
- result = Promise.all(
278
- keys.map((key) => {
279
- if (prefixID) {
280
- key = prefixID + '/' + key;
281
- }
282
- return requestToPromise(store.delete(key));
283
- }),
284
- );
285
- }
286
- // Clear the table from IDB
287
- return result;
288
- }
289
- }
290
- // Private
291
- private getMetadataTableName(table: string, config: PersistOptions) {
292
- const configIDB = config.indexedDB;
293
- let name = '';
294
- if (configIDB) {
295
- const { prefixID, itemID } = configIDB;
296
-
297
- if (itemID) {
298
- name = itemID;
299
- }
300
- if (prefixID) {
301
- name = prefixID + (name ? '/' + name : '');
302
- }
303
- }
304
-
305
- return { tableNameBase: name, tableName: name ? table + '/' + name : table };
306
- }
307
- private initTable(table: string, transaction: IDBTransaction): Promise<void> {
308
- const store = transaction.objectStore(table);
309
- const allRequest = store.getAll();
310
-
311
- if (!this.tableData[table]) {
312
- this.tableData[table] = {};
313
- }
314
- return new Promise((resolve) => {
315
- allRequest.onsuccess = () => {
316
- const arr = allRequest.result;
317
- let metadata: PersistMetadata;
318
- if (!this.tableData[table]) {
319
- this.tableData[table] = {};
320
- }
321
- for (let i = 0; i < arr.length; i++) {
322
- const val = arr[i];
323
-
324
- // In case id is a number convert it to a string
325
- if (!val.id.includes) {
326
- val.id = val.id + '';
327
- }
328
-
329
- if (val.id.endsWith(MetadataSuffix)) {
330
- const id = val.id.replace(MetadataSuffix, '');
331
- // Save this as metadata
332
- delete val.id;
333
- metadata = val;
334
- const tableName = id ? table + '/' + id : table;
335
- this.tableMetadata[tableName] = metadata;
336
- } else {
337
- let tableName = table;
338
-
339
- if (val.id.includes('/')) {
340
- const [prefix, id] = val.id.split('/');
341
- tableName += '/' + prefix;
342
- val.id = id;
343
- }
344
-
345
- if (!this.tableData[tableName]) {
346
- this.tableData[tableName] = {};
347
- }
348
- this.tableData[tableName][val.id] = val;
349
- }
350
- }
351
- resolve();
352
- };
353
- });
354
- }
355
- private transactionStore(table: string) {
356
- const transaction = this.db!.transaction(table, 'readwrite');
357
- return transaction.objectStore(table);
358
- }
359
- private _setItem(table: string, key: string, value: any, store: IDBObjectStore, config: PersistOptions) {
360
- if (!value) {
361
- if (this.tableData[table]) {
362
- delete this.tableData[table][key];
363
- }
364
- return store.delete(key);
365
- } else {
366
- if (isPrimitive(value)) return;
367
-
368
- if (value.id === undefined) {
369
- // If value does not have its own ID, assign it the key from the Record
370
- value.id = key;
371
- }
372
-
373
- if (config) {
374
- if (!this.tableData[table]) {
375
- this.tableData[table] = {};
376
- }
377
- this.tableData[table][key] = value;
378
-
379
- const didClone = false;
380
-
381
- const prefixID = config.indexedDB?.prefixID;
382
- if (prefixID) {
383
- if (didClone) {
384
- value.id = prefixID + '/' + value.id;
385
- } else {
386
- value = Object.assign({}, value, {
387
- id: prefixID + '/' + value.id,
388
- });
389
- }
390
- }
391
- }
392
-
393
- return store.put(value);
394
- }
395
- }
396
- private async _setTable(
397
- table: string,
398
- prev: object,
399
- value: Record<string, any>,
400
- store: IDBObjectStore,
401
- config: PersistOptions,
402
- ) {
403
- const keys = Object.keys(value);
404
- let lastSet: IDBRequest | undefined;
405
- // Do a set for each key in the object
406
- const sets = await Promise.all(
407
- keys.map((key) => {
408
- const val = value[key];
409
- return this._setItem(table, key, val, store, config);
410
- }),
411
- );
412
- lastSet = sets[sets.length - 1];
413
-
414
- // Delete keys that are no longer in the object
415
- if (prev) {
416
- const keysOld = Object.keys(prev);
417
- const deletes = (
418
- await Promise.all(
419
- keysOld.map((key) => {
420
- if (value[key] === undefined) {
421
- return this._setItem(table, key, null, store, config);
422
- }
423
- }),
424
- )
425
- ).filter((a) => !!a);
426
- if (deletes.length > 0) {
427
- lastSet = deletes[deletes.length - 1];
428
- }
429
- }
430
- return lastSet!;
431
- }
432
- }
@@ -1,86 +0,0 @@
1
- import type { Change } from '@legendapp/state';
2
- import { applyChanges, internal } from '@legendapp/state';
3
- import type { ObservablePersistPlugin, PersistMetadata } from '@legendapp/state/sync';
4
-
5
- const { safeParse, safeStringify } = internal;
6
-
7
- const MetadataSuffix = '__m';
8
-
9
- export class ObservablePersistLocalStorageBase implements ObservablePersistPlugin {
10
- private data: Record<string, any> = {};
11
- private storage: Storage | undefined;
12
- constructor(storage: Storage | undefined) {
13
- this.storage = storage;
14
- }
15
- public getTable(table: string, init: any) {
16
- if (!this.storage) return undefined;
17
- if (this.data[table] === undefined) {
18
- try {
19
- const value = this.storage.getItem(table);
20
- this.data[table] = value ? safeParse(value) : init;
21
- } catch {
22
- console.error('[legend-state] ObservablePersistLocalStorageBase failed to parse', table);
23
- }
24
- }
25
- return this.data[table];
26
- }
27
- public getMetadata(table: string): PersistMetadata {
28
- return this.getTable(table + MetadataSuffix, {});
29
- }
30
- public set(table: string, changes: Change[]): void {
31
- if (!this.data[table]) {
32
- this.data[table] = {};
33
- }
34
- this.data[table] = applyChanges(this.data[table], changes);
35
- this.save(table);
36
- }
37
- public setMetadata(table: string, metadata: PersistMetadata) {
38
- table = table + MetadataSuffix;
39
- this.data[table] = metadata;
40
- this.save(table);
41
- }
42
- public deleteTable(table: string) {
43
- if (!this.storage) return undefined;
44
- delete this.data[table];
45
- this.storage.removeItem(table);
46
- }
47
- public deleteMetadata(table: string) {
48
- this.deleteTable(table + MetadataSuffix);
49
- }
50
- // Private
51
- private save(table: string) {
52
- if (!this.storage) return undefined;
53
-
54
- const v = this.data[table];
55
-
56
- if (v !== undefined && v !== null) {
57
- this.storage.setItem(table, safeStringify(v));
58
- } else {
59
- this.storage.removeItem(table);
60
- }
61
- }
62
- }
63
- export class ObservablePersistLocalStorage extends ObservablePersistLocalStorageBase {
64
- constructor() {
65
- super(
66
- typeof localStorage !== 'undefined'
67
- ? localStorage
68
- : process.env.NODE_ENV === 'test'
69
- ? // @ts-expect-error This is ok to do in jest
70
- globalThis._testlocalStorage
71
- : undefined,
72
- );
73
- }
74
- }
75
- export class ObservablePersistSessionStorage extends ObservablePersistLocalStorageBase {
76
- constructor() {
77
- super(
78
- typeof sessionStorage !== 'undefined'
79
- ? sessionStorage
80
- : process.env.NODE_ENV === 'test'
81
- ? // @ts-expect-error This is ok to do in jest
82
- globalThis._testlocalStorage
83
- : undefined,
84
- );
85
- }
86
- }