atomirx 0.0.7 → 0.1.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 (138) hide show
  1. package/README.md +198 -2234
  2. package/bin/cli.js +90 -0
  3. package/dist/core/derived.d.ts +2 -2
  4. package/dist/core/effect.d.ts +3 -2
  5. package/dist/core/onCreateHook.d.ts +15 -2
  6. package/dist/core/onErrorHook.d.ts +4 -1
  7. package/dist/core/pool.d.ts +78 -0
  8. package/dist/core/pool.test.d.ts +1 -0
  9. package/dist/core/select-boolean.test.d.ts +1 -0
  10. package/dist/core/select-pool.test.d.ts +1 -0
  11. package/dist/core/select.d.ts +278 -86
  12. package/dist/core/types.d.ts +233 -1
  13. package/dist/core/withAbort.d.ts +95 -0
  14. package/dist/core/withReady.d.ts +3 -3
  15. package/dist/devtools/constants.d.ts +41 -0
  16. package/dist/devtools/index.cjs +1 -0
  17. package/dist/devtools/index.d.ts +29 -0
  18. package/dist/devtools/index.js +429 -0
  19. package/dist/devtools/registry.d.ts +98 -0
  20. package/dist/devtools/registry.test.d.ts +1 -0
  21. package/dist/devtools/setup.d.ts +61 -0
  22. package/dist/devtools/types.d.ts +311 -0
  23. package/dist/index-BZEnfIcB.cjs +1 -0
  24. package/dist/index-BbPZhsDl.js +1653 -0
  25. package/dist/index.cjs +1 -1
  26. package/dist/index.d.ts +4 -3
  27. package/dist/index.js +18 -14
  28. package/dist/onDispatchHook-C8yLzr-o.cjs +1 -0
  29. package/dist/onDispatchHook-SKbiIUaJ.js +5 -0
  30. package/dist/onErrorHook-BGGy3tqK.js +38 -0
  31. package/dist/onErrorHook-DHBASmYw.cjs +1 -0
  32. package/dist/react/index.cjs +1 -30
  33. package/dist/react/index.js +206 -791
  34. package/dist/react/onDispatchHook.d.ts +106 -0
  35. package/dist/react/useAction.d.ts +4 -1
  36. package/dist/react-devtools/DevToolsPanel.d.ts +93 -0
  37. package/dist/react-devtools/EntityDetails.d.ts +10 -0
  38. package/dist/react-devtools/EntityList.d.ts +15 -0
  39. package/dist/react-devtools/LogList.d.ts +12 -0
  40. package/dist/react-devtools/hooks.d.ts +50 -0
  41. package/dist/react-devtools/index.cjs +1 -0
  42. package/dist/react-devtools/index.d.ts +31 -0
  43. package/dist/react-devtools/index.js +1589 -0
  44. package/dist/react-devtools/styles.d.ts +148 -0
  45. package/package.json +26 -2
  46. package/skills/atomirx/SKILL.md +456 -0
  47. package/skills/atomirx/references/async-patterns.md +188 -0
  48. package/skills/atomirx/references/atom-patterns.md +238 -0
  49. package/skills/atomirx/references/deferred-loading.md +191 -0
  50. package/skills/atomirx/references/derived-patterns.md +428 -0
  51. package/skills/atomirx/references/effect-patterns.md +426 -0
  52. package/skills/atomirx/references/error-handling.md +140 -0
  53. package/skills/atomirx/references/hooks.md +322 -0
  54. package/skills/atomirx/references/pool-patterns.md +229 -0
  55. package/skills/atomirx/references/react-integration.md +411 -0
  56. package/skills/atomirx/references/rules.md +407 -0
  57. package/skills/atomirx/references/select-context.md +309 -0
  58. package/skills/atomirx/references/service-template.md +172 -0
  59. package/skills/atomirx/references/store-template.md +205 -0
  60. package/skills/atomirx/references/testing-patterns.md +431 -0
  61. package/coverage/base.css +0 -224
  62. package/coverage/block-navigation.js +0 -87
  63. package/coverage/clover.xml +0 -1440
  64. package/coverage/coverage-final.json +0 -14
  65. package/coverage/favicon.png +0 -0
  66. package/coverage/index.html +0 -131
  67. package/coverage/prettify.css +0 -1
  68. package/coverage/prettify.js +0 -2
  69. package/coverage/sort-arrow-sprite.png +0 -0
  70. package/coverage/sorter.js +0 -210
  71. package/coverage/src/core/atom.ts.html +0 -889
  72. package/coverage/src/core/batch.ts.html +0 -223
  73. package/coverage/src/core/define.ts.html +0 -805
  74. package/coverage/src/core/emitter.ts.html +0 -919
  75. package/coverage/src/core/equality.ts.html +0 -631
  76. package/coverage/src/core/hook.ts.html +0 -460
  77. package/coverage/src/core/index.html +0 -281
  78. package/coverage/src/core/isAtom.ts.html +0 -100
  79. package/coverage/src/core/isPromiseLike.ts.html +0 -133
  80. package/coverage/src/core/onCreateHook.ts.html +0 -138
  81. package/coverage/src/core/scheduleNotifyHook.ts.html +0 -94
  82. package/coverage/src/core/types.ts.html +0 -523
  83. package/coverage/src/core/withUse.ts.html +0 -253
  84. package/coverage/src/index.html +0 -116
  85. package/coverage/src/index.ts.html +0 -106
  86. package/dist/index-CBVj1kSj.js +0 -1350
  87. package/dist/index-Cxk9v0um.cjs +0 -1
  88. package/scripts/publish.js +0 -198
  89. package/src/core/atom.test.ts +0 -633
  90. package/src/core/atom.ts +0 -311
  91. package/src/core/atomState.test.ts +0 -342
  92. package/src/core/atomState.ts +0 -256
  93. package/src/core/batch.test.ts +0 -257
  94. package/src/core/batch.ts +0 -172
  95. package/src/core/define.test.ts +0 -343
  96. package/src/core/define.ts +0 -243
  97. package/src/core/derived.test.ts +0 -1215
  98. package/src/core/derived.ts +0 -450
  99. package/src/core/effect.test.ts +0 -802
  100. package/src/core/effect.ts +0 -188
  101. package/src/core/emitter.test.ts +0 -364
  102. package/src/core/emitter.ts +0 -392
  103. package/src/core/equality.test.ts +0 -392
  104. package/src/core/equality.ts +0 -182
  105. package/src/core/getAtomState.ts +0 -69
  106. package/src/core/hook.test.ts +0 -227
  107. package/src/core/hook.ts +0 -177
  108. package/src/core/isAtom.ts +0 -27
  109. package/src/core/isPromiseLike.test.ts +0 -72
  110. package/src/core/isPromiseLike.ts +0 -16
  111. package/src/core/onCreateHook.ts +0 -107
  112. package/src/core/onErrorHook.test.ts +0 -350
  113. package/src/core/onErrorHook.ts +0 -52
  114. package/src/core/promiseCache.test.ts +0 -241
  115. package/src/core/promiseCache.ts +0 -284
  116. package/src/core/scheduleNotifyHook.ts +0 -53
  117. package/src/core/select.ts +0 -729
  118. package/src/core/selector.test.ts +0 -799
  119. package/src/core/types.ts +0 -389
  120. package/src/core/withReady.test.ts +0 -534
  121. package/src/core/withReady.ts +0 -191
  122. package/src/core/withUse.test.ts +0 -249
  123. package/src/core/withUse.ts +0 -56
  124. package/src/index.test.ts +0 -80
  125. package/src/index.ts +0 -65
  126. package/src/react/index.ts +0 -21
  127. package/src/react/rx.test.tsx +0 -571
  128. package/src/react/rx.tsx +0 -531
  129. package/src/react/strictModeTest.tsx +0 -71
  130. package/src/react/useAction.test.ts +0 -987
  131. package/src/react/useAction.ts +0 -607
  132. package/src/react/useSelector.test.ts +0 -182
  133. package/src/react/useSelector.ts +0 -292
  134. package/src/react/useStable.test.ts +0 -553
  135. package/src/react/useStable.ts +0 -288
  136. package/tsconfig.json +0 -9
  137. package/v2.md +0 -725
  138. package/vite.config.ts +0 -39
@@ -1,392 +0,0 @@
1
- import { Listener, SingleOrMultipleListeners } from "./types";
2
-
3
- /**
4
- * Event emitter interface for pub/sub pattern.
5
- *
6
- * @template T - The type of payload emitted to listeners (defaults to void)
7
- */
8
- export interface Emitter<T = void> {
9
- /**
10
- * Subscribe to events with one or more listeners.
11
- *
12
- * @param listeners - Single listener or array of listeners
13
- * @returns Unsubscribe function (idempotent - safe to call multiple times)
14
- */
15
- on(listeners: SingleOrMultipleListeners<T>): VoidFunction;
16
-
17
- /**
18
- * Subscribe with a mapping function that filters and transforms events.
19
- *
20
- * The map function receives the emitted value and returns either:
21
- * - `{ value: TValue }` - Listener is called with the transformed value
22
- * - `undefined` - Listener is NOT called (event filtered out)
23
- *
24
- * @template TValue - The transformed value type passed to listeners
25
- * @param map - Transform function that can filter (return undefined) or map values
26
- * @param listeners - Single listener or array of listeners for transformed values
27
- * @returns Unsubscribe function
28
- *
29
- * @example Filter and transform
30
- * ```ts
31
- * const emitter = emitter<{ type: string; data: number }>();
32
- *
33
- * // Only listen to 'success' events, extract just the data
34
- * emitter.on(
35
- * (event) => event.type === 'success' ? { value: event.data } : undefined,
36
- * (data) => console.log('Success data:', data)
37
- * );
38
- * ```
39
- */
40
- on<TValue>(
41
- map: (value: T) => { value: TValue } | undefined,
42
- listeners: SingleOrMultipleListeners<TValue>
43
- ): VoidFunction;
44
-
45
- /**
46
- * Emit an event to all registered listeners.
47
- *
48
- * @param payload - The value to pass to all listeners
49
- */
50
- emit(payload: T): void;
51
-
52
- /**
53
- * Emit an event to all registered listeners in LIFO (reverse) order.
54
- * Useful for cleanup scenarios where resources should be released
55
- * in reverse order of acquisition.
56
- *
57
- * @param payload - The value to pass to all listeners
58
- */
59
- emitLifo(payload: T): void;
60
-
61
- /**
62
- * Remove all registered listeners.
63
- */
64
- clear(): void;
65
-
66
- /**
67
- * Emit an event to all listeners, then clear all listeners.
68
- * Useful for one-time events like disposal.
69
- *
70
- * @param payload - The value to pass to all listeners
71
- */
72
- emitAndClear(payload: T): void;
73
-
74
- /**
75
- * Emit an event to all listeners in LIFO (reverse) order, then clear.
76
- * Useful for cleanup scenarios where resources should be released
77
- * in reverse order of acquisition.
78
- *
79
- * @param payload - The value to pass to all listeners
80
- */
81
- emitAndClearLifo(payload: T): void;
82
-
83
- /**
84
- * Emit to all listeners, clear, and "settle" the emitter.
85
- *
86
- * After settling:
87
- * - Any new `on()` call immediately invokes the listener with the settled payload
88
- * - Returns a no-op unsubscribe function
89
- * - `emit()` and `emitAndClear()` become no-ops
90
- *
91
- * Useful for one-time events where late subscribers should still receive the value
92
- * (similar to Promise behavior).
93
- *
94
- * @param payload - The final value to pass to all listeners
95
- */
96
- settle(payload: T): void;
97
-
98
- /** Number of registered listeners */
99
- size(): number;
100
-
101
- /** Whether the emitter has been settled */
102
- settled(): boolean;
103
-
104
- /**
105
- * Iterate over all registered listeners.
106
- * Used for batching to dedupe listeners across multiple atoms.
107
- */
108
- forEach(callback: (listener: Listener<T>) => void): void;
109
- }
110
-
111
- const noop = () => {};
112
-
113
- /**
114
- * Class-based emitter implementation for better V8 optimization.
115
- * All instances share methods via prototype.
116
- */
117
- class EmitterImpl<T = void> implements Emitter<T> {
118
- /** Set of registered listeners */
119
- private _listeners: Set<Listener<T>>;
120
- /** Settled payload (if settled) */
121
- private _settledPayload: T | undefined = undefined;
122
- /** Whether the emitter has been settled */
123
- private _isSettled = false;
124
-
125
- constructor(initialListeners?: Listener<T>[]) {
126
- this._listeners = new Set<Listener<T>>(initialListeners);
127
- // Bind 'on' to preserve 'this' context when passed as callback
128
- }
129
-
130
- size = (): number => {
131
- return this._listeners.size;
132
- };
133
-
134
- settled = (): boolean => {
135
- return this._isSettled;
136
- };
137
-
138
- forEach = (callback: (listener: Listener<T>) => void): void => {
139
- this._listeners.forEach(callback);
140
- };
141
-
142
- on = (listenersOrMap: any, mappedListeners?: any): VoidFunction => {
143
- let newListeners: Listener<T>[];
144
-
145
- if (mappedListeners === undefined) {
146
- // Simple form: on(listeners)
147
- newListeners = Array.isArray(listenersOrMap)
148
- ? listenersOrMap
149
- : [listenersOrMap];
150
- } else {
151
- // Mapped form: on(map, listeners)
152
- const map = listenersOrMap as (value: T) => { value: any } | undefined;
153
- const sourceListeners: Listener<any>[] = Array.isArray(mappedListeners)
154
- ? mappedListeners
155
- : [mappedListeners];
156
-
157
- newListeners = [
158
- (value: T) => {
159
- const mappedValue = map(value);
160
- if (mappedValue) {
161
- for (let i = 0; i < sourceListeners.length; i++) {
162
- sourceListeners[i]!(mappedValue.value);
163
- }
164
- }
165
- },
166
- ];
167
- }
168
-
169
- // If settled, call listeners immediately and return no-op
170
- if (this._isSettled) {
171
- const payload = this._settledPayload as T;
172
- for (let i = 0; i < newListeners.length; i++) {
173
- newListeners[i]!(payload);
174
- }
175
- return noop;
176
- }
177
-
178
- const listeners = this._listeners;
179
- for (let i = 0; i < newListeners.length; i++) {
180
- listeners.add(newListeners[i]!);
181
- }
182
-
183
- return () => {
184
- for (let i = 0; i < newListeners.length; i++) {
185
- listeners.delete(newListeners[i]!);
186
- }
187
- };
188
- };
189
-
190
- emit = (payload: T): void => {
191
- if (this._isSettled) return;
192
- this._doEmit(payload, false, false);
193
- };
194
-
195
- emitLifo = (payload: T): void => {
196
- if (this._isSettled) return;
197
- this._doEmit(payload, false, true);
198
- };
199
-
200
- clear = (): void => {
201
- this._listeners.clear();
202
- };
203
-
204
- emitAndClear = (payload: T): void => {
205
- if (this._isSettled) return;
206
- this._doEmit(payload, true, false);
207
- };
208
-
209
- emitAndClearLifo = (payload: T): void => {
210
- if (this._isSettled) return;
211
- this._doEmit(payload, true, true);
212
- };
213
-
214
- settle = (payload: T): void => {
215
- if (this._isSettled) return;
216
- this._settledPayload = payload;
217
- this._isSettled = true;
218
- this._doEmit(payload, true, false);
219
- };
220
-
221
- /**
222
- * Internal emit implementation.
223
- * Creates snapshot to handle modifications during iteration.
224
- */
225
- private _doEmit = (payload: T, clear: boolean, lifo: boolean): void => {
226
- const listeners = this._listeners;
227
- const size = listeners.size;
228
- if (size === 0) return;
229
-
230
- // Create snapshot - necessary because Set.forEach includes items added during iteration
231
- const copy = Array.from(listeners);
232
- if (clear) {
233
- listeners.clear();
234
- }
235
-
236
- // Use traditional for loop for maximum performance
237
- if (lifo) {
238
- for (let i = size - 1; i >= 0; i--) {
239
- copy[i]!(payload);
240
- }
241
- } else {
242
- for (let i = 0; i < size; i++) {
243
- copy[i]!(payload);
244
- }
245
- }
246
- };
247
- }
248
-
249
- /**
250
- * Creates an event emitter for managing and notifying listeners.
251
- *
252
- * An emitter provides a simple pub/sub pattern for managing event listeners.
253
- * It's used internally by atoms and effects to manage subscriptions and notifications.
254
- *
255
- * ## Key Features
256
- *
257
- * 1. **Subscribe/Unsubscribe**: Add listeners that will be notified when events are emitted
258
- * 2. **Emit events**: Notify all registered listeners with a payload
259
- * 3. **Mapped subscriptions**: Filter and transform events before they reach listeners
260
- * 4. **LIFO emission**: Emit in reverse order for cleanup scenarios
261
- * 5. **Settle pattern**: One-time events where late subscribers still receive the value
262
- * 6. **Idempotent unsubscribe**: Safe to call unsubscribe multiple times
263
- *
264
- * ## Emission Methods
265
- *
266
- * | Method | Order | Clears | Settles | Use Case |
267
- * |--------|-------|--------|---------|----------|
268
- * | `emit()` | FIFO | No | No | Normal notifications |
269
- * | `emitLifo()` | LIFO | No | No | Cleanup in reverse order |
270
- * | `emitAndClear()` | FIFO | Yes | No | One-time broadcast |
271
- * | `emitAndClearLifo()` | LIFO | Yes | No | One-time cleanup |
272
- * | `settle()` | FIFO | Yes | Yes | Promise-like one-time events |
273
- *
274
- * ## Settle Behavior
275
- *
276
- * After `settle()` is called:
277
- * - New `on()` calls immediately invoke the listener with the settled payload
278
- * - Returns a no-op unsubscribe function
279
- * - `emit()` and other methods become no-ops
280
- *
281
- * This is similar to how Promises work - late `.then()` calls still receive the value.
282
- *
283
- * @template T - The type of payload emitted to listeners (defaults to void)
284
- * @param initialListeners - Optional array of listeners to start with
285
- * @returns An Emitter instance
286
- *
287
- * @example Basic usage
288
- * ```ts
289
- * const events = emitter<string>();
290
- *
291
- * // Subscribe
292
- * const unsubscribe = events.on((message) => {
293
- * console.log('Received:', message);
294
- * });
295
- *
296
- * // Emit
297
- * events.emit('Hello'); // Logs: "Received: Hello"
298
- *
299
- * // Unsubscribe
300
- * unsubscribe();
301
- *
302
- * events.emit('World'); // Nothing logged
303
- * ```
304
- *
305
- * @example Multiple listeners
306
- * ```ts
307
- * const events = emitter<number>();
308
- *
309
- * events.on((n) => console.log('A:', n));
310
- * events.on((n) => console.log('B:', n));
311
- *
312
- * events.emit(42);
313
- * // Logs: "A: 42"
314
- * // Logs: "B: 42"
315
- * ```
316
- *
317
- * @example Mapped subscriptions (filter and transform)
318
- * ```ts
319
- * const events = emitter<{ type: string; data: number }>();
320
- *
321
- * // Only listen to 'success' events, extract just the data
322
- * events.on(
323
- * (event) => event.type === 'success' ? { value: event.data } : undefined,
324
- * (data) => console.log('Success data:', data)
325
- * );
326
- *
327
- * events.emit({ type: 'error', data: 0 }); // Nothing logged (filtered)
328
- * events.emit({ type: 'success', data: 42 }); // Logs: "Success data: 42"
329
- * ```
330
- *
331
- * @example LIFO emission for cleanup
332
- * ```ts
333
- * const cleanup = emitter();
334
- *
335
- * cleanup.on(() => console.log('First registered, last to clean'));
336
- * cleanup.on(() => console.log('Second registered, second to clean'));
337
- * cleanup.on(() => console.log('Last registered, first to clean'));
338
- *
339
- * cleanup.emitLifo();
340
- * // Logs in reverse order:
341
- * // "Last registered, first to clean"
342
- * // "Second registered, second to clean"
343
- * // "First registered, last to clean"
344
- * ```
345
- *
346
- * @example Settle pattern (Promise-like)
347
- * ```ts
348
- * const ready = emitter<{ config: Config }>();
349
- *
350
- * // Early subscriber
351
- * ready.on((payload) => console.log('Early:', payload.config));
352
- *
353
- * // Settle the emitter
354
- * ready.settle({ config: loadedConfig });
355
- * // Logs: "Early: ..."
356
- *
357
- * // Late subscriber - still receives the value immediately
358
- * ready.on((payload) => console.log('Late:', payload.config));
359
- * // Logs: "Late: ..." (immediately)
360
- * ```
361
- *
362
- * @example Void emitter (no payload)
363
- * ```ts
364
- * const tick = emitter(); // emitter<void>
365
- *
366
- * tick.on(() => console.log('Tick!'));
367
- *
368
- * tick.emit(); // Logs: "Tick!"
369
- * ```
370
- *
371
- * @example Array of listeners
372
- * ```ts
373
- * const events = emitter<string>();
374
- *
375
- * // Subscribe multiple listeners at once
376
- * const unsubscribe = events.on([
377
- * (msg) => console.log('Logger 1:', msg),
378
- * (msg) => console.log('Logger 2:', msg),
379
- * ]);
380
- *
381
- * events.emit('Hello');
382
- * // Logs: "Logger 1: Hello"
383
- * // Logs: "Logger 2: Hello"
384
- *
385
- * unsubscribe(); // Removes both listeners
386
- * ```
387
- */
388
- export function emitter<T = void>(
389
- initialListeners?: Listener<T>[]
390
- ): Emitter<T> {
391
- return new EmitterImpl<T>(initialListeners);
392
- }