@legendapp/state 2.2.0-next.7 → 2.2.0-next.71

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