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,364 @@
|
|
|
1
|
+
import { describe, it, expect, vi } from "vitest";
|
|
2
|
+
import { emitter } from "./emitter";
|
|
3
|
+
|
|
4
|
+
describe("emitter", () => {
|
|
5
|
+
describe("basic functionality", () => {
|
|
6
|
+
it("should emit to listeners in order", () => {
|
|
7
|
+
const e = emitter<number>();
|
|
8
|
+
const order: number[] = [];
|
|
9
|
+
|
|
10
|
+
e.on(() => order.push(1));
|
|
11
|
+
e.on(() => order.push(2));
|
|
12
|
+
e.on(() => order.push(3));
|
|
13
|
+
|
|
14
|
+
e.emit(42);
|
|
15
|
+
|
|
16
|
+
expect(order).toEqual([1, 2, 3]);
|
|
17
|
+
});
|
|
18
|
+
|
|
19
|
+
it("should support array of listeners in on()", () => {
|
|
20
|
+
const e = emitter<number>();
|
|
21
|
+
const calls: number[] = [];
|
|
22
|
+
|
|
23
|
+
e.on([
|
|
24
|
+
(n) => calls.push(n),
|
|
25
|
+
(n) => calls.push(n * 2),
|
|
26
|
+
(n) => calls.push(n * 3),
|
|
27
|
+
]);
|
|
28
|
+
|
|
29
|
+
e.emit(10);
|
|
30
|
+
|
|
31
|
+
expect(calls).toEqual([10, 20, 30]);
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
it("should return size of listeners", () => {
|
|
35
|
+
const e = emitter<number>();
|
|
36
|
+
|
|
37
|
+
expect(e.size()).toBe(0);
|
|
38
|
+
|
|
39
|
+
e.on(() => {});
|
|
40
|
+
expect(e.size()).toBe(1);
|
|
41
|
+
|
|
42
|
+
e.on(() => {});
|
|
43
|
+
expect(e.size()).toBe(2);
|
|
44
|
+
});
|
|
45
|
+
|
|
46
|
+
it("should unsubscribe listener", () => {
|
|
47
|
+
const e = emitter<number>();
|
|
48
|
+
const listener = vi.fn();
|
|
49
|
+
|
|
50
|
+
const unsub = e.on(listener);
|
|
51
|
+
e.emit(1);
|
|
52
|
+
expect(listener).toHaveBeenCalledTimes(1);
|
|
53
|
+
|
|
54
|
+
unsub();
|
|
55
|
+
e.emit(2);
|
|
56
|
+
expect(listener).toHaveBeenCalledTimes(1);
|
|
57
|
+
});
|
|
58
|
+
|
|
59
|
+
it("should support initial listeners", () => {
|
|
60
|
+
const listener1 = vi.fn();
|
|
61
|
+
const listener2 = vi.fn();
|
|
62
|
+
|
|
63
|
+
const e = emitter<number>([listener1, listener2]);
|
|
64
|
+
e.emit(42);
|
|
65
|
+
|
|
66
|
+
expect(listener1).toHaveBeenCalledWith(42);
|
|
67
|
+
expect(listener2).toHaveBeenCalledWith(42);
|
|
68
|
+
});
|
|
69
|
+
});
|
|
70
|
+
|
|
71
|
+
describe("on() with map function", () => {
|
|
72
|
+
it("should transform values with map function", () => {
|
|
73
|
+
const e = emitter<{ type: string; data: number }>();
|
|
74
|
+
const listener = vi.fn();
|
|
75
|
+
|
|
76
|
+
e.on((event) => ({ value: event.data * 2 }), listener);
|
|
77
|
+
|
|
78
|
+
e.emit({ type: "test", data: 5 });
|
|
79
|
+
|
|
80
|
+
expect(listener).toHaveBeenCalledWith(10);
|
|
81
|
+
});
|
|
82
|
+
|
|
83
|
+
it("should filter events when map returns undefined", () => {
|
|
84
|
+
const e = emitter<{ type: string; data: number }>();
|
|
85
|
+
const listener = vi.fn();
|
|
86
|
+
|
|
87
|
+
e.on(
|
|
88
|
+
(event) =>
|
|
89
|
+
event.type === "success" ? { value: event.data } : undefined,
|
|
90
|
+
listener
|
|
91
|
+
);
|
|
92
|
+
|
|
93
|
+
e.emit({ type: "error", data: 1 });
|
|
94
|
+
expect(listener).not.toHaveBeenCalled();
|
|
95
|
+
|
|
96
|
+
e.emit({ type: "success", data: 2 });
|
|
97
|
+
expect(listener).toHaveBeenCalledWith(2);
|
|
98
|
+
});
|
|
99
|
+
|
|
100
|
+
it("should support array of listeners with map function", () => {
|
|
101
|
+
const e = emitter<number>();
|
|
102
|
+
const listener1 = vi.fn();
|
|
103
|
+
const listener2 = vi.fn();
|
|
104
|
+
|
|
105
|
+
e.on((n) => ({ value: n * 10 }), [listener1, listener2]);
|
|
106
|
+
|
|
107
|
+
e.emit(5);
|
|
108
|
+
|
|
109
|
+
expect(listener1).toHaveBeenCalledWith(50);
|
|
110
|
+
expect(listener2).toHaveBeenCalledWith(50);
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
it("should unsubscribe mapped listeners correctly", () => {
|
|
114
|
+
const e = emitter<number>();
|
|
115
|
+
const listener = vi.fn();
|
|
116
|
+
|
|
117
|
+
const unsub = e.on((n) => ({ value: n }), listener);
|
|
118
|
+
|
|
119
|
+
e.emit(1);
|
|
120
|
+
expect(listener).toHaveBeenCalledTimes(1);
|
|
121
|
+
|
|
122
|
+
unsub();
|
|
123
|
+
|
|
124
|
+
e.emit(2);
|
|
125
|
+
expect(listener).toHaveBeenCalledTimes(1);
|
|
126
|
+
});
|
|
127
|
+
});
|
|
128
|
+
|
|
129
|
+
describe("emitLifo", () => {
|
|
130
|
+
it("should emit in reverse order (LIFO)", () => {
|
|
131
|
+
const e = emitter<number>();
|
|
132
|
+
const order: number[] = [];
|
|
133
|
+
|
|
134
|
+
e.on(() => order.push(1));
|
|
135
|
+
e.on(() => order.push(2));
|
|
136
|
+
e.on(() => order.push(3));
|
|
137
|
+
|
|
138
|
+
e.emitLifo(42);
|
|
139
|
+
|
|
140
|
+
expect(order).toEqual([3, 2, 1]);
|
|
141
|
+
});
|
|
142
|
+
|
|
143
|
+
it("should be no-op when settled", () => {
|
|
144
|
+
const e = emitter<number>();
|
|
145
|
+
const listener = vi.fn();
|
|
146
|
+
|
|
147
|
+
e.on(listener);
|
|
148
|
+
e.settle(1);
|
|
149
|
+
listener.mockClear();
|
|
150
|
+
|
|
151
|
+
e.emitLifo(2);
|
|
152
|
+
|
|
153
|
+
expect(listener).not.toHaveBeenCalled();
|
|
154
|
+
});
|
|
155
|
+
});
|
|
156
|
+
|
|
157
|
+
describe("clear", () => {
|
|
158
|
+
it("should remove all listeners", () => {
|
|
159
|
+
const e = emitter<number>();
|
|
160
|
+
const listener1 = vi.fn();
|
|
161
|
+
const listener2 = vi.fn();
|
|
162
|
+
|
|
163
|
+
e.on(listener1);
|
|
164
|
+
e.on(listener2);
|
|
165
|
+
|
|
166
|
+
expect(e.size()).toBe(2);
|
|
167
|
+
|
|
168
|
+
e.clear();
|
|
169
|
+
|
|
170
|
+
expect(e.size()).toBe(0);
|
|
171
|
+
e.emit(1);
|
|
172
|
+
expect(listener1).not.toHaveBeenCalled();
|
|
173
|
+
expect(listener2).not.toHaveBeenCalled();
|
|
174
|
+
});
|
|
175
|
+
});
|
|
176
|
+
|
|
177
|
+
describe("emitAndClear", () => {
|
|
178
|
+
it("should emit to all listeners then clear", () => {
|
|
179
|
+
const e = emitter<number>();
|
|
180
|
+
const listener1 = vi.fn();
|
|
181
|
+
const listener2 = vi.fn();
|
|
182
|
+
|
|
183
|
+
e.on(listener1);
|
|
184
|
+
e.on(listener2);
|
|
185
|
+
|
|
186
|
+
e.emitAndClear(42);
|
|
187
|
+
|
|
188
|
+
expect(listener1).toHaveBeenCalledTimes(1);
|
|
189
|
+
expect(listener2).toHaveBeenCalledTimes(1);
|
|
190
|
+
expect(listener1).toHaveBeenCalledWith(42);
|
|
191
|
+
expect(listener2).toHaveBeenCalledWith(42);
|
|
192
|
+
expect(e.size()).toBe(0);
|
|
193
|
+
});
|
|
194
|
+
|
|
195
|
+
it("should be no-op when settled", () => {
|
|
196
|
+
const e = emitter<number>();
|
|
197
|
+
const listener = vi.fn();
|
|
198
|
+
|
|
199
|
+
e.on(listener);
|
|
200
|
+
e.settle(1);
|
|
201
|
+
listener.mockClear();
|
|
202
|
+
|
|
203
|
+
e.emitAndClear(2);
|
|
204
|
+
|
|
205
|
+
expect(listener).not.toHaveBeenCalled();
|
|
206
|
+
});
|
|
207
|
+
});
|
|
208
|
+
|
|
209
|
+
describe("emitAndClearLifo", () => {
|
|
210
|
+
it("should emit in reverse order then clear", () => {
|
|
211
|
+
const e = emitter<number>();
|
|
212
|
+
const order: number[] = [];
|
|
213
|
+
|
|
214
|
+
e.on(() => order.push(1));
|
|
215
|
+
e.on(() => order.push(2));
|
|
216
|
+
e.on(() => order.push(3));
|
|
217
|
+
|
|
218
|
+
e.emitAndClearLifo(42);
|
|
219
|
+
|
|
220
|
+
expect(order).toEqual([3, 2, 1]);
|
|
221
|
+
expect(e.size()).toBe(0);
|
|
222
|
+
});
|
|
223
|
+
|
|
224
|
+
it("should be no-op when settled", () => {
|
|
225
|
+
const e = emitter<number>();
|
|
226
|
+
const listener = vi.fn();
|
|
227
|
+
|
|
228
|
+
e.on(listener);
|
|
229
|
+
e.settle(1);
|
|
230
|
+
listener.mockClear();
|
|
231
|
+
|
|
232
|
+
e.emitAndClearLifo(2);
|
|
233
|
+
|
|
234
|
+
expect(listener).not.toHaveBeenCalled();
|
|
235
|
+
});
|
|
236
|
+
});
|
|
237
|
+
|
|
238
|
+
describe("settle", () => {
|
|
239
|
+
it("should emit to all listeners and clear", () => {
|
|
240
|
+
const e = emitter<number>();
|
|
241
|
+
const listener = vi.fn();
|
|
242
|
+
|
|
243
|
+
e.on(listener);
|
|
244
|
+
|
|
245
|
+
e.settle(99);
|
|
246
|
+
|
|
247
|
+
expect(listener).toHaveBeenCalledWith(99);
|
|
248
|
+
expect(e.size()).toBe(0);
|
|
249
|
+
});
|
|
250
|
+
|
|
251
|
+
it("should mark emitter as settled", () => {
|
|
252
|
+
const e = emitter<number>();
|
|
253
|
+
|
|
254
|
+
expect(e.settled()).toBe(false);
|
|
255
|
+
|
|
256
|
+
e.settle(1);
|
|
257
|
+
|
|
258
|
+
expect(e.settled()).toBe(true);
|
|
259
|
+
});
|
|
260
|
+
|
|
261
|
+
it("should be no-op if already settled", () => {
|
|
262
|
+
const e = emitter<number>();
|
|
263
|
+
const listener = vi.fn();
|
|
264
|
+
|
|
265
|
+
e.settle(1);
|
|
266
|
+
|
|
267
|
+
e.on(listener);
|
|
268
|
+
listener.mockClear();
|
|
269
|
+
|
|
270
|
+
e.settle(2);
|
|
271
|
+
|
|
272
|
+
expect(listener).not.toHaveBeenCalled();
|
|
273
|
+
});
|
|
274
|
+
|
|
275
|
+
it("should call new subscribers immediately with settled value", () => {
|
|
276
|
+
const e = emitter<number>();
|
|
277
|
+
|
|
278
|
+
e.settle(42);
|
|
279
|
+
|
|
280
|
+
const listener = vi.fn();
|
|
281
|
+
const unsub = e.on(listener);
|
|
282
|
+
|
|
283
|
+
expect(listener).toHaveBeenCalledWith(42);
|
|
284
|
+
expect(unsub).toBeDefined();
|
|
285
|
+
// Unsub should be no-op for settled emitters
|
|
286
|
+
unsub();
|
|
287
|
+
});
|
|
288
|
+
|
|
289
|
+
it("should call new mapped subscribers immediately with settled value", () => {
|
|
290
|
+
const e = emitter<number>();
|
|
291
|
+
|
|
292
|
+
e.settle(10);
|
|
293
|
+
|
|
294
|
+
const listener = vi.fn();
|
|
295
|
+
e.on((n) => ({ value: n * 2 }), listener);
|
|
296
|
+
|
|
297
|
+
expect(listener).toHaveBeenCalledWith(20);
|
|
298
|
+
});
|
|
299
|
+
|
|
300
|
+
it("should call multiple new subscribers immediately when settled", () => {
|
|
301
|
+
const e = emitter<number>();
|
|
302
|
+
|
|
303
|
+
e.settle(5);
|
|
304
|
+
|
|
305
|
+
const listener1 = vi.fn();
|
|
306
|
+
const listener2 = vi.fn();
|
|
307
|
+
e.on([listener1, listener2]);
|
|
308
|
+
|
|
309
|
+
expect(listener1).toHaveBeenCalledWith(5);
|
|
310
|
+
expect(listener2).toHaveBeenCalledWith(5);
|
|
311
|
+
});
|
|
312
|
+
});
|
|
313
|
+
|
|
314
|
+
describe("emit when settled", () => {
|
|
315
|
+
it("should be no-op", () => {
|
|
316
|
+
const e = emitter<number>();
|
|
317
|
+
const listener = vi.fn();
|
|
318
|
+
|
|
319
|
+
e.on(listener);
|
|
320
|
+
e.settle(1);
|
|
321
|
+
listener.mockClear();
|
|
322
|
+
|
|
323
|
+
e.emit(2);
|
|
324
|
+
|
|
325
|
+
expect(listener).not.toHaveBeenCalled();
|
|
326
|
+
});
|
|
327
|
+
});
|
|
328
|
+
|
|
329
|
+
describe("edge cases", () => {
|
|
330
|
+
it("should handle emit with no listeners", () => {
|
|
331
|
+
const e = emitter<number>();
|
|
332
|
+
|
|
333
|
+
// Should not throw
|
|
334
|
+
expect(() => e.emit(1)).not.toThrow();
|
|
335
|
+
expect(() => e.emitLifo(1)).not.toThrow();
|
|
336
|
+
expect(() => e.emitAndClear(1)).not.toThrow();
|
|
337
|
+
expect(() => e.emitAndClearLifo(1)).not.toThrow();
|
|
338
|
+
expect(() => e.settle(1)).not.toThrow();
|
|
339
|
+
});
|
|
340
|
+
|
|
341
|
+
it("should handle void emitter", () => {
|
|
342
|
+
const e = emitter();
|
|
343
|
+
const listener = vi.fn();
|
|
344
|
+
|
|
345
|
+
e.on(listener);
|
|
346
|
+
e.emit();
|
|
347
|
+
|
|
348
|
+
expect(listener).toHaveBeenCalledWith(undefined);
|
|
349
|
+
});
|
|
350
|
+
|
|
351
|
+
it("should pass on() as a callback without losing context", () => {
|
|
352
|
+
const e = emitter<number>();
|
|
353
|
+
const listener = vi.fn();
|
|
354
|
+
|
|
355
|
+
// Use the bound 'on' method
|
|
356
|
+
const { on } = e;
|
|
357
|
+
on(listener);
|
|
358
|
+
|
|
359
|
+
e.emit(42);
|
|
360
|
+
|
|
361
|
+
expect(listener).toHaveBeenCalledWith(42);
|
|
362
|
+
});
|
|
363
|
+
});
|
|
364
|
+
});
|