@legendapp/state 2.2.0-next.8 → 2.2.0-next.80

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 (321) 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 +34 -32
  14. package/config/enableReactTracking.js.map +1 -1
  15. package/config/enableReactTracking.mjs +35 -33
  16. package/config/enableReactTracking.mjs.map +1 -1
  17. package/config/enableReactUse.d.ts +1 -1
  18. package/config/enableReactUse.js +9 -1
  19. package/config/enableReactUse.js.map +1 -1
  20. package/config/enableReactUse.mjs +9 -1
  21. package/config/enableReactUse.mjs.map +1 -1
  22. package/config/enable_peek.d.ts +8 -0
  23. package/config/{enableDirectPeek.js → enable_peek.js} +6 -3
  24. package/config/enable_peek.js.map +1 -0
  25. package/config/{enableDirectPeek.mjs → enable_peek.mjs} +5 -3
  26. package/config/enable_peek.mjs.map +1 -0
  27. package/helpers/fetch.d.ts +4 -3
  28. package/helpers/fetch.js.map +1 -1
  29. package/helpers/fetch.mjs.map +1 -1
  30. package/helpers/pageHash.js.map +1 -1
  31. package/helpers/pageHash.mjs.map +1 -1
  32. package/helpers/pageHashParams.js.map +1 -1
  33. package/helpers/pageHashParams.mjs.map +1 -1
  34. package/helpers/time.d.ts +2 -2
  35. package/helpers/time.js.map +1 -1
  36. package/helpers/time.mjs.map +1 -1
  37. package/history.js +2 -2
  38. package/history.js.map +1 -1
  39. package/history.mjs +3 -3
  40. package/history.mjs.map +1 -1
  41. package/index.d.ts +29 -9
  42. package/index.js +964 -671
  43. package/index.js.map +1 -1
  44. package/index.mjs +959 -668
  45. package/index.mjs.map +1 -1
  46. package/package.json +37 -25
  47. package/persist-plugins/async-storage.d.ts +4 -3
  48. package/persist-plugins/async-storage.js +8 -7
  49. package/persist-plugins/async-storage.js.map +1 -1
  50. package/persist-plugins/async-storage.mjs +9 -8
  51. package/persist-plugins/async-storage.mjs.map +1 -1
  52. package/persist-plugins/fetch.d.ts +1 -1
  53. package/persist-plugins/fetch.js.map +1 -1
  54. package/persist-plugins/fetch.mjs.map +1 -1
  55. package/persist-plugins/firebase.d.ts +2 -2
  56. package/persist-plugins/firebase.js +12 -11
  57. package/persist-plugins/firebase.js.map +1 -1
  58. package/persist-plugins/firebase.mjs +13 -12
  59. package/persist-plugins/firebase.mjs.map +1 -1
  60. package/persist-plugins/indexeddb.d.ts +11 -10
  61. package/persist-plugins/indexeddb.js +2 -2
  62. package/persist-plugins/indexeddb.js.map +1 -1
  63. package/persist-plugins/indexeddb.mjs +2 -2
  64. package/persist-plugins/indexeddb.mjs.map +1 -1
  65. package/persist-plugins/local-storage.d.ts +4 -4
  66. package/persist-plugins/local-storage.js +19 -7
  67. package/persist-plugins/local-storage.js.map +1 -1
  68. package/persist-plugins/local-storage.mjs +20 -9
  69. package/persist-plugins/local-storage.mjs.map +1 -1
  70. package/persist-plugins/mmkv.d.ts +9 -8
  71. package/persist-plugins/mmkv.js +5 -4
  72. package/persist-plugins/mmkv.js.map +1 -1
  73. package/persist-plugins/mmkv.mjs +6 -5
  74. package/persist-plugins/mmkv.mjs.map +1 -1
  75. package/persist-plugins/query.d.ts +1 -1
  76. package/persist-plugins/query.js.map +1 -1
  77. package/persist-plugins/query.mjs.map +1 -1
  78. package/persist.d.ts +2 -14
  79. package/persist.js +1270 -269
  80. package/persist.js.map +1 -1
  81. package/persist.mjs +1270 -270
  82. package/persist.mjs.map +1 -1
  83. package/react-hooks/createObservableHook.js +1 -1
  84. package/react-hooks/createObservableHook.js.map +1 -1
  85. package/react-hooks/createObservableHook.mjs +1 -1
  86. package/react-hooks/createObservableHook.mjs.map +1 -1
  87. package/react-hooks/useFetch.d.ts +4 -3
  88. package/react-hooks/useFetch.js.map +1 -1
  89. package/react-hooks/useFetch.mjs.map +1 -1
  90. package/react-hooks/useHover.js.map +1 -1
  91. package/react-hooks/useHover.mjs.map +1 -1
  92. package/react-hooks/useMeasure.js.map +1 -1
  93. package/react-hooks/useMeasure.mjs.map +1 -1
  94. package/react-hooks/useObservableNextRouter.js.map +1 -1
  95. package/react-hooks/useObservableNextRouter.mjs.map +1 -1
  96. package/react-hooks/useObservableQuery.js.map +1 -1
  97. package/react-hooks/useObservableQuery.mjs.map +1 -1
  98. package/react-hooks/usePersistedObservable.d.ts +6 -3
  99. package/react-hooks/usePersistedObservable.js +5 -2
  100. package/react-hooks/usePersistedObservable.js.map +1 -1
  101. package/react-hooks/usePersistedObservable.mjs +5 -2
  102. package/react-hooks/usePersistedObservable.mjs.map +1 -1
  103. package/react.js +73 -93
  104. package/react.js.map +1 -1
  105. package/react.mjs +73 -93
  106. package/react.mjs.map +1 -1
  107. package/src/ObservableObject.ts +1217 -0
  108. package/src/ObservablePrimitive.ts +62 -0
  109. package/src/babel/index.ts +70 -0
  110. package/src/batching.ts +378 -0
  111. package/src/computed.ts +18 -0
  112. package/src/config/enable$get.ts +30 -0
  113. package/src/config/enableReactComponents.ts +26 -0
  114. package/src/config/enableReactNativeComponents.ts +102 -0
  115. package/src/config/enableReactTracking.ts +62 -0
  116. package/src/config/enableReactUse.ts +32 -0
  117. package/src/config/enable_peek.ts +31 -0
  118. package/src/config.ts +47 -0
  119. package/src/createObservable.ts +46 -0
  120. package/src/event.ts +26 -0
  121. package/src/globals.ts +234 -0
  122. package/src/helpers/fetch.ts +26 -0
  123. package/src/helpers/pageHash.ts +41 -0
  124. package/src/helpers/pageHashParams.ts +55 -0
  125. package/src/helpers/time.ts +30 -0
  126. package/src/helpers.ts +224 -0
  127. package/src/history/trackHistory.ts +29 -0
  128. package/src/history/undoRedo.ts +111 -0
  129. package/src/is.ts +63 -0
  130. package/src/linked.ts +6 -0
  131. package/src/observable.ts +32 -0
  132. package/src/observableInterfaces.ts +148 -0
  133. package/src/observableTypes.ts +226 -0
  134. package/src/observe.ts +89 -0
  135. package/src/onChange.ts +136 -0
  136. package/src/persist/configureObservablePersistence.ts +7 -0
  137. package/src/persist/fieldTransformer.ts +149 -0
  138. package/src/persist/observablePersistRemoteFunctionsAdapter.ts +39 -0
  139. package/src/persist/persistObservable.ts +1031 -0
  140. package/src/persist-plugins/async-storage.ts +102 -0
  141. package/src/persist-plugins/fetch.ts +34 -0
  142. package/src/persist-plugins/firebase.ts +1052 -0
  143. package/src/persist-plugins/indexeddb.ts +432 -0
  144. package/src/persist-plugins/local-storage.ts +91 -0
  145. package/src/persist-plugins/mmkv.ts +91 -0
  146. package/src/persist-plugins/query.ts +129 -0
  147. package/src/proxy.ts +28 -0
  148. package/src/react/Computed.tsx +7 -0
  149. package/src/react/For.tsx +116 -0
  150. package/src/react/Memo.tsx +4 -0
  151. package/src/react/Reactive.tsx +53 -0
  152. package/src/react/Show.tsx +33 -0
  153. package/src/react/Switch.tsx +43 -0
  154. package/src/react/react-globals.ts +3 -0
  155. package/src/react/{reactInterfaces.d.ts → reactInterfaces.ts} +15 -7
  156. package/src/react/reactive-observer.tsx +210 -0
  157. package/src/react/useComputed.ts +36 -0
  158. package/src/react/useEffectOnce.ts +41 -0
  159. package/src/react/useIsMounted.ts +16 -0
  160. package/src/react/useMount.ts +15 -0
  161. package/src/react/useObservable.ts +24 -0
  162. package/src/react/useObservableReducer.ts +52 -0
  163. package/src/react/useObservableState.ts +30 -0
  164. package/src/react/useObserve.ts +54 -0
  165. package/src/react/useObserveEffect.ts +40 -0
  166. package/src/react/usePauseProvider.tsx +13 -0
  167. package/src/react/useSelector.ts +167 -0
  168. package/src/react/useUnmount.ts +8 -0
  169. package/src/react/useWhen.ts +9 -0
  170. package/src/react-hooks/createObservableHook.ts +53 -0
  171. package/src/react-hooks/useFetch.ts +16 -0
  172. package/src/react-hooks/useHover.ts +40 -0
  173. package/src/react-hooks/useMeasure.ts +48 -0
  174. package/src/react-hooks/useObservableNextRouter.ts +137 -0
  175. package/src/react-hooks/useObservableQuery.ts +205 -0
  176. package/src/react-hooks/usePersistedObservable.ts +25 -0
  177. package/src/retry.ts +71 -0
  178. package/src/setupTracking.ts +26 -0
  179. package/src/sync/activateSyncedNode.ts +128 -0
  180. package/src/sync/configureObservableSync.ts +7 -0
  181. package/src/sync/persistTypes.ts +226 -0
  182. package/src/sync/syncHelpers.ts +56 -0
  183. package/src/sync/syncObservable.ts +1040 -0
  184. package/src/sync/syncObservableAdapter.ts +31 -0
  185. package/src/sync/syncTypes.ts +188 -0
  186. package/src/sync/synced.ts +20 -0
  187. package/src/sync-plugins/crud.ts +404 -0
  188. package/src/sync-plugins/fetch.ts +72 -0
  189. package/src/sync-plugins/keel.ts +452 -0
  190. package/src/sync-plugins/supabase.ts +261 -0
  191. package/src/trace/traceHelpers.ts +11 -0
  192. package/src/trace/useTraceListeners.ts +34 -0
  193. package/src/trace/useTraceUpdates.ts +24 -0
  194. package/src/trace/useVerifyNotTracking.ts +33 -0
  195. package/src/trace/useVerifyOneRender.ts +10 -0
  196. package/src/trackSelector.ts +52 -0
  197. package/src/tracking.ts +43 -0
  198. package/src/types/babel.d.ts +12 -0
  199. package/src/when.ts +75 -0
  200. package/sync-plugins/crud.d.ts +41 -0
  201. package/sync-plugins/crud.js +290 -0
  202. package/sync-plugins/crud.js.map +1 -0
  203. package/sync-plugins/crud.mjs +286 -0
  204. package/sync-plugins/crud.mjs.map +1 -0
  205. package/sync-plugins/fetch.d.ts +13 -0
  206. package/sync-plugins/fetch.js +46 -0
  207. package/sync-plugins/fetch.js.map +1 -0
  208. package/sync-plugins/fetch.mjs +44 -0
  209. package/sync-plugins/fetch.mjs.map +1 -0
  210. package/sync-plugins/keel.d.ts +91 -0
  211. package/sync-plugins/keel.js +277 -0
  212. package/sync-plugins/keel.js.map +1 -0
  213. package/sync-plugins/keel.mjs +273 -0
  214. package/sync-plugins/keel.mjs.map +1 -0
  215. package/sync-plugins/supabase.d.ts +36 -0
  216. package/sync-plugins/supabase.js +152 -0
  217. package/sync-plugins/supabase.js.map +1 -0
  218. package/sync-plugins/supabase.mjs +149 -0
  219. package/sync-plugins/supabase.mjs.map +1 -0
  220. package/sync.d.ts +11 -0
  221. package/sync.js +976 -0
  222. package/sync.js.map +1 -0
  223. package/sync.mjs +966 -0
  224. package/sync.mjs.map +1 -0
  225. package/trace.js +13 -10
  226. package/trace.js.map +1 -1
  227. package/trace.mjs +11 -8
  228. package/trace.mjs.map +1 -1
  229. package/types/babel.d.ts +3 -3
  230. package/config/enableDirectAccess.d.ts +0 -7
  231. package/config/enableDirectAccess.js +0 -25
  232. package/config/enableDirectAccess.js.map +0 -1
  233. package/config/enableDirectAccess.mjs +0 -23
  234. package/config/enableDirectAccess.mjs.map +0 -1
  235. package/config/enableDirectPeek.d.ts +0 -7
  236. package/config/enableDirectPeek.js.map +0 -1
  237. package/config/enableDirectPeek.mjs.map +0 -1
  238. package/config/enableReactDirectRender.d.ts +0 -2
  239. package/config/enableReactDirectRender.js +0 -78
  240. package/config/enableReactDirectRender.js.map +0 -1
  241. package/config/enableReactDirectRender.mjs +0 -75
  242. package/config/enableReactDirectRender.mjs.map +0 -1
  243. package/src/ObservableObject.d.ts +0 -14
  244. package/src/ObservablePrimitive.d.ts +0 -7
  245. package/src/babel/index.d.ts +0 -17
  246. package/src/batching.d.ts +0 -6
  247. package/src/computed.d.ts +0 -4
  248. package/src/config/enableDirectAccess.d.ts +0 -7
  249. package/src/config/enableDirectPeek.d.ts +0 -7
  250. package/src/config/enableReactComponents.d.ts +0 -7
  251. package/src/config/enableReactDirectRender.d.ts +0 -2
  252. package/src/config/enableReactNativeComponents.d.ts +0 -20
  253. package/src/config/enableReactTracking.d.ts +0 -15
  254. package/src/config/enableReactUse.d.ts +0 -7
  255. package/src/config.d.ts +0 -8
  256. package/src/createObservable.d.ts +0 -2
  257. package/src/event.d.ts +0 -2
  258. package/src/globals.d.ts +0 -32
  259. package/src/helpers/fetch.d.ts +0 -6
  260. package/src/helpers/pageHash.d.ts +0 -7
  261. package/src/helpers/pageHashParams.d.ts +0 -7
  262. package/src/helpers/time.d.ts +0 -3
  263. package/src/helpers.d.ts +0 -13
  264. package/src/history/trackHistory.d.ts +0 -4
  265. package/src/is.d.ts +0 -10
  266. package/src/observable.d.ts +0 -16
  267. package/src/observableInterfaces.d.ts +0 -458
  268. package/src/observe.d.ts +0 -6
  269. package/src/onChange.d.ts +0 -7
  270. package/src/persist/configureObservablePersistence.d.ts +0 -3
  271. package/src/persist/fieldTransformer.d.ts +0 -8
  272. package/src/persist/observablePersistRemoteFunctionsAdapter.d.ts +0 -2
  273. package/src/persist/persistActivateNode.d.ts +0 -1
  274. package/src/persist/persistHelpers.d.ts +0 -1
  275. package/src/persist/persistObservable.d.ts +0 -25
  276. package/src/persist-plugins/async-storage.d.ts +0 -14
  277. package/src/persist-plugins/fetch.d.ts +0 -10
  278. package/src/persist-plugins/firebase.d.ts +0 -51
  279. package/src/persist-plugins/indexeddb.d.ts +0 -25
  280. package/src/persist-plugins/local-storage.d.ts +0 -21
  281. package/src/persist-plugins/mmkv.d.ts +0 -14
  282. package/src/persist-plugins/query.d.ts +0 -18
  283. package/src/proxy.d.ts +0 -5
  284. package/src/react/Computed.d.ts +0 -5
  285. package/src/react/For.d.ts +0 -15
  286. package/src/react/Memo.d.ts +0 -3
  287. package/src/react/Reactive.d.ts +0 -9
  288. package/src/react/Show.d.ts +0 -18
  289. package/src/react/Switch.d.ts +0 -14
  290. package/src/react/react-globals.d.ts +0 -3
  291. package/src/react/reactive-observer.d.ts +0 -14
  292. package/src/react/useComputed.d.ts +0 -5
  293. package/src/react/useEffectOnce.d.ts +0 -1
  294. package/src/react/useIsMounted.d.ts +0 -2
  295. package/src/react/useMount.d.ts +0 -2
  296. package/src/react/useObservable.d.ts +0 -9
  297. package/src/react/useObservableReducer.d.ts +0 -7
  298. package/src/react/useObservableState.d.ts +0 -2
  299. package/src/react/useObserve.d.ts +0 -4
  300. package/src/react/useObserveEffect.d.ts +0 -4
  301. package/src/react/usePauseProvider.d.ts +0 -8
  302. package/src/react/useSelector.d.ts +0 -3
  303. package/src/react/useUnmount.d.ts +0 -2
  304. package/src/react/useWhen.d.ts +0 -3
  305. package/src/react-hooks/createObservableHook.d.ts +0 -2
  306. package/src/react-hooks/useFetch.d.ts +0 -6
  307. package/src/react-hooks/useHover.d.ts +0 -3
  308. package/src/react-hooks/useMeasure.d.ts +0 -6
  309. package/src/react-hooks/useObservableNextRouter.d.ts +0 -33
  310. package/src/react-hooks/useObservableQuery.d.ts +0 -6
  311. package/src/react-hooks/usePersistedObservable.d.ts +0 -11
  312. package/src/retry.d.ts +0 -9
  313. package/src/setupTracking.d.ts +0 -2
  314. package/src/trace/traceHelpers.d.ts +0 -2
  315. package/src/trace/useTraceListeners.d.ts +0 -1
  316. package/src/trace/useTraceUpdates.d.ts +0 -1
  317. package/src/trace/useVerifyNotTracking.d.ts +0 -1
  318. package/src/trace/useVerifyOneRender.d.ts +0 -1
  319. package/src/trackSelector.d.ts +0 -7
  320. package/src/tracking.d.ts +0 -13
  321. package/src/when.d.ts +0 -3
@@ -0,0 +1,432 @@
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
+ }
@@ -0,0 +1,91 @@
1
+ import type { Change } from '@legendapp/state';
2
+ import { internal, setAtPath } from '@legendapp/state';
3
+ import type { ObservablePersistPlugin, PersistMetadata } from '@legendapp/state/sync';
4
+
5
+ const MetadataSuffix = '__m';
6
+
7
+ const { safeParse, safeStringify } = internal;
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
+ for (let i = 0; i < changes.length; i++) {
35
+ const { path, valueAtPath, pathTypes } = changes[i];
36
+ this.data[table] = setAtPath(this.data[table], path as string[], pathTypes, valueAtPath);
37
+ }
38
+ this.save(table);
39
+ }
40
+ public setMetadata(table: string, metadata: PersistMetadata) {
41
+ return this.setValue(table + MetadataSuffix, metadata);
42
+ }
43
+ public deleteTable(table: string) {
44
+ if (!this.storage) return undefined;
45
+ delete this.data[table];
46
+ this.storage.removeItem(table);
47
+ }
48
+ public deleteMetadata(table: string) {
49
+ this.deleteTable(table + MetadataSuffix);
50
+ }
51
+ // Private
52
+ private setValue(table: string, value: any) {
53
+ this.data[table] = value;
54
+ this.save(table);
55
+ }
56
+ private save(table: string) {
57
+ if (!this.storage) return undefined;
58
+
59
+ const v = this.data[table];
60
+
61
+ if (v !== undefined && v !== null) {
62
+ this.storage.setItem(table, safeStringify(v));
63
+ } else {
64
+ this.storage.removeItem(table);
65
+ }
66
+ }
67
+ }
68
+ export class ObservablePersistLocalStorage extends ObservablePersistLocalStorageBase {
69
+ constructor() {
70
+ super(
71
+ typeof localStorage !== 'undefined'
72
+ ? localStorage
73
+ : process.env.NODE_ENV === 'test'
74
+ ? // @ts-expect-error This is ok to do in jest
75
+ globalThis._testlocalStorage
76
+ : undefined,
77
+ );
78
+ }
79
+ }
80
+ export class ObservablePersistSessionStorage extends ObservablePersistLocalStorageBase {
81
+ constructor() {
82
+ super(
83
+ typeof sessionStorage !== 'undefined'
84
+ ? sessionStorage
85
+ : process.env.NODE_ENV === 'test'
86
+ ? // @ts-expect-error This is ok to do in jest
87
+ globalThis._testlocalStorage
88
+ : undefined,
89
+ );
90
+ }
91
+ }
@@ -0,0 +1,91 @@
1
+ import type { Change } from '@legendapp/state';
2
+ import { internal, setAtPath } from '@legendapp/state';
3
+ import type { ObservablePersistPlugin, PersistMetadata, PersistOptions } from '@legendapp/state/sync';
4
+ import { MMKV } from 'react-native-mmkv';
5
+
6
+ const symbolDefault = Symbol();
7
+ const MetadataSuffix = '__m';
8
+
9
+ const { safeParse, safeStringify } = internal;
10
+
11
+ export class ObservablePersistMMKV implements ObservablePersistPlugin {
12
+ private data: Record<string, any> = {};
13
+ private storages = new Map<symbol | string, MMKV>([
14
+ [
15
+ symbolDefault,
16
+ new MMKV({
17
+ id: `obsPersist`,
18
+ }),
19
+ ],
20
+ ]);
21
+ // Gets
22
+ public getTable<T = any>(table: string, init: object, config: PersistOptions): T {
23
+ const storage = this.getStorage(config);
24
+ if (this.data[table] === undefined) {
25
+ try {
26
+ const value = storage.getString(table);
27
+ this.data[table] = value ? safeParse(value) : init;
28
+ } catch {
29
+ console.error('[legend-state] MMKV failed to parse', table);
30
+ }
31
+ }
32
+ return this.data[table];
33
+ }
34
+ public getMetadata(table: string, config: PersistOptions): PersistMetadata {
35
+ return this.getTable(table + MetadataSuffix, {}, config);
36
+ }
37
+ // Sets
38
+ public set(table: string, changes: Change[], config: PersistOptions) {
39
+ if (!this.data[table]) {
40
+ this.data[table] = {};
41
+ }
42
+ for (let i = 0; i < changes.length; i++) {
43
+ const { path, valueAtPath, pathTypes } = changes[i];
44
+ this.data[table] = setAtPath(this.data[table], path as string[], pathTypes, valueAtPath);
45
+ }
46
+ this.save(table, config);
47
+ }
48
+ public setMetadata(table: string, metadata: PersistMetadata, config: PersistOptions) {
49
+ return this.setValue(table + MetadataSuffix, metadata, config);
50
+ }
51
+ public deleteTable(table: string, config: PersistOptions): void {
52
+ const storage = this.getStorage(config);
53
+ delete this.data[table];
54
+ storage.delete(table);
55
+ }
56
+ public deleteMetadata(table: string, config: PersistOptions) {
57
+ this.deleteTable(table + MetadataSuffix, config);
58
+ }
59
+ // Private
60
+ private getStorage(config: PersistOptions): MMKV {
61
+ const { mmkv } = config;
62
+ if (mmkv) {
63
+ const key = JSON.stringify(mmkv);
64
+ let storage = this.storages.get(key);
65
+ if (!storage) {
66
+ storage = new MMKV(mmkv);
67
+ this.storages.set(key, storage);
68
+ }
69
+ return storage;
70
+ } else {
71
+ return this.storages.get(symbolDefault)!;
72
+ }
73
+ }
74
+ private async setValue(table: string, value: any, config: PersistOptions) {
75
+ this.data[table] = value;
76
+ this.save(table, config);
77
+ }
78
+ private save(table: string, config: PersistOptions) {
79
+ const storage = this.getStorage(config);
80
+ const v = this.data[table];
81
+ if (v !== undefined) {
82
+ try {
83
+ storage.set(table, safeStringify(v));
84
+ } catch (err) {
85
+ console.error(err);
86
+ }
87
+ } else {
88
+ storage.delete(table);
89
+ }
90
+ }
91
+ }