mvc-kit 2.12.4 → 2.13.0

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 (186) hide show
  1. package/agent-config/bin/postinstall.mjs +4 -3
  2. package/agent-config/bin/setup.mjs +5 -1
  3. package/agent-config/claude-code/agents/mvc-kit-architect.md +11 -8
  4. package/agent-config/claude-code/skills/guide/SKILL.md +20 -7
  5. package/agent-config/claude-code/skills/guide/patterns.md +12 -0
  6. package/agent-config/claude-code/skills/guide/recipes.md +510 -0
  7. package/agent-config/claude-code/skills/guide/testing.md +297 -0
  8. package/agent-config/claude-code/skills/review/SKILL.md +3 -13
  9. package/agent-config/claude-code/skills/review/checklist.md +30 -5
  10. package/agent-config/claude-code/skills/scaffold/SKILL.md +4 -13
  11. package/agent-config/lib/install-claude.mjs +84 -25
  12. package/dist/Channel.cjs +276 -300
  13. package/dist/Channel.cjs.map +1 -1
  14. package/dist/Channel.js +275 -299
  15. package/dist/Channel.js.map +1 -1
  16. package/dist/Collection.cjs +424 -504
  17. package/dist/Collection.cjs.map +1 -1
  18. package/dist/Collection.js +423 -503
  19. package/dist/Collection.js.map +1 -1
  20. package/dist/Controller.cjs +70 -67
  21. package/dist/Controller.cjs.map +1 -1
  22. package/dist/Controller.js +69 -66
  23. package/dist/Controller.js.map +1 -1
  24. package/dist/EventBus.cjs +77 -88
  25. package/dist/EventBus.cjs.map +1 -1
  26. package/dist/EventBus.js +76 -87
  27. package/dist/EventBus.js.map +1 -1
  28. package/dist/Feed.cjs +81 -77
  29. package/dist/Feed.cjs.map +1 -1
  30. package/dist/Feed.js +80 -76
  31. package/dist/Feed.js.map +1 -1
  32. package/dist/Model.cjs +181 -207
  33. package/dist/Model.cjs.map +1 -1
  34. package/dist/Model.js +179 -205
  35. package/dist/Model.js.map +1 -1
  36. package/dist/Pagination.cjs +75 -73
  37. package/dist/Pagination.cjs.map +1 -1
  38. package/dist/Pagination.js +74 -72
  39. package/dist/Pagination.js.map +1 -1
  40. package/dist/Pending.cjs +255 -287
  41. package/dist/Pending.cjs.map +1 -1
  42. package/dist/Pending.js +253 -285
  43. package/dist/Pending.js.map +1 -1
  44. package/dist/PersistentCollection.cjs +242 -285
  45. package/dist/PersistentCollection.cjs.map +1 -1
  46. package/dist/PersistentCollection.js +241 -284
  47. package/dist/PersistentCollection.js.map +1 -1
  48. package/dist/Resource.cjs +166 -174
  49. package/dist/Resource.cjs.map +1 -1
  50. package/dist/Resource.js +164 -172
  51. package/dist/Resource.js.map +1 -1
  52. package/dist/Selection.cjs +84 -94
  53. package/dist/Selection.cjs.map +1 -1
  54. package/dist/Selection.js +83 -93
  55. package/dist/Selection.js.map +1 -1
  56. package/dist/Service.cjs +54 -55
  57. package/dist/Service.cjs.map +1 -1
  58. package/dist/Service.js +53 -54
  59. package/dist/Service.js.map +1 -1
  60. package/dist/Sorting.cjs +102 -101
  61. package/dist/Sorting.cjs.map +1 -1
  62. package/dist/Sorting.js +102 -101
  63. package/dist/Sorting.js.map +1 -1
  64. package/dist/Trackable.cjs +112 -80
  65. package/dist/Trackable.cjs.map +1 -1
  66. package/dist/Trackable.js +111 -79
  67. package/dist/Trackable.js.map +1 -1
  68. package/dist/ViewModel.cjs +528 -576
  69. package/dist/ViewModel.cjs.map +1 -1
  70. package/dist/ViewModel.js +525 -573
  71. package/dist/ViewModel.js.map +1 -1
  72. package/dist/bindPublicMethods.cjs +43 -24
  73. package/dist/bindPublicMethods.cjs.map +1 -1
  74. package/dist/bindPublicMethods.js +43 -24
  75. package/dist/bindPublicMethods.js.map +1 -1
  76. package/dist/errors.cjs +67 -68
  77. package/dist/errors.cjs.map +1 -1
  78. package/dist/errors.js +68 -71
  79. package/dist/errors.js.map +1 -1
  80. package/dist/mvc-kit.cjs +44 -46
  81. package/dist/mvc-kit.js +5 -32
  82. package/dist/produceDraft.cjs +105 -95
  83. package/dist/produceDraft.cjs.map +1 -1
  84. package/dist/produceDraft.js +106 -97
  85. package/dist/produceDraft.js.map +1 -1
  86. package/dist/react/components/CardList.cjs +30 -40
  87. package/dist/react/components/CardList.cjs.map +1 -1
  88. package/dist/react/components/CardList.js +31 -41
  89. package/dist/react/components/CardList.js.map +1 -1
  90. package/dist/react/components/DataTable.cjs +146 -169
  91. package/dist/react/components/DataTable.cjs.map +1 -1
  92. package/dist/react/components/DataTable.js +147 -170
  93. package/dist/react/components/DataTable.js.map +1 -1
  94. package/dist/react/components/InfiniteScroll.cjs +51 -42
  95. package/dist/react/components/InfiniteScroll.cjs.map +1 -1
  96. package/dist/react/components/InfiniteScroll.js +52 -43
  97. package/dist/react/components/InfiniteScroll.js.map +1 -1
  98. package/dist/react/components/types.cjs +10 -6
  99. package/dist/react/components/types.cjs.map +1 -1
  100. package/dist/react/components/types.js +11 -9
  101. package/dist/react/components/types.js.map +1 -1
  102. package/dist/react/guards.cjs +10 -6
  103. package/dist/react/guards.cjs.map +1 -1
  104. package/dist/react/guards.js +11 -9
  105. package/dist/react/guards.js.map +1 -1
  106. package/dist/react/provider.cjs +23 -20
  107. package/dist/react/provider.cjs.map +1 -1
  108. package/dist/react/provider.js +23 -21
  109. package/dist/react/provider.js.map +1 -1
  110. package/dist/react/use-event-bus.cjs +24 -20
  111. package/dist/react/use-event-bus.cjs.map +1 -1
  112. package/dist/react/use-event-bus.js +24 -21
  113. package/dist/react/use-event-bus.js.map +1 -1
  114. package/dist/react/use-instance.cjs +43 -36
  115. package/dist/react/use-instance.cjs.map +1 -1
  116. package/dist/react/use-instance.js +43 -36
  117. package/dist/react/use-instance.js.map +1 -1
  118. package/dist/react/use-local.cjs +48 -64
  119. package/dist/react/use-local.cjs.map +1 -1
  120. package/dist/react/use-local.js +47 -63
  121. package/dist/react/use-local.js.map +1 -1
  122. package/dist/react/use-model.cjs +84 -98
  123. package/dist/react/use-model.cjs.map +1 -1
  124. package/dist/react/use-model.js +84 -100
  125. package/dist/react/use-model.js.map +1 -1
  126. package/dist/react/use-singleton.cjs +19 -23
  127. package/dist/react/use-singleton.cjs.map +1 -1
  128. package/dist/react/use-singleton.js +16 -20
  129. package/dist/react/use-singleton.js.map +1 -1
  130. package/dist/react/use-subscribe-only.cjs +28 -22
  131. package/dist/react/use-subscribe-only.cjs.map +1 -1
  132. package/dist/react/use-subscribe-only.js +28 -22
  133. package/dist/react/use-subscribe-only.js.map +1 -1
  134. package/dist/react/use-teardown.cjs +20 -19
  135. package/dist/react/use-teardown.cjs.map +1 -1
  136. package/dist/react/use-teardown.js +20 -19
  137. package/dist/react/use-teardown.js.map +1 -1
  138. package/dist/react-native/NativeCollection.cjs +98 -78
  139. package/dist/react-native/NativeCollection.cjs.map +1 -1
  140. package/dist/react-native/NativeCollection.js +97 -77
  141. package/dist/react-native/NativeCollection.js.map +1 -1
  142. package/dist/react-native.cjs +2 -4
  143. package/dist/react-native.js +1 -4
  144. package/dist/react.cjs +24 -26
  145. package/dist/react.js +1 -17
  146. package/dist/singleton.cjs +28 -22
  147. package/dist/singleton.cjs.map +1 -1
  148. package/dist/singleton.js +29 -26
  149. package/dist/singleton.js.map +1 -1
  150. package/dist/walkPrototypeChain.cjs +20 -12
  151. package/dist/walkPrototypeChain.cjs.map +1 -1
  152. package/dist/walkPrototypeChain.js +21 -13
  153. package/dist/walkPrototypeChain.js.map +1 -1
  154. package/dist/web/IndexedDBCollection.cjs +53 -36
  155. package/dist/web/IndexedDBCollection.cjs.map +1 -1
  156. package/dist/web/IndexedDBCollection.js +52 -35
  157. package/dist/web/IndexedDBCollection.js.map +1 -1
  158. package/dist/web/WebStorageCollection.cjs +82 -84
  159. package/dist/web/WebStorageCollection.cjs.map +1 -1
  160. package/dist/web/WebStorageCollection.js +81 -83
  161. package/dist/web/WebStorageCollection.js.map +1 -1
  162. package/dist/web/idb.cjs +107 -99
  163. package/dist/web/idb.cjs.map +1 -1
  164. package/dist/web/idb.js +108 -105
  165. package/dist/web/idb.js.map +1 -1
  166. package/dist/web.cjs +4 -6
  167. package/dist/web.js +1 -5
  168. package/dist/wrapAsyncMethods.cjs +141 -168
  169. package/dist/wrapAsyncMethods.cjs.map +1 -1
  170. package/dist/wrapAsyncMethods.js +141 -168
  171. package/dist/wrapAsyncMethods.js.map +1 -1
  172. package/package.json +8 -8
  173. package/src/Pending.test.ts +1 -2
  174. package/src/Sorting.test.ts +1 -1
  175. package/src/produceDraft.test.ts +3 -3
  176. package/src/react/components/CardList.test.tsx +1 -1
  177. package/src/react/components/DataTable.test.tsx +1 -1
  178. package/src/react/components/InfiniteScroll.test.tsx +5 -5
  179. package/dist/mvc-kit.cjs.map +0 -1
  180. package/dist/mvc-kit.js.map +0 -1
  181. package/dist/react-native.cjs.map +0 -1
  182. package/dist/react-native.js.map +0 -1
  183. package/dist/react.cjs.map +0 -1
  184. package/dist/react.js.map +0 -1
  185. package/dist/web.cjs.map +0 -1
  186. package/dist/web.js.map +0 -1
@@ -1,286 +1,243 @@
1
1
  import { Collection } from "./Collection.js";
2
- const __DEV__ = typeof __MVC_KIT_DEV__ !== "undefined" && __MVC_KIT_DEV__;
3
- const _registeredKeys = __DEV__ ? /* @__PURE__ */ new Map() : null;
4
- class PersistentCollection extends Collection {
5
- /** Debounce delay in ms for storage writes. 0 = immediate. */
6
- static WRITE_DELAY = 0;
7
- // ── Serialization hooks ──
8
- /** Serialize items to a string. Used by string-based adapters (WebStorage, NativeCollection). */
9
- serialize(items) {
10
- return JSON.stringify(items);
11
- }
12
- /** Deserialize a string back to items. Used by string-based adapters. */
13
- deserialize(raw) {
14
- return JSON.parse(raw);
15
- }
16
- // ── Internal state ──
17
- _hydrated = false;
18
- _hydrating = false;
19
- // Suppresses the self-subscriber during reset/clear overrides,
20
- // which queue deltas manually instead of relying on diff.
21
- _suppressSubscriber = false;
22
- _persistenceReady = false;
23
- _preHydrationWarned = false;
24
- _pendingWrites = /* @__PURE__ */ new Map();
25
- _pendingRemoves = /* @__PURE__ */ new Set();
26
- _diffMap = /* @__PURE__ */ new Map();
27
- _pendingClear = false;
28
- _flushTimer = null;
29
- constructor(initialItems = []) {
30
- super(initialItems);
31
- const unsub = this.subscribe((current, prev) => {
32
- if (this._hydrating || this._suppressSubscriber) return;
33
- this._ensurePersistenceReady();
34
- this._diffAndQueue(current, prev);
35
- this._scheduleSave();
36
- });
37
- this.addCleanup(unsub);
38
- }
39
- /**
40
- * DEV check for duplicate storageKey. Called lazily since storageKey is an abstract
41
- * field that isn't available during the parent constructor chain.
42
- */
43
- _ensurePersistenceReady() {
44
- if (this._persistenceReady) return;
45
- this._persistenceReady = true;
46
- if (__DEV__ && _registeredKeys) {
47
- const className = this.constructor.name;
48
- const existing = _registeredKeys.get(this.storageKey);
49
- if (existing && existing !== className) {
50
- console.warn(
51
- `[mvc-kit] Duplicate storageKey "${this.storageKey}" used by "${className}" and "${existing}". Each PersistentCollection should have a unique storageKey.`
52
- );
53
- }
54
- _registeredKeys.set(this.storageKey, className);
55
- }
56
- }
57
- // ── Public API ──
58
- /** Whether storage data has been loaded. */
59
- get hydrated() {
60
- return this._hydrated;
61
- }
62
- /**
63
- * Load data from storage into the collection. Idempotent — subsequent calls return current items.
64
- * Returns the items after hydration.
65
- */
66
- async hydrate() {
67
- if (this._hydrated) return this.items;
68
- this._ensurePersistenceReady();
69
- this._hydrating = true;
70
- try {
71
- const stored = await this.persistGetAll();
72
- if (stored.length > 0) {
73
- super.reset(stored);
74
- }
75
- this._hydrated = true;
76
- return this.items;
77
- } catch (err) {
78
- this._handlePersistError(err);
79
- this._hydrated = true;
80
- return this.items;
81
- } finally {
82
- this._hydrating = false;
83
- }
84
- }
85
- /**
86
- * Synchronous hydration for sync adapters (e.g., WebStorage).
87
- * Call from the **leaf class** constructor (after field initializers have run).
88
- */
89
- _hydrateSync() {
90
- if (this._hydrated) return;
91
- this._ensurePersistenceReady();
92
- this._hydrating = true;
93
- try {
94
- const stored = this.persistGetAll();
95
- if (stored instanceof Promise) {
96
- throw new Error("[mvc-kit] _hydrateSync called with async persistGetAll");
97
- }
98
- if (stored.length > 0) {
99
- super.reset(stored);
100
- }
101
- this._hydrated = true;
102
- } catch (err) {
103
- this._handlePersistError(err);
104
- this._hydrated = true;
105
- } finally {
106
- this._hydrating = false;
107
- }
108
- }
109
- /**
110
- * Clear all data from storage AND from the in-memory collection.
111
- */
112
- clearStorage() {
113
- this._ensurePersistenceReady();
114
- this._pendingWrites.clear();
115
- this._pendingRemoves.clear();
116
- this._pendingClear = false;
117
- this._cancelSave();
118
- if (this.length > 0) {
119
- super.clear();
120
- }
121
- try {
122
- const result = this.persistClear();
123
- if (result instanceof Promise) {
124
- return result.catch((err) => this._handlePersistError(err));
125
- }
126
- } catch (err) {
127
- this._handlePersistError(err);
128
- }
129
- }
130
- // ── Overrides for clear/reset tracking ──
131
- reset(items) {
132
- this._pendingClear = true;
133
- this._pendingWrites.clear();
134
- this._pendingRemoves.clear();
135
- for (const item of items) {
136
- this._pendingWrites.set(item.id, item);
137
- }
138
- this._suppressSubscriber = true;
139
- try {
140
- super.reset(items);
141
- } finally {
142
- this._suppressSubscriber = false;
143
- }
144
- this._scheduleSave();
145
- }
146
- clear() {
147
- this._pendingClear = true;
148
- this._pendingWrites.clear();
149
- this._pendingRemoves.clear();
150
- this._suppressSubscriber = true;
151
- try {
152
- super.clear();
153
- } finally {
154
- this._suppressSubscriber = false;
155
- }
156
- this._scheduleSave();
157
- }
158
- // ── Override items getter for DEV pre-hydration warning ──
159
- get items() {
160
- if (__DEV__ && !this._hydrated && !this._hydrating && !this._preHydrationWarned) {
161
- this._preHydrationWarned = true;
162
- console.warn(
163
- `[mvc-kit] Accessing items on "${this.constructor.name}" before hydrate() has been called. Data may be incomplete. Call hydrate() first.`
164
- );
165
- }
166
- return super.items;
167
- }
168
- get state() {
169
- return this.items;
170
- }
171
- // ── Dispose ──
172
- dispose() {
173
- if (this.disposed) return;
174
- this._cancelSave();
175
- if (this._hasPending()) {
176
- try {
177
- const result = this._flush();
178
- if (result instanceof Promise) {
179
- result.catch((err) => this._handlePersistError(err));
180
- }
181
- } catch (err) {
182
- this._handlePersistError(err);
183
- }
184
- }
185
- if (__DEV__ && _registeredKeys && this._persistenceReady) {
186
- _registeredKeys.delete(this.storageKey);
187
- }
188
- super.dispose();
189
- }
190
- // ── Private: delta tracking ──
191
- _diffAndQueue(current, prev) {
192
- const prevMap = this._diffMap;
193
- prevMap.clear();
194
- for (const item of prev) {
195
- prevMap.set(item.id, item);
196
- }
197
- for (const item of current) {
198
- const prevItem = prevMap.get(item.id);
199
- if (!prevItem || prevItem !== item) {
200
- this._pendingWrites.set(item.id, item);
201
- this._pendingRemoves.delete(item.id);
202
- }
203
- prevMap.delete(item.id);
204
- }
205
- for (const [id] of prevMap) {
206
- this._pendingRemoves.add(id);
207
- this._pendingWrites.delete(id);
208
- }
209
- prevMap.clear();
210
- }
211
- _hasPending() {
212
- return this._pendingClear || this._pendingWrites.size > 0 || this._pendingRemoves.size > 0;
213
- }
214
- // ── Private: debounce + flush ──
215
- _scheduleSave() {
216
- this._cancelSave();
217
- const delay = this.constructor.WRITE_DELAY;
218
- if (delay <= 0) {
219
- this._doFlush();
220
- return;
221
- }
222
- this._flushTimer = setTimeout(() => this._doFlush(), delay);
223
- }
224
- _cancelSave() {
225
- if (this._flushTimer !== null) {
226
- clearTimeout(this._flushTimer);
227
- this._flushTimer = null;
228
- }
229
- }
230
- _doFlush() {
231
- if (!this._hasPending()) return;
232
- try {
233
- const result = this._flush();
234
- if (result instanceof Promise) {
235
- result.catch((err) => this._handlePersistError(err));
236
- }
237
- } catch (err) {
238
- this._handlePersistError(err);
239
- }
240
- }
241
- _flush() {
242
- const doClear = this._pendingClear;
243
- const writes = this._pendingWrites.size > 0 ? [...this._pendingWrites.values()] : null;
244
- const removes = this._pendingRemoves.size > 0 ? [...this._pendingRemoves] : null;
245
- this._pendingClear = false;
246
- this._pendingWrites.clear();
247
- this._pendingRemoves.clear();
248
- if (doClear) {
249
- const clearResult = this.persistClear();
250
- if (clearResult instanceof Promise) {
251
- return clearResult.then(() => {
252
- if (writes) return this.persistSet(writes);
253
- });
254
- }
255
- if (writes) {
256
- return this.persistSet(writes);
257
- }
258
- return;
259
- }
260
- if (removes) {
261
- const removeResult = this.persistRemove(removes);
262
- if (removeResult instanceof Promise) {
263
- return removeResult.then(() => {
264
- if (writes) return this.persistSet(writes);
265
- });
266
- }
267
- }
268
- if (writes) {
269
- return this.persistSet(writes);
270
- }
271
- }
272
- // ── Private: error handling ──
273
- _handlePersistError(err) {
274
- if (this.onPersistError) {
275
- this.onPersistError(err);
276
- return;
277
- }
278
- if (__DEV__) {
279
- console.warn("[mvc-kit] Storage error:", err);
280
- }
281
- }
282
- }
283
- export {
284
- PersistentCollection
2
+ //#region src/PersistentCollection.ts
3
+ var __DEV__ = typeof __MVC_KIT_DEV__ !== "undefined" && __MVC_KIT_DEV__;
4
+ var _registeredKeys = __DEV__ ? /* @__PURE__ */ new Map() : null;
5
+ /**
6
+ * Abstract base for Collections that persist to external storage.
7
+ * Tracks deltas per mutation and flushes via debounced writes.
8
+ * Subclasses implement the storage-specific `persist*` methods.
9
+ */
10
+ var PersistentCollection = class extends Collection {
11
+ /** Debounce delay in ms for storage writes. 0 = immediate. */
12
+ static WRITE_DELAY = 0;
13
+ /** Serialize items to a string. Used by string-based adapters (WebStorage, NativeCollection). */
14
+ serialize(items) {
15
+ return JSON.stringify(items);
16
+ }
17
+ /** Deserialize a string back to items. Used by string-based adapters. */
18
+ deserialize(raw) {
19
+ return JSON.parse(raw);
20
+ }
21
+ _hydrated = false;
22
+ _hydrating = false;
23
+ _suppressSubscriber = false;
24
+ _persistenceReady = false;
25
+ _preHydrationWarned = false;
26
+ _pendingWrites = /* @__PURE__ */ new Map();
27
+ _pendingRemoves = /* @__PURE__ */ new Set();
28
+ _diffMap = /* @__PURE__ */ new Map();
29
+ _pendingClear = false;
30
+ _flushTimer = null;
31
+ constructor(initialItems = []) {
32
+ super(initialItems);
33
+ const unsub = this.subscribe((current, prev) => {
34
+ if (this._hydrating || this._suppressSubscriber) return;
35
+ this._ensurePersistenceReady();
36
+ this._diffAndQueue(current, prev);
37
+ this._scheduleSave();
38
+ });
39
+ this.addCleanup(unsub);
40
+ }
41
+ /**
42
+ * DEV check for duplicate storageKey. Called lazily since storageKey is an abstract
43
+ * field that isn't available during the parent constructor chain.
44
+ */
45
+ _ensurePersistenceReady() {
46
+ if (this._persistenceReady) return;
47
+ this._persistenceReady = true;
48
+ if (__DEV__ && _registeredKeys) {
49
+ const className = this.constructor.name;
50
+ const existing = _registeredKeys.get(this.storageKey);
51
+ if (existing && existing !== className) console.warn(`[mvc-kit] Duplicate storageKey "${this.storageKey}" used by "${className}" and "${existing}". Each PersistentCollection should have a unique storageKey.`);
52
+ _registeredKeys.set(this.storageKey, className);
53
+ }
54
+ }
55
+ /** Whether storage data has been loaded. */
56
+ get hydrated() {
57
+ return this._hydrated;
58
+ }
59
+ /**
60
+ * Load data from storage into the collection. Idempotent — subsequent calls return current items.
61
+ * Returns the items after hydration.
62
+ */
63
+ async hydrate() {
64
+ if (this._hydrated) return this.items;
65
+ this._ensurePersistenceReady();
66
+ this._hydrating = true;
67
+ try {
68
+ const stored = await this.persistGetAll();
69
+ if (stored.length > 0) super.reset(stored);
70
+ this._hydrated = true;
71
+ return this.items;
72
+ } catch (err) {
73
+ this._handlePersistError(err);
74
+ this._hydrated = true;
75
+ return this.items;
76
+ } finally {
77
+ this._hydrating = false;
78
+ }
79
+ }
80
+ /**
81
+ * Synchronous hydration for sync adapters (e.g., WebStorage).
82
+ * Call from the **leaf class** constructor (after field initializers have run).
83
+ */
84
+ _hydrateSync() {
85
+ if (this._hydrated) return;
86
+ this._ensurePersistenceReady();
87
+ this._hydrating = true;
88
+ try {
89
+ const stored = this.persistGetAll();
90
+ if (stored instanceof Promise) throw new Error("[mvc-kit] _hydrateSync called with async persistGetAll");
91
+ if (stored.length > 0) super.reset(stored);
92
+ this._hydrated = true;
93
+ } catch (err) {
94
+ this._handlePersistError(err);
95
+ this._hydrated = true;
96
+ } finally {
97
+ this._hydrating = false;
98
+ }
99
+ }
100
+ /**
101
+ * Clear all data from storage AND from the in-memory collection.
102
+ */
103
+ clearStorage() {
104
+ this._ensurePersistenceReady();
105
+ this._pendingWrites.clear();
106
+ this._pendingRemoves.clear();
107
+ this._pendingClear = false;
108
+ this._cancelSave();
109
+ if (this.length > 0) super.clear();
110
+ try {
111
+ const result = this.persistClear();
112
+ if (result instanceof Promise) return result.catch((err) => this._handlePersistError(err));
113
+ } catch (err) {
114
+ this._handlePersistError(err);
115
+ }
116
+ }
117
+ reset(items) {
118
+ this._pendingClear = true;
119
+ this._pendingWrites.clear();
120
+ this._pendingRemoves.clear();
121
+ for (const item of items) this._pendingWrites.set(item.id, item);
122
+ this._suppressSubscriber = true;
123
+ try {
124
+ super.reset(items);
125
+ } finally {
126
+ this._suppressSubscriber = false;
127
+ }
128
+ this._scheduleSave();
129
+ }
130
+ clear() {
131
+ this._pendingClear = true;
132
+ this._pendingWrites.clear();
133
+ this._pendingRemoves.clear();
134
+ this._suppressSubscriber = true;
135
+ try {
136
+ super.clear();
137
+ } finally {
138
+ this._suppressSubscriber = false;
139
+ }
140
+ this._scheduleSave();
141
+ }
142
+ get items() {
143
+ if (__DEV__ && !this._hydrated && !this._hydrating && !this._preHydrationWarned) {
144
+ this._preHydrationWarned = true;
145
+ console.warn(`[mvc-kit] Accessing items on "${this.constructor.name}" before hydrate() has been called. Data may be incomplete. Call hydrate() first.`);
146
+ }
147
+ return super.items;
148
+ }
149
+ get state() {
150
+ return this.items;
151
+ }
152
+ dispose() {
153
+ if (this.disposed) return;
154
+ this._cancelSave();
155
+ if (this._hasPending()) try {
156
+ const result = this._flush();
157
+ if (result instanceof Promise) result.catch((err) => this._handlePersistError(err));
158
+ } catch (err) {
159
+ this._handlePersistError(err);
160
+ }
161
+ if (__DEV__ && _registeredKeys && this._persistenceReady) _registeredKeys.delete(this.storageKey);
162
+ super.dispose();
163
+ }
164
+ _diffAndQueue(current, prev) {
165
+ const prevMap = this._diffMap;
166
+ prevMap.clear();
167
+ for (const item of prev) prevMap.set(item.id, item);
168
+ for (const item of current) {
169
+ const prevItem = prevMap.get(item.id);
170
+ if (!prevItem || prevItem !== item) {
171
+ this._pendingWrites.set(item.id, item);
172
+ this._pendingRemoves.delete(item.id);
173
+ }
174
+ prevMap.delete(item.id);
175
+ }
176
+ for (const [id] of prevMap) {
177
+ this._pendingRemoves.add(id);
178
+ this._pendingWrites.delete(id);
179
+ }
180
+ prevMap.clear();
181
+ }
182
+ _hasPending() {
183
+ return this._pendingClear || this._pendingWrites.size > 0 || this._pendingRemoves.size > 0;
184
+ }
185
+ _scheduleSave() {
186
+ this._cancelSave();
187
+ const delay = this.constructor.WRITE_DELAY;
188
+ if (delay <= 0) {
189
+ this._doFlush();
190
+ return;
191
+ }
192
+ this._flushTimer = setTimeout(() => this._doFlush(), delay);
193
+ }
194
+ _cancelSave() {
195
+ if (this._flushTimer !== null) {
196
+ clearTimeout(this._flushTimer);
197
+ this._flushTimer = null;
198
+ }
199
+ }
200
+ _doFlush() {
201
+ if (!this._hasPending()) return;
202
+ try {
203
+ const result = this._flush();
204
+ if (result instanceof Promise) result.catch((err) => this._handlePersistError(err));
205
+ } catch (err) {
206
+ this._handlePersistError(err);
207
+ }
208
+ }
209
+ _flush() {
210
+ const doClear = this._pendingClear;
211
+ const writes = this._pendingWrites.size > 0 ? [...this._pendingWrites.values()] : null;
212
+ const removes = this._pendingRemoves.size > 0 ? [...this._pendingRemoves] : null;
213
+ this._pendingClear = false;
214
+ this._pendingWrites.clear();
215
+ this._pendingRemoves.clear();
216
+ if (doClear) {
217
+ const clearResult = this.persistClear();
218
+ if (clearResult instanceof Promise) return clearResult.then(() => {
219
+ if (writes) return this.persistSet(writes);
220
+ });
221
+ if (writes) return this.persistSet(writes);
222
+ return;
223
+ }
224
+ if (removes) {
225
+ const removeResult = this.persistRemove(removes);
226
+ if (removeResult instanceof Promise) return removeResult.then(() => {
227
+ if (writes) return this.persistSet(writes);
228
+ });
229
+ }
230
+ if (writes) return this.persistSet(writes);
231
+ }
232
+ _handlePersistError(err) {
233
+ if (this.onPersistError) {
234
+ this.onPersistError(err);
235
+ return;
236
+ }
237
+ if (__DEV__) console.warn("[mvc-kit] Storage error:", err);
238
+ }
285
239
  };
286
- //# sourceMappingURL=PersistentCollection.js.map
240
+ //#endregion
241
+ export { PersistentCollection };
242
+
243
+ //# sourceMappingURL=PersistentCollection.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"PersistentCollection.js","sources":["../src/PersistentCollection.ts"],"sourcesContent":["import { Collection } from './Collection';\n\nconst __DEV__ = typeof __MVC_KIT_DEV__ !== 'undefined' && __MVC_KIT_DEV__;\n\n// Track storageKey uniqueness in DEV\nconst _registeredKeys: Map<string, string> | null = __DEV__ ? new Map() : null;\n\n/**\n * Abstract base for Collections that persist to external storage.\n * Tracks deltas per mutation and flushes via debounced writes.\n * Subclasses implement the storage-specific `persist*` methods.\n */\nexport abstract class PersistentCollection<\n T extends { id: string | number },\n> extends Collection<T> {\n /** Debounce delay in ms for storage writes. 0 = immediate. */\n static WRITE_DELAY = 0;\n\n /** Unique key identifying this collection in storage. */\n protected abstract readonly storageKey: string;\n\n // ── Abstract persistence methods ──\n\n /** Retrieve a single item by id from storage. @protected */\n protected abstract persistGet(id: T['id']): T | null | Promise<T | null>;\n /** Retrieve all items from storage. @protected */\n protected abstract persistGetAll(): T[] | Promise<T[]>;\n /** Upsert semantics — insert or replace the given items in storage. @protected */\n protected abstract persistSet(items: T[]): void | Promise<void>;\n /** Remove items by their ids from storage. @protected */\n protected abstract persistRemove(ids: T['id'][]): void | Promise<void>;\n /** Remove all items from storage. @protected */\n protected abstract persistClear(): void | Promise<void>;\n\n // ── Serialization hooks ──\n\n /** Serialize items to a string. Used by string-based adapters (WebStorage, NativeCollection). */\n protected serialize(items: T[]): string {\n return JSON.stringify(items);\n }\n\n /** Deserialize a string back to items. Used by string-based adapters. */\n protected deserialize(raw: string): T[] {\n return JSON.parse(raw);\n }\n\n // ── Error hook ──\n\n /** Called when a storage operation fails. Override for custom error handling. */\n protected onPersistError?(error: unknown): void;\n\n // ── Internal state ──\n\n private _hydrated = false;\n private _hydrating = false;\n // Suppresses the self-subscriber during reset/clear overrides,\n // which queue deltas manually instead of relying on diff.\n private _suppressSubscriber = false;\n private _persistenceReady = false;\n private _preHydrationWarned = false;\n private _pendingWrites = new Map<T['id'], T>();\n private _pendingRemoves = new Set<T['id']>();\n private _diffMap = new Map<T['id'], T>();\n private _pendingClear = false;\n private _flushTimer: ReturnType<typeof setTimeout> | null = null;\n\n constructor(initialItems: T[] = []) {\n super(initialItems);\n\n // Self-subscribe to detect mutations via diff.\n // storageKey may not be available yet (class field initializers run after super()),\n // but that's fine — the subscriber only fires on mutations, not during construction.\n const unsub = this.subscribe((current, prev) => {\n if (this._hydrating || this._suppressSubscriber) return;\n this._ensurePersistenceReady();\n this._diffAndQueue(current, prev);\n this._scheduleSave();\n });\n this.addCleanup(unsub);\n }\n\n /**\n * DEV check for duplicate storageKey. Called lazily since storageKey is an abstract\n * field that isn't available during the parent constructor chain.\n */\n private _ensurePersistenceReady(): void {\n if (this._persistenceReady) return;\n this._persistenceReady = true;\n\n if (__DEV__ && _registeredKeys) {\n const className = this.constructor.name;\n const existing = _registeredKeys.get(this.storageKey);\n if (existing && existing !== className) {\n console.warn(\n `[mvc-kit] Duplicate storageKey \"${this.storageKey}\" used by \"${className}\" ` +\n `and \"${existing}\". Each PersistentCollection should have a unique storageKey.`,\n );\n }\n _registeredKeys.set(this.storageKey, className);\n }\n }\n\n // ── Public API ──\n\n /** Whether storage data has been loaded. */\n get hydrated(): boolean {\n return this._hydrated;\n }\n\n /**\n * Load data from storage into the collection. Idempotent — subsequent calls return current items.\n * Returns the items after hydration.\n */\n async hydrate(): Promise<T[]> {\n if (this._hydrated) return this.items;\n this._ensurePersistenceReady();\n\n this._hydrating = true;\n try {\n const stored = await this.persistGetAll();\n if (stored.length > 0) {\n super.reset(stored);\n }\n this._hydrated = true;\n return this.items;\n } catch (err) {\n this._handlePersistError(err);\n this._hydrated = true;\n return this.items;\n } finally {\n this._hydrating = false;\n }\n }\n\n /**\n * Synchronous hydration for sync adapters (e.g., WebStorage).\n * Call from the **leaf class** constructor (after field initializers have run).\n */\n protected _hydrateSync(): void {\n if (this._hydrated) return;\n this._ensurePersistenceReady();\n\n this._hydrating = true;\n try {\n const stored = this.persistGetAll();\n if (stored instanceof Promise) {\n throw new Error('[mvc-kit] _hydrateSync called with async persistGetAll');\n }\n if (stored.length > 0) {\n super.reset(stored);\n }\n this._hydrated = true;\n } catch (err) {\n this._handlePersistError(err);\n this._hydrated = true;\n } finally {\n this._hydrating = false;\n }\n }\n\n /**\n * Clear all data from storage AND from the in-memory collection.\n */\n clearStorage(): void | Promise<void> {\n this._ensurePersistenceReady();\n\n // Clear pending queues — we're wiping everything\n this._pendingWrites.clear();\n this._pendingRemoves.clear();\n this._pendingClear = false;\n this._cancelSave();\n\n // Clear in-memory\n if (this.length > 0) {\n super.clear();\n }\n\n try {\n const result = this.persistClear();\n if (result instanceof Promise) {\n return result.catch((err) => this._handlePersistError(err));\n }\n } catch (err) {\n this._handlePersistError(err);\n }\n }\n\n // ── Overrides for clear/reset tracking ──\n\n reset(items: T[]): void {\n this._pendingClear = true;\n this._pendingWrites.clear();\n this._pendingRemoves.clear();\n for (const item of items) {\n this._pendingWrites.set(item.id, item);\n }\n // Suppress the self-subscriber — deltas are queued manually above\n this._suppressSubscriber = true;\n try {\n super.reset(items);\n } finally {\n this._suppressSubscriber = false;\n }\n this._scheduleSave();\n }\n\n clear(): void {\n this._pendingClear = true;\n this._pendingWrites.clear();\n this._pendingRemoves.clear();\n // Suppress the self-subscriber — deltas are queued manually above\n this._suppressSubscriber = true;\n try {\n super.clear();\n } finally {\n this._suppressSubscriber = false;\n }\n this._scheduleSave();\n }\n\n // ── Override items getter for DEV pre-hydration warning ──\n\n get items(): T[] {\n if (__DEV__ && !this._hydrated && !this._hydrating && !this._preHydrationWarned) {\n this._preHydrationWarned = true;\n console.warn(\n `[mvc-kit] Accessing items on \"${this.constructor.name}\" before hydrate() has been called. ` +\n `Data may be incomplete. Call hydrate() first.`,\n );\n }\n return super.items;\n }\n\n get state(): T[] {\n return this.items;\n }\n\n // ── Dispose ──\n\n dispose(): void {\n if (this.disposed) return;\n\n // Flush any pending saves before disposing\n this._cancelSave();\n if (this._hasPending()) {\n try {\n const result = this._flush();\n if (result instanceof Promise) {\n result.catch((err) => this._handlePersistError(err));\n }\n } catch (err) {\n this._handlePersistError(err);\n }\n }\n\n // DEV: unregister storageKey\n if (__DEV__ && _registeredKeys && this._persistenceReady) {\n _registeredKeys.delete(this.storageKey);\n }\n\n super.dispose();\n }\n\n // ── Private: delta tracking ──\n\n private _diffAndQueue(current: readonly T[], prev: readonly T[]): void {\n const prevMap = this._diffMap;\n prevMap.clear();\n for (const item of prev) {\n prevMap.set(item.id, item);\n }\n\n // Added or updated: in current but different reference in prev\n for (const item of current) {\n const prevItem = prevMap.get(item.id);\n if (!prevItem || prevItem !== item) {\n this._pendingWrites.set(item.id, item);\n this._pendingRemoves.delete(item.id);\n }\n prevMap.delete(item.id); // consume matched items\n }\n\n // Remaining in prevMap = removed (in prev but not in current)\n for (const [id] of prevMap) {\n this._pendingRemoves.add(id);\n this._pendingWrites.delete(id);\n }\n\n prevMap.clear();\n }\n\n private _hasPending(): boolean {\n return this._pendingClear || this._pendingWrites.size > 0 || this._pendingRemoves.size > 0;\n }\n\n // ── Private: debounce + flush ──\n\n private _scheduleSave(): void {\n this._cancelSave();\n const delay = (this.constructor as typeof PersistentCollection).WRITE_DELAY;\n if (delay <= 0) {\n this._doFlush();\n return;\n }\n this._flushTimer = setTimeout(() => this._doFlush(), delay);\n }\n\n private _cancelSave(): void {\n if (this._flushTimer !== null) {\n clearTimeout(this._flushTimer);\n this._flushTimer = null;\n }\n }\n\n private _doFlush(): void {\n if (!this._hasPending()) return;\n try {\n const result = this._flush();\n if (result instanceof Promise) {\n result.catch((err) => this._handlePersistError(err));\n }\n } catch (err) {\n this._handlePersistError(err);\n }\n }\n\n private _flush(): void | Promise<void> {\n const doClear = this._pendingClear;\n const writes = this._pendingWrites.size > 0 ? [...this._pendingWrites.values()] : null;\n const removes = this._pendingRemoves.size > 0 ? [...this._pendingRemoves] : null;\n\n // Clear queues\n this._pendingClear = false;\n this._pendingWrites.clear();\n this._pendingRemoves.clear();\n\n if (doClear) {\n const clearResult = this.persistClear();\n if (clearResult instanceof Promise) {\n return clearResult.then(() => {\n if (writes) return this.persistSet(writes);\n });\n }\n if (writes) {\n return this.persistSet(writes);\n }\n return;\n }\n\n // Non-clear: removes then writes\n if (removes) {\n const removeResult = this.persistRemove(removes);\n if (removeResult instanceof Promise) {\n return removeResult.then(() => {\n if (writes) return this.persistSet(writes);\n });\n }\n }\n if (writes) {\n return this.persistSet(writes);\n }\n }\n\n // ── Private: error handling ──\n\n private _handlePersistError(err: unknown): void {\n if (this.onPersistError) {\n this.onPersistError(err);\n return;\n }\n if (__DEV__) {\n console.warn('[mvc-kit] Storage error:', err);\n }\n }\n}\n"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"PersistentCollection.js","names":[],"sources":["../src/PersistentCollection.ts"],"sourcesContent":["import { Collection } from './Collection';\n\nconst __DEV__ = typeof __MVC_KIT_DEV__ !== 'undefined' && __MVC_KIT_DEV__;\n\n// Track storageKey uniqueness in DEV\nconst _registeredKeys: Map<string, string> | null = __DEV__ ? new Map() : null;\n\n/**\n * Abstract base for Collections that persist to external storage.\n * Tracks deltas per mutation and flushes via debounced writes.\n * Subclasses implement the storage-specific `persist*` methods.\n */\nexport abstract class PersistentCollection<\n T extends { id: string | number },\n> extends Collection<T> {\n /** Debounce delay in ms for storage writes. 0 = immediate. */\n static WRITE_DELAY = 0;\n\n /** Unique key identifying this collection in storage. */\n protected abstract readonly storageKey: string;\n\n // ── Abstract persistence methods ──\n\n /** Retrieve a single item by id from storage. @protected */\n protected abstract persistGet(id: T['id']): T | null | Promise<T | null>;\n /** Retrieve all items from storage. @protected */\n protected abstract persistGetAll(): T[] | Promise<T[]>;\n /** Upsert semantics — insert or replace the given items in storage. @protected */\n protected abstract persistSet(items: T[]): void | Promise<void>;\n /** Remove items by their ids from storage. @protected */\n protected abstract persistRemove(ids: T['id'][]): void | Promise<void>;\n /** Remove all items from storage. @protected */\n protected abstract persistClear(): void | Promise<void>;\n\n // ── Serialization hooks ──\n\n /** Serialize items to a string. Used by string-based adapters (WebStorage, NativeCollection). */\n protected serialize(items: T[]): string {\n return JSON.stringify(items);\n }\n\n /** Deserialize a string back to items. Used by string-based adapters. */\n protected deserialize(raw: string): T[] {\n return JSON.parse(raw);\n }\n\n // ── Error hook ──\n\n /** Called when a storage operation fails. Override for custom error handling. */\n protected onPersistError?(error: unknown): void;\n\n // ── Internal state ──\n\n private _hydrated = false;\n private _hydrating = false;\n // Suppresses the self-subscriber during reset/clear overrides,\n // which queue deltas manually instead of relying on diff.\n private _suppressSubscriber = false;\n private _persistenceReady = false;\n private _preHydrationWarned = false;\n private _pendingWrites = new Map<T['id'], T>();\n private _pendingRemoves = new Set<T['id']>();\n private _diffMap = new Map<T['id'], T>();\n private _pendingClear = false;\n private _flushTimer: ReturnType<typeof setTimeout> | null = null;\n\n constructor(initialItems: T[] = []) {\n super(initialItems);\n\n // Self-subscribe to detect mutations via diff.\n // storageKey may not be available yet (class field initializers run after super()),\n // but that's fine — the subscriber only fires on mutations, not during construction.\n const unsub = this.subscribe((current, prev) => {\n if (this._hydrating || this._suppressSubscriber) return;\n this._ensurePersistenceReady();\n this._diffAndQueue(current, prev);\n this._scheduleSave();\n });\n this.addCleanup(unsub);\n }\n\n /**\n * DEV check for duplicate storageKey. Called lazily since storageKey is an abstract\n * field that isn't available during the parent constructor chain.\n */\n private _ensurePersistenceReady(): void {\n if (this._persistenceReady) return;\n this._persistenceReady = true;\n\n if (__DEV__ && _registeredKeys) {\n const className = this.constructor.name;\n const existing = _registeredKeys.get(this.storageKey);\n if (existing && existing !== className) {\n console.warn(\n `[mvc-kit] Duplicate storageKey \"${this.storageKey}\" used by \"${className}\" ` +\n `and \"${existing}\". Each PersistentCollection should have a unique storageKey.`,\n );\n }\n _registeredKeys.set(this.storageKey, className);\n }\n }\n\n // ── Public API ──\n\n /** Whether storage data has been loaded. */\n get hydrated(): boolean {\n return this._hydrated;\n }\n\n /**\n * Load data from storage into the collection. Idempotent — subsequent calls return current items.\n * Returns the items after hydration.\n */\n async hydrate(): Promise<T[]> {\n if (this._hydrated) return this.items;\n this._ensurePersistenceReady();\n\n this._hydrating = true;\n try {\n const stored = await this.persistGetAll();\n if (stored.length > 0) {\n super.reset(stored);\n }\n this._hydrated = true;\n return this.items;\n } catch (err) {\n this._handlePersistError(err);\n this._hydrated = true;\n return this.items;\n } finally {\n this._hydrating = false;\n }\n }\n\n /**\n * Synchronous hydration for sync adapters (e.g., WebStorage).\n * Call from the **leaf class** constructor (after field initializers have run).\n */\n protected _hydrateSync(): void {\n if (this._hydrated) return;\n this._ensurePersistenceReady();\n\n this._hydrating = true;\n try {\n const stored = this.persistGetAll();\n if (stored instanceof Promise) {\n throw new Error('[mvc-kit] _hydrateSync called with async persistGetAll');\n }\n if (stored.length > 0) {\n super.reset(stored);\n }\n this._hydrated = true;\n } catch (err) {\n this._handlePersistError(err);\n this._hydrated = true;\n } finally {\n this._hydrating = false;\n }\n }\n\n /**\n * Clear all data from storage AND from the in-memory collection.\n */\n clearStorage(): void | Promise<void> {\n this._ensurePersistenceReady();\n\n // Clear pending queues — we're wiping everything\n this._pendingWrites.clear();\n this._pendingRemoves.clear();\n this._pendingClear = false;\n this._cancelSave();\n\n // Clear in-memory\n if (this.length > 0) {\n super.clear();\n }\n\n try {\n const result = this.persistClear();\n if (result instanceof Promise) {\n return result.catch((err) => this._handlePersistError(err));\n }\n } catch (err) {\n this._handlePersistError(err);\n }\n }\n\n // ── Overrides for clear/reset tracking ──\n\n reset(items: T[]): void {\n this._pendingClear = true;\n this._pendingWrites.clear();\n this._pendingRemoves.clear();\n for (const item of items) {\n this._pendingWrites.set(item.id, item);\n }\n // Suppress the self-subscriber — deltas are queued manually above\n this._suppressSubscriber = true;\n try {\n super.reset(items);\n } finally {\n this._suppressSubscriber = false;\n }\n this._scheduleSave();\n }\n\n clear(): void {\n this._pendingClear = true;\n this._pendingWrites.clear();\n this._pendingRemoves.clear();\n // Suppress the self-subscriber — deltas are queued manually above\n this._suppressSubscriber = true;\n try {\n super.clear();\n } finally {\n this._suppressSubscriber = false;\n }\n this._scheduleSave();\n }\n\n // ── Override items getter for DEV pre-hydration warning ──\n\n get items(): T[] {\n if (__DEV__ && !this._hydrated && !this._hydrating && !this._preHydrationWarned) {\n this._preHydrationWarned = true;\n console.warn(\n `[mvc-kit] Accessing items on \"${this.constructor.name}\" before hydrate() has been called. ` +\n `Data may be incomplete. Call hydrate() first.`,\n );\n }\n return super.items;\n }\n\n get state(): T[] {\n return this.items;\n }\n\n // ── Dispose ──\n\n dispose(): void {\n if (this.disposed) return;\n\n // Flush any pending saves before disposing\n this._cancelSave();\n if (this._hasPending()) {\n try {\n const result = this._flush();\n if (result instanceof Promise) {\n result.catch((err) => this._handlePersistError(err));\n }\n } catch (err) {\n this._handlePersistError(err);\n }\n }\n\n // DEV: unregister storageKey\n if (__DEV__ && _registeredKeys && this._persistenceReady) {\n _registeredKeys.delete(this.storageKey);\n }\n\n super.dispose();\n }\n\n // ── Private: delta tracking ──\n\n private _diffAndQueue(current: readonly T[], prev: readonly T[]): void {\n const prevMap = this._diffMap;\n prevMap.clear();\n for (const item of prev) {\n prevMap.set(item.id, item);\n }\n\n // Added or updated: in current but different reference in prev\n for (const item of current) {\n const prevItem = prevMap.get(item.id);\n if (!prevItem || prevItem !== item) {\n this._pendingWrites.set(item.id, item);\n this._pendingRemoves.delete(item.id);\n }\n prevMap.delete(item.id); // consume matched items\n }\n\n // Remaining in prevMap = removed (in prev but not in current)\n for (const [id] of prevMap) {\n this._pendingRemoves.add(id);\n this._pendingWrites.delete(id);\n }\n\n prevMap.clear();\n }\n\n private _hasPending(): boolean {\n return this._pendingClear || this._pendingWrites.size > 0 || this._pendingRemoves.size > 0;\n }\n\n // ── Private: debounce + flush ──\n\n private _scheduleSave(): void {\n this._cancelSave();\n const delay = (this.constructor as typeof PersistentCollection).WRITE_DELAY;\n if (delay <= 0) {\n this._doFlush();\n return;\n }\n this._flushTimer = setTimeout(() => this._doFlush(), delay);\n }\n\n private _cancelSave(): void {\n if (this._flushTimer !== null) {\n clearTimeout(this._flushTimer);\n this._flushTimer = null;\n }\n }\n\n private _doFlush(): void {\n if (!this._hasPending()) return;\n try {\n const result = this._flush();\n if (result instanceof Promise) {\n result.catch((err) => this._handlePersistError(err));\n }\n } catch (err) {\n this._handlePersistError(err);\n }\n }\n\n private _flush(): void | Promise<void> {\n const doClear = this._pendingClear;\n const writes = this._pendingWrites.size > 0 ? [...this._pendingWrites.values()] : null;\n const removes = this._pendingRemoves.size > 0 ? [...this._pendingRemoves] : null;\n\n // Clear queues\n this._pendingClear = false;\n this._pendingWrites.clear();\n this._pendingRemoves.clear();\n\n if (doClear) {\n const clearResult = this.persistClear();\n if (clearResult instanceof Promise) {\n return clearResult.then(() => {\n if (writes) return this.persistSet(writes);\n });\n }\n if (writes) {\n return this.persistSet(writes);\n }\n return;\n }\n\n // Non-clear: removes then writes\n if (removes) {\n const removeResult = this.persistRemove(removes);\n if (removeResult instanceof Promise) {\n return removeResult.then(() => {\n if (writes) return this.persistSet(writes);\n });\n }\n }\n if (writes) {\n return this.persistSet(writes);\n }\n }\n\n // ── Private: error handling ──\n\n private _handlePersistError(err: unknown): void {\n if (this.onPersistError) {\n this.onPersistError(err);\n return;\n }\n if (__DEV__) {\n console.warn('[mvc-kit] Storage error:', err);\n }\n }\n}\n"],"mappings":";;AAEA,IAAM,UAAU,OAAO,oBAAoB,eAAe;AAG1D,IAAM,kBAA8C,0BAAU,IAAI,KAAK,GAAG;;;;;;AAO1E,IAAsB,uBAAtB,cAEU,WAAc;;CAEtB,OAAO,cAAc;;CAqBrB,UAAoB,OAAoB;AACtC,SAAO,KAAK,UAAU,MAAM;;;CAI9B,YAAsB,KAAkB;AACtC,SAAO,KAAK,MAAM,IAAI;;CAUxB,YAAoB;CACpB,aAAqB;CAGrB,sBAA8B;CAC9B,oBAA4B;CAC5B,sBAA8B;CAC9B,iCAAyB,IAAI,KAAiB;CAC9C,kCAA0B,IAAI,KAAc;CAC5C,2BAAmB,IAAI,KAAiB;CACxC,gBAAwB;CACxB,cAA4D;CAE5D,YAAY,eAAoB,EAAE,EAAE;AAClC,QAAM,aAAa;EAKnB,MAAM,QAAQ,KAAK,WAAW,SAAS,SAAS;AAC9C,OAAI,KAAK,cAAc,KAAK,oBAAqB;AACjD,QAAK,yBAAyB;AAC9B,QAAK,cAAc,SAAS,KAAK;AACjC,QAAK,eAAe;IACpB;AACF,OAAK,WAAW,MAAM;;;;;;CAOxB,0BAAwC;AACtC,MAAI,KAAK,kBAAmB;AAC5B,OAAK,oBAAoB;AAEzB,MAAI,WAAW,iBAAiB;GAC9B,MAAM,YAAY,KAAK,YAAY;GACnC,MAAM,WAAW,gBAAgB,IAAI,KAAK,WAAW;AACrD,OAAI,YAAY,aAAa,UAC3B,SAAQ,KACN,mCAAmC,KAAK,WAAW,aAAa,UAAU,SAChE,SAAS,+DACpB;AAEH,mBAAgB,IAAI,KAAK,YAAY,UAAU;;;;CAOnD,IAAI,WAAoB;AACtB,SAAO,KAAK;;;;;;CAOd,MAAM,UAAwB;AAC5B,MAAI,KAAK,UAAW,QAAO,KAAK;AAChC,OAAK,yBAAyB;AAE9B,OAAK,aAAa;AAClB,MAAI;GACF,MAAM,SAAS,MAAM,KAAK,eAAe;AACzC,OAAI,OAAO,SAAS,EAClB,OAAM,MAAM,OAAO;AAErB,QAAK,YAAY;AACjB,UAAO,KAAK;WACL,KAAK;AACZ,QAAK,oBAAoB,IAAI;AAC7B,QAAK,YAAY;AACjB,UAAO,KAAK;YACJ;AACR,QAAK,aAAa;;;;;;;CAQtB,eAA+B;AAC7B,MAAI,KAAK,UAAW;AACpB,OAAK,yBAAyB;AAE9B,OAAK,aAAa;AAClB,MAAI;GACF,MAAM,SAAS,KAAK,eAAe;AACnC,OAAI,kBAAkB,QACpB,OAAM,IAAI,MAAM,yDAAyD;AAE3E,OAAI,OAAO,SAAS,EAClB,OAAM,MAAM,OAAO;AAErB,QAAK,YAAY;WACV,KAAK;AACZ,QAAK,oBAAoB,IAAI;AAC7B,QAAK,YAAY;YACT;AACR,QAAK,aAAa;;;;;;CAOtB,eAAqC;AACnC,OAAK,yBAAyB;AAG9B,OAAK,eAAe,OAAO;AAC3B,OAAK,gBAAgB,OAAO;AAC5B,OAAK,gBAAgB;AACrB,OAAK,aAAa;AAGlB,MAAI,KAAK,SAAS,EAChB,OAAM,OAAO;AAGf,MAAI;GACF,MAAM,SAAS,KAAK,cAAc;AAClC,OAAI,kBAAkB,QACpB,QAAO,OAAO,OAAO,QAAQ,KAAK,oBAAoB,IAAI,CAAC;WAEtD,KAAK;AACZ,QAAK,oBAAoB,IAAI;;;CAMjC,MAAM,OAAkB;AACtB,OAAK,gBAAgB;AACrB,OAAK,eAAe,OAAO;AAC3B,OAAK,gBAAgB,OAAO;AAC5B,OAAK,MAAM,QAAQ,MACjB,MAAK,eAAe,IAAI,KAAK,IAAI,KAAK;AAGxC,OAAK,sBAAsB;AAC3B,MAAI;AACF,SAAM,MAAM,MAAM;YACV;AACR,QAAK,sBAAsB;;AAE7B,OAAK,eAAe;;CAGtB,QAAc;AACZ,OAAK,gBAAgB;AACrB,OAAK,eAAe,OAAO;AAC3B,OAAK,gBAAgB,OAAO;AAE5B,OAAK,sBAAsB;AAC3B,MAAI;AACF,SAAM,OAAO;YACL;AACR,QAAK,sBAAsB;;AAE7B,OAAK,eAAe;;CAKtB,IAAI,QAAa;AACf,MAAI,WAAW,CAAC,KAAK,aAAa,CAAC,KAAK,cAAc,CAAC,KAAK,qBAAqB;AAC/E,QAAK,sBAAsB;AAC3B,WAAQ,KACN,iCAAiC,KAAK,YAAY,KAAK,mFAExD;;AAEH,SAAO,MAAM;;CAGf,IAAI,QAAa;AACf,SAAO,KAAK;;CAKd,UAAgB;AACd,MAAI,KAAK,SAAU;AAGnB,OAAK,aAAa;AAClB,MAAI,KAAK,aAAa,CACpB,KAAI;GACF,MAAM,SAAS,KAAK,QAAQ;AAC5B,OAAI,kBAAkB,QACpB,QAAO,OAAO,QAAQ,KAAK,oBAAoB,IAAI,CAAC;WAE/C,KAAK;AACZ,QAAK,oBAAoB,IAAI;;AAKjC,MAAI,WAAW,mBAAmB,KAAK,kBACrC,iBAAgB,OAAO,KAAK,WAAW;AAGzC,QAAM,SAAS;;CAKjB,cAAsB,SAAuB,MAA0B;EACrE,MAAM,UAAU,KAAK;AACrB,UAAQ,OAAO;AACf,OAAK,MAAM,QAAQ,KACjB,SAAQ,IAAI,KAAK,IAAI,KAAK;AAI5B,OAAK,MAAM,QAAQ,SAAS;GAC1B,MAAM,WAAW,QAAQ,IAAI,KAAK,GAAG;AACrC,OAAI,CAAC,YAAY,aAAa,MAAM;AAClC,SAAK,eAAe,IAAI,KAAK,IAAI,KAAK;AACtC,SAAK,gBAAgB,OAAO,KAAK,GAAG;;AAEtC,WAAQ,OAAO,KAAK,GAAG;;AAIzB,OAAK,MAAM,CAAC,OAAO,SAAS;AAC1B,QAAK,gBAAgB,IAAI,GAAG;AAC5B,QAAK,eAAe,OAAO,GAAG;;AAGhC,UAAQ,OAAO;;CAGjB,cAA+B;AAC7B,SAAO,KAAK,iBAAiB,KAAK,eAAe,OAAO,KAAK,KAAK,gBAAgB,OAAO;;CAK3F,gBAA8B;AAC5B,OAAK,aAAa;EAClB,MAAM,QAAS,KAAK,YAA4C;AAChE,MAAI,SAAS,GAAG;AACd,QAAK,UAAU;AACf;;AAEF,OAAK,cAAc,iBAAiB,KAAK,UAAU,EAAE,MAAM;;CAG7D,cAA4B;AAC1B,MAAI,KAAK,gBAAgB,MAAM;AAC7B,gBAAa,KAAK,YAAY;AAC9B,QAAK,cAAc;;;CAIvB,WAAyB;AACvB,MAAI,CAAC,KAAK,aAAa,CAAE;AACzB,MAAI;GACF,MAAM,SAAS,KAAK,QAAQ;AAC5B,OAAI,kBAAkB,QACpB,QAAO,OAAO,QAAQ,KAAK,oBAAoB,IAAI,CAAC;WAE/C,KAAK;AACZ,QAAK,oBAAoB,IAAI;;;CAIjC,SAAuC;EACrC,MAAM,UAAU,KAAK;EACrB,MAAM,SAAS,KAAK,eAAe,OAAO,IAAI,CAAC,GAAG,KAAK,eAAe,QAAQ,CAAC,GAAG;EAClF,MAAM,UAAU,KAAK,gBAAgB,OAAO,IAAI,CAAC,GAAG,KAAK,gBAAgB,GAAG;AAG5E,OAAK,gBAAgB;AACrB,OAAK,eAAe,OAAO;AAC3B,OAAK,gBAAgB,OAAO;AAE5B,MAAI,SAAS;GACX,MAAM,cAAc,KAAK,cAAc;AACvC,OAAI,uBAAuB,QACzB,QAAO,YAAY,WAAW;AAC5B,QAAI,OAAQ,QAAO,KAAK,WAAW,OAAO;KAC1C;AAEJ,OAAI,OACF,QAAO,KAAK,WAAW,OAAO;AAEhC;;AAIF,MAAI,SAAS;GACX,MAAM,eAAe,KAAK,cAAc,QAAQ;AAChD,OAAI,wBAAwB,QAC1B,QAAO,aAAa,WAAW;AAC7B,QAAI,OAAQ,QAAO,KAAK,WAAW,OAAO;KAC1C;;AAGN,MAAI,OACF,QAAO,KAAK,WAAW,OAAO;;CAMlC,oBAA4B,KAAoB;AAC9C,MAAI,KAAK,gBAAgB;AACvB,QAAK,eAAe,IAAI;AACxB;;AAEF,MAAI,QACF,SAAQ,KAAK,4BAA4B,IAAI"}