atomirx 0.0.1

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 (121) hide show
  1. package/README.md +1666 -0
  2. package/coverage/base.css +224 -0
  3. package/coverage/block-navigation.js +87 -0
  4. package/coverage/clover.xml +1440 -0
  5. package/coverage/coverage-final.json +14 -0
  6. package/coverage/favicon.png +0 -0
  7. package/coverage/index.html +131 -0
  8. package/coverage/prettify.css +1 -0
  9. package/coverage/prettify.js +2 -0
  10. package/coverage/sort-arrow-sprite.png +0 -0
  11. package/coverage/sorter.js +210 -0
  12. package/coverage/src/core/atom.ts.html +889 -0
  13. package/coverage/src/core/batch.ts.html +223 -0
  14. package/coverage/src/core/define.ts.html +805 -0
  15. package/coverage/src/core/emitter.ts.html +919 -0
  16. package/coverage/src/core/equality.ts.html +631 -0
  17. package/coverage/src/core/hook.ts.html +460 -0
  18. package/coverage/src/core/index.html +281 -0
  19. package/coverage/src/core/isAtom.ts.html +100 -0
  20. package/coverage/src/core/isPromiseLike.ts.html +133 -0
  21. package/coverage/src/core/onCreateHook.ts.html +136 -0
  22. package/coverage/src/core/scheduleNotifyHook.ts.html +94 -0
  23. package/coverage/src/core/types.ts.html +523 -0
  24. package/coverage/src/core/withUse.ts.html +253 -0
  25. package/coverage/src/index.html +116 -0
  26. package/coverage/src/index.ts.html +106 -0
  27. package/dist/core/atom.d.ts +63 -0
  28. package/dist/core/atom.test.d.ts +1 -0
  29. package/dist/core/atomState.d.ts +104 -0
  30. package/dist/core/atomState.test.d.ts +1 -0
  31. package/dist/core/batch.d.ts +126 -0
  32. package/dist/core/batch.test.d.ts +1 -0
  33. package/dist/core/define.d.ts +173 -0
  34. package/dist/core/define.test.d.ts +1 -0
  35. package/dist/core/derived.d.ts +102 -0
  36. package/dist/core/derived.test.d.ts +1 -0
  37. package/dist/core/effect.d.ts +120 -0
  38. package/dist/core/effect.test.d.ts +1 -0
  39. package/dist/core/emitter.d.ts +237 -0
  40. package/dist/core/emitter.test.d.ts +1 -0
  41. package/dist/core/equality.d.ts +62 -0
  42. package/dist/core/equality.test.d.ts +1 -0
  43. package/dist/core/hook.d.ts +134 -0
  44. package/dist/core/hook.test.d.ts +1 -0
  45. package/dist/core/isAtom.d.ts +9 -0
  46. package/dist/core/isPromiseLike.d.ts +9 -0
  47. package/dist/core/isPromiseLike.test.d.ts +1 -0
  48. package/dist/core/onCreateHook.d.ts +79 -0
  49. package/dist/core/promiseCache.d.ts +134 -0
  50. package/dist/core/promiseCache.test.d.ts +1 -0
  51. package/dist/core/scheduleNotifyHook.d.ts +51 -0
  52. package/dist/core/select.d.ts +151 -0
  53. package/dist/core/selector.test.d.ts +1 -0
  54. package/dist/core/types.d.ts +279 -0
  55. package/dist/core/withUse.d.ts +38 -0
  56. package/dist/core/withUse.test.d.ts +1 -0
  57. package/dist/index-2ok7ilik.js +1217 -0
  58. package/dist/index-B_5SFzfl.cjs +1 -0
  59. package/dist/index.cjs +1 -0
  60. package/dist/index.d.ts +14 -0
  61. package/dist/index.js +20 -0
  62. package/dist/index.test.d.ts +1 -0
  63. package/dist/react/index.cjs +30 -0
  64. package/dist/react/index.d.ts +7 -0
  65. package/dist/react/index.js +823 -0
  66. package/dist/react/rx.d.ts +250 -0
  67. package/dist/react/rx.test.d.ts +1 -0
  68. package/dist/react/strictModeTest.d.ts +10 -0
  69. package/dist/react/useAction.d.ts +381 -0
  70. package/dist/react/useAction.test.d.ts +1 -0
  71. package/dist/react/useStable.d.ts +183 -0
  72. package/dist/react/useStable.test.d.ts +1 -0
  73. package/dist/react/useValue.d.ts +134 -0
  74. package/dist/react/useValue.test.d.ts +1 -0
  75. package/package.json +57 -0
  76. package/scripts/publish.js +198 -0
  77. package/src/core/atom.test.ts +369 -0
  78. package/src/core/atom.ts +189 -0
  79. package/src/core/atomState.test.ts +342 -0
  80. package/src/core/atomState.ts +256 -0
  81. package/src/core/batch.test.ts +257 -0
  82. package/src/core/batch.ts +172 -0
  83. package/src/core/define.test.ts +342 -0
  84. package/src/core/define.ts +243 -0
  85. package/src/core/derived.test.ts +381 -0
  86. package/src/core/derived.ts +339 -0
  87. package/src/core/effect.test.ts +196 -0
  88. package/src/core/effect.ts +184 -0
  89. package/src/core/emitter.test.ts +364 -0
  90. package/src/core/emitter.ts +392 -0
  91. package/src/core/equality.test.ts +392 -0
  92. package/src/core/equality.ts +182 -0
  93. package/src/core/hook.test.ts +227 -0
  94. package/src/core/hook.ts +177 -0
  95. package/src/core/isAtom.ts +27 -0
  96. package/src/core/isPromiseLike.test.ts +72 -0
  97. package/src/core/isPromiseLike.ts +16 -0
  98. package/src/core/onCreateHook.ts +92 -0
  99. package/src/core/promiseCache.test.ts +239 -0
  100. package/src/core/promiseCache.ts +279 -0
  101. package/src/core/scheduleNotifyHook.ts +53 -0
  102. package/src/core/select.ts +454 -0
  103. package/src/core/selector.test.ts +257 -0
  104. package/src/core/types.ts +311 -0
  105. package/src/core/withUse.test.ts +249 -0
  106. package/src/core/withUse.ts +56 -0
  107. package/src/index.test.ts +80 -0
  108. package/src/index.ts +51 -0
  109. package/src/react/index.ts +20 -0
  110. package/src/react/rx.test.tsx +416 -0
  111. package/src/react/rx.tsx +300 -0
  112. package/src/react/strictModeTest.tsx +71 -0
  113. package/src/react/useAction.test.ts +989 -0
  114. package/src/react/useAction.ts +605 -0
  115. package/src/react/useStable.test.ts +553 -0
  116. package/src/react/useStable.ts +288 -0
  117. package/src/react/useValue.test.ts +182 -0
  118. package/src/react/useValue.ts +261 -0
  119. package/tsconfig.json +9 -0
  120. package/v2.md +725 -0
  121. package/vite.config.ts +39 -0
@@ -0,0 +1,392 @@
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
+ }