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.
- package/README.md +1666 -0
- package/coverage/base.css +224 -0
- package/coverage/block-navigation.js +87 -0
- package/coverage/clover.xml +1440 -0
- package/coverage/coverage-final.json +14 -0
- package/coverage/favicon.png +0 -0
- package/coverage/index.html +131 -0
- package/coverage/prettify.css +1 -0
- package/coverage/prettify.js +2 -0
- package/coverage/sort-arrow-sprite.png +0 -0
- package/coverage/sorter.js +210 -0
- package/coverage/src/core/atom.ts.html +889 -0
- package/coverage/src/core/batch.ts.html +223 -0
- package/coverage/src/core/define.ts.html +805 -0
- package/coverage/src/core/emitter.ts.html +919 -0
- package/coverage/src/core/equality.ts.html +631 -0
- package/coverage/src/core/hook.ts.html +460 -0
- package/coverage/src/core/index.html +281 -0
- package/coverage/src/core/isAtom.ts.html +100 -0
- package/coverage/src/core/isPromiseLike.ts.html +133 -0
- package/coverage/src/core/onCreateHook.ts.html +136 -0
- package/coverage/src/core/scheduleNotifyHook.ts.html +94 -0
- package/coverage/src/core/types.ts.html +523 -0
- package/coverage/src/core/withUse.ts.html +253 -0
- package/coverage/src/index.html +116 -0
- package/coverage/src/index.ts.html +106 -0
- package/dist/core/atom.d.ts +63 -0
- package/dist/core/atom.test.d.ts +1 -0
- package/dist/core/atomState.d.ts +104 -0
- package/dist/core/atomState.test.d.ts +1 -0
- package/dist/core/batch.d.ts +126 -0
- package/dist/core/batch.test.d.ts +1 -0
- package/dist/core/define.d.ts +173 -0
- package/dist/core/define.test.d.ts +1 -0
- package/dist/core/derived.d.ts +102 -0
- package/dist/core/derived.test.d.ts +1 -0
- package/dist/core/effect.d.ts +120 -0
- package/dist/core/effect.test.d.ts +1 -0
- package/dist/core/emitter.d.ts +237 -0
- package/dist/core/emitter.test.d.ts +1 -0
- package/dist/core/equality.d.ts +62 -0
- package/dist/core/equality.test.d.ts +1 -0
- package/dist/core/hook.d.ts +134 -0
- package/dist/core/hook.test.d.ts +1 -0
- package/dist/core/isAtom.d.ts +9 -0
- package/dist/core/isPromiseLike.d.ts +9 -0
- package/dist/core/isPromiseLike.test.d.ts +1 -0
- package/dist/core/onCreateHook.d.ts +79 -0
- package/dist/core/promiseCache.d.ts +134 -0
- package/dist/core/promiseCache.test.d.ts +1 -0
- package/dist/core/scheduleNotifyHook.d.ts +51 -0
- package/dist/core/select.d.ts +151 -0
- package/dist/core/selector.test.d.ts +1 -0
- package/dist/core/types.d.ts +279 -0
- package/dist/core/withUse.d.ts +38 -0
- package/dist/core/withUse.test.d.ts +1 -0
- package/dist/index-2ok7ilik.js +1217 -0
- package/dist/index-B_5SFzfl.cjs +1 -0
- package/dist/index.cjs +1 -0
- package/dist/index.d.ts +14 -0
- package/dist/index.js +20 -0
- package/dist/index.test.d.ts +1 -0
- package/dist/react/index.cjs +30 -0
- package/dist/react/index.d.ts +7 -0
- package/dist/react/index.js +823 -0
- package/dist/react/rx.d.ts +250 -0
- package/dist/react/rx.test.d.ts +1 -0
- package/dist/react/strictModeTest.d.ts +10 -0
- package/dist/react/useAction.d.ts +381 -0
- package/dist/react/useAction.test.d.ts +1 -0
- package/dist/react/useStable.d.ts +183 -0
- package/dist/react/useStable.test.d.ts +1 -0
- package/dist/react/useValue.d.ts +134 -0
- package/dist/react/useValue.test.d.ts +1 -0
- package/package.json +57 -0
- package/scripts/publish.js +198 -0
- package/src/core/atom.test.ts +369 -0
- package/src/core/atom.ts +189 -0
- package/src/core/atomState.test.ts +342 -0
- package/src/core/atomState.ts +256 -0
- package/src/core/batch.test.ts +257 -0
- package/src/core/batch.ts +172 -0
- package/src/core/define.test.ts +342 -0
- package/src/core/define.ts +243 -0
- package/src/core/derived.test.ts +381 -0
- package/src/core/derived.ts +339 -0
- package/src/core/effect.test.ts +196 -0
- package/src/core/effect.ts +184 -0
- package/src/core/emitter.test.ts +364 -0
- package/src/core/emitter.ts +392 -0
- package/src/core/equality.test.ts +392 -0
- package/src/core/equality.ts +182 -0
- package/src/core/hook.test.ts +227 -0
- package/src/core/hook.ts +177 -0
- package/src/core/isAtom.ts +27 -0
- package/src/core/isPromiseLike.test.ts +72 -0
- package/src/core/isPromiseLike.ts +16 -0
- package/src/core/onCreateHook.ts +92 -0
- package/src/core/promiseCache.test.ts +239 -0
- package/src/core/promiseCache.ts +279 -0
- package/src/core/scheduleNotifyHook.ts +53 -0
- package/src/core/select.ts +454 -0
- package/src/core/selector.test.ts +257 -0
- package/src/core/types.ts +311 -0
- package/src/core/withUse.test.ts +249 -0
- package/src/core/withUse.ts +56 -0
- package/src/index.test.ts +80 -0
- package/src/index.ts +51 -0
- package/src/react/index.ts +20 -0
- package/src/react/rx.test.tsx +416 -0
- package/src/react/rx.tsx +300 -0
- package/src/react/strictModeTest.tsx +71 -0
- package/src/react/useAction.test.ts +989 -0
- package/src/react/useAction.ts +605 -0
- package/src/react/useStable.test.ts +553 -0
- package/src/react/useStable.ts +288 -0
- package/src/react/useValue.test.ts +182 -0
- package/src/react/useValue.ts +261 -0
- package/tsconfig.json +9 -0
- package/v2.md +725 -0
- 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
|
+
}
|