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
package/dist/Resource.js CHANGED
@@ -1,175 +1,167 @@
1
- import { Collection } from "./Collection.js";
2
1
  import { walkPrototypeChain } from "./walkPrototypeChain.js";
3
2
  import { wrapAsyncMethods } from "./wrapAsyncMethods.js";
4
- const __DEV__ = typeof __MVC_KIT_DEV__ !== "undefined" && __MVC_KIT_DEV__;
5
- const DEFAULT_TASK_STATE = Object.freeze({ loading: false, error: null, errorCode: null });
6
- const RESERVED_ASYNC_KEYS = ["async", "subscribeAsync"];
7
- const LIFECYCLE_HOOKS = /* @__PURE__ */ new Set(["onInit", "onDispose"]);
8
- class Resource extends Collection {
9
- _external = null;
10
- _initialized = false;
11
- // ── Async tracking fields ──
12
- _asyncStates = /* @__PURE__ */ new Map();
13
- _asyncSnapshots = /* @__PURE__ */ new Map();
14
- _asyncListeners = /* @__PURE__ */ new Set();
15
- _asyncProxy = null;
16
- _activeOps = null;
17
- /** DEV-only timeout (ms) for detecting ghost async operations after dispose. */
18
- static GHOST_TIMEOUT = 3e3;
19
- constructor(collectionOrItems) {
20
- const isExternal = collectionOrItems != null && !Array.isArray(collectionOrItems);
21
- super(isExternal ? [] : collectionOrItems ?? []);
22
- if (isExternal) {
23
- this._external = collectionOrItems;
24
- if (__DEV__) {
25
- const Ctor = this.constructor;
26
- if (Ctor.MAX_SIZE > 0 || Ctor.TTL > 0) {
27
- console.warn(
28
- `[mvc-kit] Resource "${Ctor.name}" has MAX_SIZE or TTL set but uses an injected Collection. Configure these on the Collection instead.`
29
- );
30
- }
31
- }
32
- }
33
- this._guardReservedKeys();
34
- }
35
- // ── Lifecycle ─────────────────────────────────────────────────
36
- /** Whether init() has been called. */
37
- get initialized() {
38
- return this._initialized;
39
- }
40
- /** Initializes the instance. Called automatically by React hooks after mount. */
41
- init() {
42
- if (this._initialized || this.disposed) return;
43
- this._initialized = true;
44
- if (__DEV__) {
45
- this._activeOps = /* @__PURE__ */ new Map();
46
- }
47
- wrapAsyncMethods({
48
- instance: this,
49
- stopPrototype: Resource.prototype,
50
- reservedKeys: RESERVED_ASYNC_KEYS,
51
- lifecycleHooks: LIFECYCLE_HOOKS,
52
- isDisposed: () => this.disposed,
53
- isInitialized: () => this._initialized,
54
- asyncStates: this._asyncStates,
55
- asyncSnapshots: this._asyncSnapshots,
56
- asyncListeners: this._asyncListeners,
57
- notifyAsync: () => this._notifyAsync(),
58
- addCleanup: (fn) => this.addCleanup(fn),
59
- ghostTimeout: this.constructor.GHOST_TIMEOUT,
60
- className: "Resource",
61
- activeOps: this._activeOps
62
- });
63
- return this.onInit?.();
64
- }
65
- // ── Collection delegation ─────────────────────────────────────
66
- /** Current items array. Delegates to external Collection when injected. */
67
- get state() {
68
- return this._external ? this._external.state : super.state;
69
- }
70
- /** The raw array of items. Delegates to external Collection when injected. */
71
- get items() {
72
- return this._external ? this._external.items : super.items;
73
- }
74
- /** Number of items. Delegates to external Collection when injected. */
75
- get length() {
76
- return this._external ? this._external.length : super.length;
77
- }
78
- add(...items) {
79
- this._external ? this._external.add(...items) : super.add(...items);
80
- }
81
- upsert(...items) {
82
- this._external ? this._external.upsert(...items) : super.upsert(...items);
83
- }
84
- update(id, changes) {
85
- this._external ? this._external.update(id, changes) : super.update(id, changes);
86
- }
87
- remove(...ids) {
88
- this._external ? this._external.remove(...ids) : super.remove(...ids);
89
- }
90
- reset(items) {
91
- this._external ? this._external.reset(items) : super.reset(items);
92
- }
93
- clear() {
94
- this._external ? this._external.clear() : super.clear();
95
- }
96
- optimistic(callback) {
97
- return this._external ? this._external.optimistic(callback) : super.optimistic(callback);
98
- }
99
- get(id) {
100
- return this._external ? this._external.get(id) : super.get(id);
101
- }
102
- has(id) {
103
- return this._external ? this._external.has(id) : super.has(id);
104
- }
105
- find(predicate) {
106
- return this._external ? this._external.find(predicate) : super.find(predicate);
107
- }
108
- filter(predicate) {
109
- return this._external ? this._external.filter(predicate) : super.filter(predicate);
110
- }
111
- sorted(compareFn) {
112
- return this._external ? this._external.sorted(compareFn) : super.sorted(compareFn);
113
- }
114
- map(fn) {
115
- return this._external ? this._external.map(fn) : super.map(fn);
116
- }
117
- subscribe(listener) {
118
- if (this.disposed) return () => {
119
- };
120
- return this._external ? this._external.subscribe(listener) : super.subscribe(listener);
121
- }
122
- // ── Async tracking API ────────────────────────────────────────
123
- /** Proxy providing `TaskState` (loading, error, errorCode) per async method. */
124
- get async() {
125
- if (!this._asyncProxy) {
126
- const self = this;
127
- this._asyncProxy = new Proxy({}, {
128
- get(_, prop) {
129
- return self._asyncSnapshots.get(prop) ?? DEFAULT_TASK_STATE;
130
- },
131
- has(_, prop) {
132
- return self._asyncSnapshots.has(prop);
133
- },
134
- ownKeys() {
135
- return Array.from(self._asyncSnapshots.keys());
136
- },
137
- getOwnPropertyDescriptor(_, prop) {
138
- if (self._asyncSnapshots.has(prop)) {
139
- return { configurable: true, enumerable: true, value: self._asyncSnapshots.get(prop) };
140
- }
141
- return void 0;
142
- }
143
- });
144
- }
145
- return this._asyncProxy;
146
- }
147
- /** Subscribes to async state changes. Used by `useInstance` for React integration. */
148
- subscribeAsync(listener) {
149
- if (this.disposed) return () => {
150
- };
151
- this._asyncListeners.add(listener);
152
- return () => {
153
- this._asyncListeners.delete(listener);
154
- };
155
- }
156
- // ── Private: async tracking internals ─────────────────────────
157
- _notifyAsync() {
158
- for (const listener of this._asyncListeners) {
159
- listener();
160
- }
161
- }
162
- _guardReservedKeys() {
163
- walkPrototypeChain(this, Resource.prototype, (key) => {
164
- if (RESERVED_ASYNC_KEYS.includes(key)) {
165
- throw new Error(
166
- `[mvc-kit] "${key}" is a reserved property on Resource and cannot be overridden.`
167
- );
168
- }
169
- });
170
- }
171
- }
172
- export {
173
- Resource
3
+ import { Collection } from "./Collection.js";
4
+ //#region src/Resource.ts
5
+ var __DEV__ = typeof __MVC_KIT_DEV__ !== "undefined" && __MVC_KIT_DEV__;
6
+ var DEFAULT_TASK_STATE = Object.freeze({
7
+ loading: false,
8
+ error: null,
9
+ errorCode: null
10
+ });
11
+ var RESERVED_ASYNC_KEYS = ["async", "subscribeAsync"];
12
+ var LIFECYCLE_HOOKS = new Set(["onInit", "onDispose"]);
13
+ /**
14
+ * Collection + async tracking toolkit. Extends Collection with lifecycle
15
+ * (init/dispose) and automatic async method tracking. Optionally delegates
16
+ * to an external Collection for shared data scenarios.
17
+ */
18
+ var Resource = class Resource extends Collection {
19
+ _external = null;
20
+ _initialized = false;
21
+ _asyncStates = /* @__PURE__ */ new Map();
22
+ _asyncSnapshots = /* @__PURE__ */ new Map();
23
+ _asyncListeners = /* @__PURE__ */ new Set();
24
+ _asyncProxy = null;
25
+ _activeOps = null;
26
+ /** DEV-only timeout (ms) for detecting ghost async operations after dispose. */
27
+ static GHOST_TIMEOUT = 3e3;
28
+ constructor(collectionOrItems) {
29
+ const isExternal = collectionOrItems != null && !Array.isArray(collectionOrItems);
30
+ super(isExternal ? [] : collectionOrItems ?? []);
31
+ if (isExternal) {
32
+ this._external = collectionOrItems;
33
+ if (__DEV__) {
34
+ const Ctor = this.constructor;
35
+ if (Ctor.MAX_SIZE > 0 || Ctor.TTL > 0) console.warn(`[mvc-kit] Resource "${Ctor.name}" has MAX_SIZE or TTL set but uses an injected Collection. Configure these on the Collection instead.`);
36
+ }
37
+ }
38
+ this._guardReservedKeys();
39
+ }
40
+ /** Whether init() has been called. */
41
+ get initialized() {
42
+ return this._initialized;
43
+ }
44
+ /** Initializes the instance. Called automatically by React hooks after mount. */
45
+ init() {
46
+ if (this._initialized || this.disposed) return;
47
+ this._initialized = true;
48
+ if (__DEV__) this._activeOps = /* @__PURE__ */ new Map();
49
+ wrapAsyncMethods({
50
+ instance: this,
51
+ stopPrototype: Resource.prototype,
52
+ reservedKeys: RESERVED_ASYNC_KEYS,
53
+ lifecycleHooks: LIFECYCLE_HOOKS,
54
+ isDisposed: () => this.disposed,
55
+ isInitialized: () => this._initialized,
56
+ asyncStates: this._asyncStates,
57
+ asyncSnapshots: this._asyncSnapshots,
58
+ asyncListeners: this._asyncListeners,
59
+ notifyAsync: () => this._notifyAsync(),
60
+ addCleanup: (fn) => this.addCleanup(fn),
61
+ ghostTimeout: this.constructor.GHOST_TIMEOUT,
62
+ className: "Resource",
63
+ activeOps: this._activeOps
64
+ });
65
+ return this.onInit?.();
66
+ }
67
+ /** Current items array. Delegates to external Collection when injected. */
68
+ get state() {
69
+ return this._external ? this._external.state : super.state;
70
+ }
71
+ /** The raw array of items. Delegates to external Collection when injected. */
72
+ get items() {
73
+ return this._external ? this._external.items : super.items;
74
+ }
75
+ /** Number of items. Delegates to external Collection when injected. */
76
+ get length() {
77
+ return this._external ? this._external.length : super.length;
78
+ }
79
+ add(...items) {
80
+ this._external ? this._external.add(...items) : super.add(...items);
81
+ }
82
+ upsert(...items) {
83
+ this._external ? this._external.upsert(...items) : super.upsert(...items);
84
+ }
85
+ update(id, changes) {
86
+ this._external ? this._external.update(id, changes) : super.update(id, changes);
87
+ }
88
+ remove(...ids) {
89
+ this._external ? this._external.remove(...ids) : super.remove(...ids);
90
+ }
91
+ reset(items) {
92
+ this._external ? this._external.reset(items) : super.reset(items);
93
+ }
94
+ clear() {
95
+ this._external ? this._external.clear() : super.clear();
96
+ }
97
+ optimistic(callback) {
98
+ return this._external ? this._external.optimistic(callback) : super.optimistic(callback);
99
+ }
100
+ get(id) {
101
+ return this._external ? this._external.get(id) : super.get(id);
102
+ }
103
+ has(id) {
104
+ return this._external ? this._external.has(id) : super.has(id);
105
+ }
106
+ find(predicate) {
107
+ return this._external ? this._external.find(predicate) : super.find(predicate);
108
+ }
109
+ filter(predicate) {
110
+ return this._external ? this._external.filter(predicate) : super.filter(predicate);
111
+ }
112
+ sorted(compareFn) {
113
+ return this._external ? this._external.sorted(compareFn) : super.sorted(compareFn);
114
+ }
115
+ map(fn) {
116
+ return this._external ? this._external.map(fn) : super.map(fn);
117
+ }
118
+ subscribe(listener) {
119
+ if (this.disposed) return () => {};
120
+ return this._external ? this._external.subscribe(listener) : super.subscribe(listener);
121
+ }
122
+ /** Proxy providing `TaskState` (loading, error, errorCode) per async method. */
123
+ get async() {
124
+ if (!this._asyncProxy) {
125
+ const self = this;
126
+ this._asyncProxy = new Proxy({}, {
127
+ get(_, prop) {
128
+ return self._asyncSnapshots.get(prop) ?? DEFAULT_TASK_STATE;
129
+ },
130
+ has(_, prop) {
131
+ return self._asyncSnapshots.has(prop);
132
+ },
133
+ ownKeys() {
134
+ return Array.from(self._asyncSnapshots.keys());
135
+ },
136
+ getOwnPropertyDescriptor(_, prop) {
137
+ if (self._asyncSnapshots.has(prop)) return {
138
+ configurable: true,
139
+ enumerable: true,
140
+ value: self._asyncSnapshots.get(prop)
141
+ };
142
+ }
143
+ });
144
+ }
145
+ return this._asyncProxy;
146
+ }
147
+ /** Subscribes to async state changes. Used by `useInstance` for React integration. */
148
+ subscribeAsync(listener) {
149
+ if (this.disposed) return () => {};
150
+ this._asyncListeners.add(listener);
151
+ return () => {
152
+ this._asyncListeners.delete(listener);
153
+ };
154
+ }
155
+ _notifyAsync() {
156
+ for (const listener of this._asyncListeners) listener();
157
+ }
158
+ _guardReservedKeys() {
159
+ walkPrototypeChain(this, Resource.prototype, (key) => {
160
+ if (RESERVED_ASYNC_KEYS.includes(key)) throw new Error(`[mvc-kit] "${key}" is a reserved property on Resource and cannot be overridden.`);
161
+ });
162
+ }
174
163
  };
175
- //# sourceMappingURL=Resource.js.map
164
+ //#endregion
165
+ export { Resource };
166
+
167
+ //# sourceMappingURL=Resource.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"Resource.js","sources":["../src/Resource.ts"],"sourcesContent":["import { Collection } from './Collection';\nimport { walkPrototypeChain } from './walkPrototypeChain';\nimport { wrapAsyncMethods } from './wrapAsyncMethods';\nimport type { InternalTaskState } from './wrapAsyncMethods';\nimport type { Listener, TaskState } from './types';\n\nconst __DEV__ = typeof __MVC_KIT_DEV__ !== 'undefined' && __MVC_KIT_DEV__;\n\n// ── Async tracking types ────────────────────────────────────────\n\nconst DEFAULT_TASK_STATE: TaskState = Object.freeze({ loading: false, error: null, errorCode: null });\n\nexport type ResourceAsyncMethodKeys<T> = {\n [K in Exclude<keyof T, keyof Resource<any>>]: T[K] extends (...args: any[]) => Promise<any> ? K : never;\n}[Exclude<keyof T, keyof Resource<any>>];\n\ntype ResourceAsyncMap<T> = {\n readonly [K in ResourceAsyncMethodKeys<T>]: TaskState;\n};\n\nconst RESERVED_ASYNC_KEYS = ['async', 'subscribeAsync'] as const;\nconst LIFECYCLE_HOOKS = new Set(['onInit', 'onDispose']);\n\n// ── Resource ────────────────────────────────────────────────────\n\n/**\n * Collection + async tracking toolkit. Extends Collection with lifecycle\n * (init/dispose) and automatic async method tracking. Optionally delegates\n * to an external Collection for shared data scenarios.\n */\nexport class Resource<T extends { id: string | number }> extends Collection<T> {\n private _external: Collection<T> | null = null;\n private _initialized = false;\n\n // ── Async tracking fields ──\n private _asyncStates = new Map<string, InternalTaskState>();\n private _asyncSnapshots = new Map<string, TaskState>();\n private _asyncListeners = new Set<() => void>();\n private _asyncProxy: ResourceAsyncMap<this> | null = null;\n private _activeOps: Map<string, number> | null = null;\n\n /** DEV-only timeout (ms) for detecting ghost async operations after dispose. */\n static GHOST_TIMEOUT = 3000;\n\n constructor(collectionOrItems?: Collection<T> | T[]) {\n const isExternal = collectionOrItems != null && !Array.isArray(collectionOrItems);\n super(isExternal ? [] : (collectionOrItems as T[]) ?? []);\n\n if (isExternal) {\n this._external = collectionOrItems as Collection<T>;\n\n if (__DEV__) {\n const Ctor = this.constructor as typeof Resource;\n if (Ctor.MAX_SIZE > 0 || Ctor.TTL > 0) {\n console.warn(\n `[mvc-kit] Resource \"${Ctor.name}\" has MAX_SIZE or TTL set but uses an ` +\n `injected Collection. Configure these on the Collection instead.`\n );\n }\n }\n }\n\n this._guardReservedKeys();\n }\n\n // ── Lifecycle ─────────────────────────────────────────────────\n\n /** Whether init() has been called. */\n get initialized(): boolean {\n return this._initialized;\n }\n\n /** Initializes the instance. Called automatically by React hooks after mount. */\n init(): void | Promise<void> {\n if (this._initialized || this.disposed) return;\n this._initialized = true;\n\n if (__DEV__) {\n this._activeOps = new Map();\n }\n\n wrapAsyncMethods({\n instance: this,\n stopPrototype: Resource.prototype,\n reservedKeys: RESERVED_ASYNC_KEYS,\n lifecycleHooks: LIFECYCLE_HOOKS,\n isDisposed: () => this.disposed,\n isInitialized: () => this._initialized,\n asyncStates: this._asyncStates,\n asyncSnapshots: this._asyncSnapshots,\n asyncListeners: this._asyncListeners,\n notifyAsync: () => this._notifyAsync(),\n addCleanup: (fn) => this.addCleanup(fn),\n ghostTimeout: (this.constructor as typeof Resource).GHOST_TIMEOUT,\n className: 'Resource',\n activeOps: this._activeOps,\n });\n\n return this.onInit?.();\n }\n\n /** Lifecycle hook called at the end of init(). Override to load initial data. @protected */\n protected onInit?(): void | Promise<void>;\n\n // ── Collection delegation ─────────────────────────────────────\n\n /** Current items array. Delegates to external Collection when injected. */\n get state(): T[] {\n return this._external ? this._external.state : super.state;\n }\n\n /** The raw array of items. Delegates to external Collection when injected. */\n get items(): T[] {\n return this._external ? this._external.items : super.items;\n }\n\n /** Number of items. Delegates to external Collection when injected. */\n get length(): number {\n return this._external ? this._external.length : super.length;\n }\n\n add(...items: T[]): void {\n this._external ? this._external.add(...items) : super.add(...items);\n }\n\n upsert(...items: T[]): void {\n this._external ? this._external.upsert(...items) : super.upsert(...items);\n }\n\n update(id: T['id'], changes: Partial<T>): void {\n this._external ? this._external.update(id, changes) : super.update(id, changes);\n }\n\n remove(...ids: T['id'][]): void {\n this._external ? this._external.remove(...ids) : super.remove(...ids);\n }\n\n reset(items: T[]): void {\n this._external ? this._external.reset(items) : super.reset(items);\n }\n\n clear(): void {\n this._external ? this._external.clear() : super.clear();\n }\n\n optimistic(callback: () => void): () => void {\n return this._external ? this._external.optimistic(callback) : super.optimistic(callback);\n }\n\n get(id: T['id']): T | undefined {\n return this._external ? this._external.get(id) : super.get(id);\n }\n\n has(id: T['id']): boolean {\n return this._external ? this._external.has(id) : super.has(id);\n }\n\n find(predicate: (item: T) => boolean): T | undefined {\n return this._external ? this._external.find(predicate) : super.find(predicate);\n }\n\n filter(predicate: (item: T) => boolean): T[] {\n return this._external ? this._external.filter(predicate) : super.filter(predicate);\n }\n\n sorted(compareFn: (a: T, b: T) => number): T[] {\n return this._external ? this._external.sorted(compareFn) : super.sorted(compareFn);\n }\n\n map<U>(fn: (item: T) => U): U[] {\n return this._external ? this._external.map(fn) : super.map(fn);\n }\n\n subscribe(listener: Listener<T[]>): () => void {\n if (this.disposed) return () => {};\n return this._external ? this._external.subscribe(listener) : super.subscribe(listener);\n }\n\n // ── Async tracking API ────────────────────────────────────────\n\n /** Proxy providing `TaskState` (loading, error, errorCode) per async method. */\n get async(): ResourceAsyncMap<this> {\n if (!this._asyncProxy) {\n const self = this;\n this._asyncProxy = new Proxy({} as ResourceAsyncMap<this>, {\n get(_, prop: string) {\n return self._asyncSnapshots.get(prop) ?? DEFAULT_TASK_STATE;\n },\n has(_, prop: string) {\n return self._asyncSnapshots.has(prop);\n },\n ownKeys() {\n return Array.from(self._asyncSnapshots.keys());\n },\n getOwnPropertyDescriptor(_, prop: string) {\n if (self._asyncSnapshots.has(prop)) {\n return { configurable: true, enumerable: true, value: self._asyncSnapshots.get(prop) };\n }\n return undefined;\n },\n });\n }\n return this._asyncProxy;\n }\n\n /** Subscribes to async state changes. Used by `useInstance` for React integration. */\n subscribeAsync(listener: () => void): () => void {\n if (this.disposed) return () => {};\n this._asyncListeners.add(listener);\n return () => { this._asyncListeners.delete(listener); };\n }\n\n // ── Private: async tracking internals ─────────────────────────\n\n private _notifyAsync(): void {\n for (const listener of this._asyncListeners) {\n listener();\n }\n }\n\n private _guardReservedKeys(): void {\n walkPrototypeChain(this, Resource.prototype, (key) => {\n if (RESERVED_ASYNC_KEYS.includes(key as any)) {\n throw new Error(\n `[mvc-kit] \"${key}\" is a reserved property on Resource and cannot be overridden.`\n );\n }\n });\n }\n\n}\n"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"Resource.js","names":[],"sources":["../src/Resource.ts"],"sourcesContent":["import { Collection } from './Collection';\nimport { walkPrototypeChain } from './walkPrototypeChain';\nimport { wrapAsyncMethods } from './wrapAsyncMethods';\nimport type { InternalTaskState } from './wrapAsyncMethods';\nimport type { Listener, TaskState } from './types';\n\nconst __DEV__ = typeof __MVC_KIT_DEV__ !== 'undefined' && __MVC_KIT_DEV__;\n\n// ── Async tracking types ────────────────────────────────────────\n\nconst DEFAULT_TASK_STATE: TaskState = Object.freeze({ loading: false, error: null, errorCode: null });\n\nexport type ResourceAsyncMethodKeys<T> = {\n [K in Exclude<keyof T, keyof Resource<any>>]: T[K] extends (...args: any[]) => Promise<any> ? K : never;\n}[Exclude<keyof T, keyof Resource<any>>];\n\ntype ResourceAsyncMap<T> = {\n readonly [K in ResourceAsyncMethodKeys<T>]: TaskState;\n};\n\nconst RESERVED_ASYNC_KEYS = ['async', 'subscribeAsync'] as const;\nconst LIFECYCLE_HOOKS = new Set(['onInit', 'onDispose']);\n\n// ── Resource ────────────────────────────────────────────────────\n\n/**\n * Collection + async tracking toolkit. Extends Collection with lifecycle\n * (init/dispose) and automatic async method tracking. Optionally delegates\n * to an external Collection for shared data scenarios.\n */\nexport class Resource<T extends { id: string | number }> extends Collection<T> {\n private _external: Collection<T> | null = null;\n private _initialized = false;\n\n // ── Async tracking fields ──\n private _asyncStates = new Map<string, InternalTaskState>();\n private _asyncSnapshots = new Map<string, TaskState>();\n private _asyncListeners = new Set<() => void>();\n private _asyncProxy: ResourceAsyncMap<this> | null = null;\n private _activeOps: Map<string, number> | null = null;\n\n /** DEV-only timeout (ms) for detecting ghost async operations after dispose. */\n static GHOST_TIMEOUT = 3000;\n\n constructor(collectionOrItems?: Collection<T> | T[]) {\n const isExternal = collectionOrItems != null && !Array.isArray(collectionOrItems);\n super(isExternal ? [] : (collectionOrItems as T[]) ?? []);\n\n if (isExternal) {\n this._external = collectionOrItems as Collection<T>;\n\n if (__DEV__) {\n const Ctor = this.constructor as typeof Resource;\n if (Ctor.MAX_SIZE > 0 || Ctor.TTL > 0) {\n console.warn(\n `[mvc-kit] Resource \"${Ctor.name}\" has MAX_SIZE or TTL set but uses an ` +\n `injected Collection. Configure these on the Collection instead.`\n );\n }\n }\n }\n\n this._guardReservedKeys();\n }\n\n // ── Lifecycle ─────────────────────────────────────────────────\n\n /** Whether init() has been called. */\n get initialized(): boolean {\n return this._initialized;\n }\n\n /** Initializes the instance. Called automatically by React hooks after mount. */\n init(): void | Promise<void> {\n if (this._initialized || this.disposed) return;\n this._initialized = true;\n\n if (__DEV__) {\n this._activeOps = new Map();\n }\n\n wrapAsyncMethods({\n instance: this,\n stopPrototype: Resource.prototype,\n reservedKeys: RESERVED_ASYNC_KEYS,\n lifecycleHooks: LIFECYCLE_HOOKS,\n isDisposed: () => this.disposed,\n isInitialized: () => this._initialized,\n asyncStates: this._asyncStates,\n asyncSnapshots: this._asyncSnapshots,\n asyncListeners: this._asyncListeners,\n notifyAsync: () => this._notifyAsync(),\n addCleanup: (fn) => this.addCleanup(fn),\n ghostTimeout: (this.constructor as typeof Resource).GHOST_TIMEOUT,\n className: 'Resource',\n activeOps: this._activeOps,\n });\n\n return this.onInit?.();\n }\n\n /** Lifecycle hook called at the end of init(). Override to load initial data. @protected */\n protected onInit?(): void | Promise<void>;\n\n // ── Collection delegation ─────────────────────────────────────\n\n /** Current items array. Delegates to external Collection when injected. */\n get state(): T[] {\n return this._external ? this._external.state : super.state;\n }\n\n /** The raw array of items. Delegates to external Collection when injected. */\n get items(): T[] {\n return this._external ? this._external.items : super.items;\n }\n\n /** Number of items. Delegates to external Collection when injected. */\n get length(): number {\n return this._external ? this._external.length : super.length;\n }\n\n add(...items: T[]): void {\n this._external ? this._external.add(...items) : super.add(...items);\n }\n\n upsert(...items: T[]): void {\n this._external ? this._external.upsert(...items) : super.upsert(...items);\n }\n\n update(id: T['id'], changes: Partial<T>): void {\n this._external ? this._external.update(id, changes) : super.update(id, changes);\n }\n\n remove(...ids: T['id'][]): void {\n this._external ? this._external.remove(...ids) : super.remove(...ids);\n }\n\n reset(items: T[]): void {\n this._external ? this._external.reset(items) : super.reset(items);\n }\n\n clear(): void {\n this._external ? this._external.clear() : super.clear();\n }\n\n optimistic(callback: () => void): () => void {\n return this._external ? this._external.optimistic(callback) : super.optimistic(callback);\n }\n\n get(id: T['id']): T | undefined {\n return this._external ? this._external.get(id) : super.get(id);\n }\n\n has(id: T['id']): boolean {\n return this._external ? this._external.has(id) : super.has(id);\n }\n\n find(predicate: (item: T) => boolean): T | undefined {\n return this._external ? this._external.find(predicate) : super.find(predicate);\n }\n\n filter(predicate: (item: T) => boolean): T[] {\n return this._external ? this._external.filter(predicate) : super.filter(predicate);\n }\n\n sorted(compareFn: (a: T, b: T) => number): T[] {\n return this._external ? this._external.sorted(compareFn) : super.sorted(compareFn);\n }\n\n map<U>(fn: (item: T) => U): U[] {\n return this._external ? this._external.map(fn) : super.map(fn);\n }\n\n subscribe(listener: Listener<T[]>): () => void {\n if (this.disposed) return () => {};\n return this._external ? this._external.subscribe(listener) : super.subscribe(listener);\n }\n\n // ── Async tracking API ────────────────────────────────────────\n\n /** Proxy providing `TaskState` (loading, error, errorCode) per async method. */\n get async(): ResourceAsyncMap<this> {\n if (!this._asyncProxy) {\n const self = this;\n this._asyncProxy = new Proxy({} as ResourceAsyncMap<this>, {\n get(_, prop: string) {\n return self._asyncSnapshots.get(prop) ?? DEFAULT_TASK_STATE;\n },\n has(_, prop: string) {\n return self._asyncSnapshots.has(prop);\n },\n ownKeys() {\n return Array.from(self._asyncSnapshots.keys());\n },\n getOwnPropertyDescriptor(_, prop: string) {\n if (self._asyncSnapshots.has(prop)) {\n return { configurable: true, enumerable: true, value: self._asyncSnapshots.get(prop) };\n }\n return undefined;\n },\n });\n }\n return this._asyncProxy;\n }\n\n /** Subscribes to async state changes. Used by `useInstance` for React integration. */\n subscribeAsync(listener: () => void): () => void {\n if (this.disposed) return () => {};\n this._asyncListeners.add(listener);\n return () => { this._asyncListeners.delete(listener); };\n }\n\n // ── Private: async tracking internals ─────────────────────────\n\n private _notifyAsync(): void {\n for (const listener of this._asyncListeners) {\n listener();\n }\n }\n\n private _guardReservedKeys(): void {\n walkPrototypeChain(this, Resource.prototype, (key) => {\n if (RESERVED_ASYNC_KEYS.includes(key as any)) {\n throw new Error(\n `[mvc-kit] \"${key}\" is a reserved property on Resource and cannot be overridden.`\n );\n }\n });\n }\n\n}\n"],"mappings":"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"}
@@ -1,95 +1,85 @@
1
- "use strict";
2
- Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
3
- const Trackable = require("./Trackable.cjs");
4
- class Selection extends Trackable.Trackable {
5
- _selected = /* @__PURE__ */ new Set();
6
- _readonlyView = this._selected;
7
- constructor() {
8
- super();
9
- }
10
- // ── Readable state ──
11
- /** Read-only view of currently selected keys. */
12
- get selected() {
13
- return this._readonlyView;
14
- }
15
- /** Number of currently selected items. */
16
- get count() {
17
- return this._selected.size;
18
- }
19
- /** Whether any items are currently selected. */
20
- get hasSelection() {
21
- return this._selected.size > 0;
22
- }
23
- // ── Query ──
24
- /** Check whether a specific key is selected. */
25
- isSelected(key) {
26
- return this._selected.has(key);
27
- }
28
- // ── Actions ──
29
- /** Toggle a key's selection state (select if unselected, deselect if selected). */
30
- toggle(key) {
31
- if (this._selected.has(key)) {
32
- this._selected.delete(key);
33
- } else {
34
- this._selected.add(key);
35
- }
36
- this._publish();
37
- }
38
- /** Add one or more keys to the selection. */
39
- select(...keys) {
40
- let changed = false;
41
- for (const key of keys) {
42
- if (!this._selected.has(key)) {
43
- this._selected.add(key);
44
- changed = true;
45
- }
46
- }
47
- if (changed) this._publish();
48
- }
49
- /** Remove one or more keys from the selection. */
50
- deselect(...keys) {
51
- let changed = false;
52
- for (const key of keys) {
53
- if (this._selected.has(key)) {
54
- this._selected.delete(key);
55
- changed = true;
56
- }
57
- }
58
- if (changed) this._publish();
59
- }
60
- /** If all selected deselect all, else select all. */
61
- toggleAll(allKeys) {
62
- const allSelected = allKeys.length > 0 && allKeys.every((k) => this._selected.has(k));
63
- if (allSelected) {
64
- this._selected.clear();
65
- } else {
66
- for (const key of allKeys) this._selected.add(key);
67
- }
68
- this._publish();
69
- }
70
- /** Replace the entire selection atomically. Single notification. */
71
- set(...keys) {
72
- if (keys.length === this._selected.size && keys.every((k) => this._selected.has(k))) return;
73
- this._selected.clear();
74
- for (const key of keys) this._selected.add(key);
75
- this._publish();
76
- }
77
- /** Remove all items from the selection. */
78
- clear() {
79
- if (this._selected.size === 0) return;
80
- this._selected.clear();
81
- this._publish();
82
- }
83
- // ── Utility ──
84
- /** Filter an array to only items whose key is in the selection. */
85
- selectedFrom(items, keyOf) {
86
- return items.filter((item) => this._selected.has(keyOf(item)));
87
- }
88
- // ── Internal ──
89
- _publish() {
90
- this._readonlyView = new Set(this._selected);
91
- this.notify();
92
- }
93
- }
1
+ const require_Trackable = require("./Trackable.cjs");
2
+ //#region src/Selection.ts
3
+ /**
4
+ * Key-based selection set with toggle and select-all support.
5
+ * Tracks which items are selected by their key (id).
6
+ * Subscribable — auto-tracked when used as a ViewModel property.
7
+ */
8
+ var Selection = class extends require_Trackable.Trackable {
9
+ _selected = /* @__PURE__ */ new Set();
10
+ _readonlyView = this._selected;
11
+ constructor() {
12
+ super();
13
+ }
14
+ /** Read-only view of currently selected keys. */
15
+ get selected() {
16
+ return this._readonlyView;
17
+ }
18
+ /** Number of currently selected items. */
19
+ get count() {
20
+ return this._selected.size;
21
+ }
22
+ /** Whether any items are currently selected. */
23
+ get hasSelection() {
24
+ return this._selected.size > 0;
25
+ }
26
+ /** Check whether a specific key is selected. */
27
+ isSelected(key) {
28
+ return this._selected.has(key);
29
+ }
30
+ /** Toggle a key's selection state (select if unselected, deselect if selected). */
31
+ toggle(key) {
32
+ if (this._selected.has(key)) this._selected.delete(key);
33
+ else this._selected.add(key);
34
+ this._publish();
35
+ }
36
+ /** Add one or more keys to the selection. */
37
+ select(...keys) {
38
+ let changed = false;
39
+ for (const key of keys) if (!this._selected.has(key)) {
40
+ this._selected.add(key);
41
+ changed = true;
42
+ }
43
+ if (changed) this._publish();
44
+ }
45
+ /** Remove one or more keys from the selection. */
46
+ deselect(...keys) {
47
+ let changed = false;
48
+ for (const key of keys) if (this._selected.has(key)) {
49
+ this._selected.delete(key);
50
+ changed = true;
51
+ }
52
+ if (changed) this._publish();
53
+ }
54
+ /** If all selected → deselect all, else select all. */
55
+ toggleAll(allKeys) {
56
+ if (allKeys.length > 0 && allKeys.every((k) => this._selected.has(k))) this._selected.clear();
57
+ else for (const key of allKeys) this._selected.add(key);
58
+ this._publish();
59
+ }
60
+ /** Replace the entire selection atomically. Single notification. */
61
+ set(...keys) {
62
+ if (keys.length === this._selected.size && keys.every((k) => this._selected.has(k))) return;
63
+ this._selected.clear();
64
+ for (const key of keys) this._selected.add(key);
65
+ this._publish();
66
+ }
67
+ /** Remove all items from the selection. */
68
+ clear() {
69
+ if (this._selected.size === 0) return;
70
+ this._selected.clear();
71
+ this._publish();
72
+ }
73
+ /** Filter an array to only items whose key is in the selection. */
74
+ selectedFrom(items, keyOf) {
75
+ return items.filter((item) => this._selected.has(keyOf(item)));
76
+ }
77
+ _publish() {
78
+ this._readonlyView = new Set(this._selected);
79
+ this.notify();
80
+ }
81
+ };
82
+ //#endregion
94
83
  exports.Selection = Selection;
95
- //# sourceMappingURL=Selection.cjs.map
84
+
85
+ //# sourceMappingURL=Selection.cjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"Selection.cjs","sources":["../src/Selection.ts"],"sourcesContent":["import { Trackable } from './Trackable';\n\n/**\n * Key-based selection set with toggle and select-all support.\n * Tracks which items are selected by their key (id).\n * Subscribable — auto-tracked when used as a ViewModel property.\n */\nexport class Selection<K extends string | number = string | number> extends Trackable {\n private _selected: Set<K> = new Set();\n private _readonlyView: ReadonlySet<K> = this._selected;\n\n constructor() {\n super();\n }\n\n // ── Readable state ──\n\n /** Read-only view of currently selected keys. */\n get selected(): ReadonlySet<K> {\n return this._readonlyView;\n }\n\n /** Number of currently selected items. */\n get count(): number {\n return this._selected.size;\n }\n\n /** Whether any items are currently selected. */\n get hasSelection(): boolean {\n return this._selected.size > 0;\n }\n\n // ── Query ──\n\n /** Check whether a specific key is selected. */\n isSelected(key: K): boolean {\n return this._selected.has(key);\n }\n\n // ── Actions ──\n\n /** Toggle a key's selection state (select if unselected, deselect if selected). */\n toggle(key: K): void {\n if (this._selected.has(key)) {\n this._selected.delete(key);\n } else {\n this._selected.add(key);\n }\n this._publish();\n }\n\n /** Add one or more keys to the selection. */\n select(...keys: K[]): void {\n let changed = false;\n for (const key of keys) {\n if (!this._selected.has(key)) {\n this._selected.add(key);\n changed = true;\n }\n }\n if (changed) this._publish();\n }\n\n /** Remove one or more keys from the selection. */\n deselect(...keys: K[]): void {\n let changed = false;\n for (const key of keys) {\n if (this._selected.has(key)) {\n this._selected.delete(key);\n changed = true;\n }\n }\n if (changed) this._publish();\n }\n\n /** If all selected → deselect all, else select all. */\n toggleAll(allKeys: K[]): void {\n const allSelected = allKeys.length > 0 && allKeys.every(k => this._selected.has(k));\n if (allSelected) {\n this._selected.clear();\n } else {\n for (const key of allKeys) this._selected.add(key);\n }\n this._publish();\n }\n\n /** Replace the entire selection atomically. Single notification. */\n set(...keys: K[]): void {\n // Check if anything actually changed\n if (keys.length === this._selected.size && keys.every(k => this._selected.has(k))) return;\n this._selected.clear();\n for (const key of keys) this._selected.add(key);\n this._publish();\n }\n\n /** Remove all items from the selection. */\n clear(): void {\n if (this._selected.size === 0) return;\n this._selected.clear();\n this._publish();\n }\n\n // ── Utility ──\n\n /** Filter an array to only items whose key is in the selection. */\n selectedFrom<T>(items: T[], keyOf: (item: T) => K): T[] {\n return items.filter(item => this._selected.has(keyOf(item)));\n }\n\n // ── Internal ──\n\n private _publish(): void {\n // Replace readonlyView so reference equality changes (needed for React)\n this._readonlyView = new Set(this._selected);\n this.notify();\n }\n}\n"],"names":["Trackable"],"mappings":";;;AAOO,MAAM,kBAA+DA,UAAAA,UAAU;AAAA,EAC5E,gCAAwB,IAAA;AAAA,EACxB,gBAAgC,KAAK;AAAA,EAE7C,cAAc;AACZ,UAAA;AAAA,EACF;AAAA;AAAA;AAAA,EAKA,IAAI,WAA2B;AAC7B,WAAO,KAAK;AAAA,EACd;AAAA;AAAA,EAGA,IAAI,QAAgB;AAClB,WAAO,KAAK,UAAU;AAAA,EACxB;AAAA;AAAA,EAGA,IAAI,eAAwB;AAC1B,WAAO,KAAK,UAAU,OAAO;AAAA,EAC/B;AAAA;AAAA;AAAA,EAKA,WAAW,KAAiB;AAC1B,WAAO,KAAK,UAAU,IAAI,GAAG;AAAA,EAC/B;AAAA;AAAA;AAAA,EAKA,OAAO,KAAc;AACnB,QAAI,KAAK,UAAU,IAAI,GAAG,GAAG;AAC3B,WAAK,UAAU,OAAO,GAAG;AAAA,IAC3B,OAAO;AACL,WAAK,UAAU,IAAI,GAAG;AAAA,IACxB;AACA,SAAK,SAAA;AAAA,EACP;AAAA;AAAA,EAGA,UAAU,MAAiB;AACzB,QAAI,UAAU;AACd,eAAW,OAAO,MAAM;AACtB,UAAI,CAAC,KAAK,UAAU,IAAI,GAAG,GAAG;AAC5B,aAAK,UAAU,IAAI,GAAG;AACtB,kBAAU;AAAA,MACZ;AAAA,IACF;AACA,QAAI,cAAc,SAAA;AAAA,EACpB;AAAA;AAAA,EAGA,YAAY,MAAiB;AAC3B,QAAI,UAAU;AACd,eAAW,OAAO,MAAM;AACtB,UAAI,KAAK,UAAU,IAAI,GAAG,GAAG;AAC3B,aAAK,UAAU,OAAO,GAAG;AACzB,kBAAU;AAAA,MACZ;AAAA,IACF;AACA,QAAI,cAAc,SAAA;AAAA,EACpB;AAAA;AAAA,EAGA,UAAU,SAAoB;AAC5B,UAAM,cAAc,QAAQ,SAAS,KAAK,QAAQ,MAAM,CAAA,MAAK,KAAK,UAAU,IAAI,CAAC,CAAC;AAClF,QAAI,aAAa;AACf,WAAK,UAAU,MAAA;AAAA,IACjB,OAAO;AACL,iBAAW,OAAO,QAAS,MAAK,UAAU,IAAI,GAAG;AAAA,IACnD;AACA,SAAK,SAAA;AAAA,EACP;AAAA;AAAA,EAGA,OAAO,MAAiB;AAEtB,QAAI,KAAK,WAAW,KAAK,UAAU,QAAQ,KAAK,MAAM,CAAA,MAAK,KAAK,UAAU,IAAI,CAAC,CAAC,EAAG;AACnF,SAAK,UAAU,MAAA;AACf,eAAW,OAAO,KAAM,MAAK,UAAU,IAAI,GAAG;AAC9C,SAAK,SAAA;AAAA,EACP;AAAA;AAAA,EAGA,QAAc;AACZ,QAAI,KAAK,UAAU,SAAS,EAAG;AAC/B,SAAK,UAAU,MAAA;AACf,SAAK,SAAA;AAAA,EACP;AAAA;AAAA;AAAA,EAKA,aAAgB,OAAY,OAA4B;AACtD,WAAO,MAAM,OAAO,CAAA,SAAQ,KAAK,UAAU,IAAI,MAAM,IAAI,CAAC,CAAC;AAAA,EAC7D;AAAA;AAAA,EAIQ,WAAiB;AAEvB,SAAK,gBAAgB,IAAI,IAAI,KAAK,SAAS;AAC3C,SAAK,OAAA;AAAA,EACP;AACF;;"}
1
+ {"version":3,"file":"Selection.cjs","names":[],"sources":["../src/Selection.ts"],"sourcesContent":["import { Trackable } from './Trackable';\n\n/**\n * Key-based selection set with toggle and select-all support.\n * Tracks which items are selected by their key (id).\n * Subscribable — auto-tracked when used as a ViewModel property.\n */\nexport class Selection<K extends string | number = string | number> extends Trackable {\n private _selected: Set<K> = new Set();\n private _readonlyView: ReadonlySet<K> = this._selected;\n\n constructor() {\n super();\n }\n\n // ── Readable state ──\n\n /** Read-only view of currently selected keys. */\n get selected(): ReadonlySet<K> {\n return this._readonlyView;\n }\n\n /** Number of currently selected items. */\n get count(): number {\n return this._selected.size;\n }\n\n /** Whether any items are currently selected. */\n get hasSelection(): boolean {\n return this._selected.size > 0;\n }\n\n // ── Query ──\n\n /** Check whether a specific key is selected. */\n isSelected(key: K): boolean {\n return this._selected.has(key);\n }\n\n // ── Actions ──\n\n /** Toggle a key's selection state (select if unselected, deselect if selected). */\n toggle(key: K): void {\n if (this._selected.has(key)) {\n this._selected.delete(key);\n } else {\n this._selected.add(key);\n }\n this._publish();\n }\n\n /** Add one or more keys to the selection. */\n select(...keys: K[]): void {\n let changed = false;\n for (const key of keys) {\n if (!this._selected.has(key)) {\n this._selected.add(key);\n changed = true;\n }\n }\n if (changed) this._publish();\n }\n\n /** Remove one or more keys from the selection. */\n deselect(...keys: K[]): void {\n let changed = false;\n for (const key of keys) {\n if (this._selected.has(key)) {\n this._selected.delete(key);\n changed = true;\n }\n }\n if (changed) this._publish();\n }\n\n /** If all selected → deselect all, else select all. */\n toggleAll(allKeys: K[]): void {\n const allSelected = allKeys.length > 0 && allKeys.every(k => this._selected.has(k));\n if (allSelected) {\n this._selected.clear();\n } else {\n for (const key of allKeys) this._selected.add(key);\n }\n this._publish();\n }\n\n /** Replace the entire selection atomically. Single notification. */\n set(...keys: K[]): void {\n // Check if anything actually changed\n if (keys.length === this._selected.size && keys.every(k => this._selected.has(k))) return;\n this._selected.clear();\n for (const key of keys) this._selected.add(key);\n this._publish();\n }\n\n /** Remove all items from the selection. */\n clear(): void {\n if (this._selected.size === 0) return;\n this._selected.clear();\n this._publish();\n }\n\n // ── Utility ──\n\n /** Filter an array to only items whose key is in the selection. */\n selectedFrom<T>(items: T[], keyOf: (item: T) => K): T[] {\n return items.filter(item => this._selected.has(keyOf(item)));\n }\n\n // ── Internal ──\n\n private _publish(): void {\n // Replace readonlyView so reference equality changes (needed for React)\n this._readonlyView = new Set(this._selected);\n this.notify();\n }\n}\n"],"mappings":";;;;;;;AAOA,IAAa,YAAb,cAA4E,kBAAA,UAAU;CACpF,4BAA4B,IAAI,KAAK;CACrC,gBAAwC,KAAK;CAE7C,cAAc;AACZ,SAAO;;;CAMT,IAAI,WAA2B;AAC7B,SAAO,KAAK;;;CAId,IAAI,QAAgB;AAClB,SAAO,KAAK,UAAU;;;CAIxB,IAAI,eAAwB;AAC1B,SAAO,KAAK,UAAU,OAAO;;;CAM/B,WAAW,KAAiB;AAC1B,SAAO,KAAK,UAAU,IAAI,IAAI;;;CAMhC,OAAO,KAAc;AACnB,MAAI,KAAK,UAAU,IAAI,IAAI,CACzB,MAAK,UAAU,OAAO,IAAI;MAE1B,MAAK,UAAU,IAAI,IAAI;AAEzB,OAAK,UAAU;;;CAIjB,OAAO,GAAG,MAAiB;EACzB,IAAI,UAAU;AACd,OAAK,MAAM,OAAO,KAChB,KAAI,CAAC,KAAK,UAAU,IAAI,IAAI,EAAE;AAC5B,QAAK,UAAU,IAAI,IAAI;AACvB,aAAU;;AAGd,MAAI,QAAS,MAAK,UAAU;;;CAI9B,SAAS,GAAG,MAAiB;EAC3B,IAAI,UAAU;AACd,OAAK,MAAM,OAAO,KAChB,KAAI,KAAK,UAAU,IAAI,IAAI,EAAE;AAC3B,QAAK,UAAU,OAAO,IAAI;AAC1B,aAAU;;AAGd,MAAI,QAAS,MAAK,UAAU;;;CAI9B,UAAU,SAAoB;AAE5B,MADoB,QAAQ,SAAS,KAAK,QAAQ,OAAM,MAAK,KAAK,UAAU,IAAI,EAAE,CAAC,CAEjF,MAAK,UAAU,OAAO;MAEtB,MAAK,MAAM,OAAO,QAAS,MAAK,UAAU,IAAI,IAAI;AAEpD,OAAK,UAAU;;;CAIjB,IAAI,GAAG,MAAiB;AAEtB,MAAI,KAAK,WAAW,KAAK,UAAU,QAAQ,KAAK,OAAM,MAAK,KAAK,UAAU,IAAI,EAAE,CAAC,CAAE;AACnF,OAAK,UAAU,OAAO;AACtB,OAAK,MAAM,OAAO,KAAM,MAAK,UAAU,IAAI,IAAI;AAC/C,OAAK,UAAU;;;CAIjB,QAAc;AACZ,MAAI,KAAK,UAAU,SAAS,EAAG;AAC/B,OAAK,UAAU,OAAO;AACtB,OAAK,UAAU;;;CAMjB,aAAgB,OAAY,OAA4B;AACtD,SAAO,MAAM,QAAO,SAAQ,KAAK,UAAU,IAAI,MAAM,KAAK,CAAC,CAAC;;CAK9D,WAAyB;AAEvB,OAAK,gBAAgB,IAAI,IAAI,KAAK,UAAU;AAC5C,OAAK,QAAQ"}