rexfect 0.0.7

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 (190) hide show
  1. package/README.md +1756 -0
  2. package/dist/abortableContext.d.ts +3 -0
  3. package/dist/abortableContext.d.ts.map +1 -0
  4. package/dist/abortableContext.js +48 -0
  5. package/dist/abortableContext.js.map +1 -0
  6. package/dist/action.d.ts +64 -0
  7. package/dist/action.d.ts.map +1 -0
  8. package/dist/action.js +208 -0
  9. package/dist/action.js.map +1 -0
  10. package/dist/action.test.d.ts +2 -0
  11. package/dist/action.test.d.ts.map +1 -0
  12. package/dist/action.test.js +189 -0
  13. package/dist/action.test.js.map +1 -0
  14. package/dist/async/abortable-guard.d.ts +25 -0
  15. package/dist/async/abortable-guard.d.ts.map +1 -0
  16. package/dist/async/abortable-guard.js +33 -0
  17. package/dist/async/abortable-guard.js.map +1 -0
  18. package/dist/async/abortable.d.ts +331 -0
  19. package/dist/async/abortable.d.ts.map +1 -0
  20. package/dist/async/abortable.js +410 -0
  21. package/dist/async/abortable.js.map +1 -0
  22. package/dist/async/abortable.test.d.ts +2 -0
  23. package/dist/async/abortable.test.d.ts.map +1 -0
  24. package/dist/async/abortable.test.js +535 -0
  25. package/dist/async/abortable.test.js.map +1 -0
  26. package/dist/async/abortable.typeCheck.d.ts +8 -0
  27. package/dist/async/abortable.typeCheck.d.ts.map +1 -0
  28. package/dist/async/abortable.typeCheck.js +138 -0
  29. package/dist/async/abortable.typeCheck.js.map +1 -0
  30. package/dist/async/async.d.ts +18 -0
  31. package/dist/async/async.d.ts.map +1 -0
  32. package/dist/async/async.js +20 -0
  33. package/dist/async/async.js.map +1 -0
  34. package/dist/async/index.d.ts +15 -0
  35. package/dist/async/index.d.ts.map +1 -0
  36. package/dist/async/index.js +13 -0
  37. package/dist/async/index.js.map +1 -0
  38. package/dist/async/loadable.d.ts +7 -0
  39. package/dist/async/loadable.d.ts.map +1 -0
  40. package/dist/async/loadable.js +52 -0
  41. package/dist/async/loadable.js.map +1 -0
  42. package/dist/async/loadable.test.d.ts +2 -0
  43. package/dist/async/loadable.test.d.ts.map +1 -0
  44. package/dist/async/loadable.test.js +322 -0
  45. package/dist/async/loadable.test.js.map +1 -0
  46. package/dist/async/promiseCache.d.ts +14 -0
  47. package/dist/async/promiseCache.d.ts.map +1 -0
  48. package/dist/async/promiseCache.js +29 -0
  49. package/dist/async/promiseCache.js.map +1 -0
  50. package/dist/async/read.d.ts +120 -0
  51. package/dist/async/read.d.ts.map +1 -0
  52. package/dist/async/read.js +286 -0
  53. package/dist/async/read.js.map +1 -0
  54. package/dist/async/read.test.d.ts +2 -0
  55. package/dist/async/read.test.d.ts.map +1 -0
  56. package/dist/async/read.test.js +419 -0
  57. package/dist/async/read.test.js.map +1 -0
  58. package/dist/async/read.typeCheck.d.ts +6 -0
  59. package/dist/async/read.typeCheck.d.ts.map +1 -0
  60. package/dist/async/read.typeCheck.js +101 -0
  61. package/dist/async/read.typeCheck.js.map +1 -0
  62. package/dist/async/safe.d.ts +230 -0
  63. package/dist/async/safe.d.ts.map +1 -0
  64. package/dist/async/safe.js +247 -0
  65. package/dist/async/safe.js.map +1 -0
  66. package/dist/async/safe.test.d.ts +2 -0
  67. package/dist/async/safe.test.d.ts.map +1 -0
  68. package/dist/async/safe.test.js +447 -0
  69. package/dist/async/safe.test.js.map +1 -0
  70. package/dist/async/utils.d.ts +17 -0
  71. package/dist/async/utils.d.ts.map +1 -0
  72. package/dist/async/utils.js +38 -0
  73. package/dist/async/utils.js.map +1 -0
  74. package/dist/async/wait.d.ts +120 -0
  75. package/dist/async/wait.d.ts.map +1 -0
  76. package/dist/async/wait.js +112 -0
  77. package/dist/async/wait.js.map +1 -0
  78. package/dist/async/wait.test.d.ts +2 -0
  79. package/dist/async/wait.test.d.ts.map +1 -0
  80. package/dist/async/wait.test.js +122 -0
  81. package/dist/async/wait.test.js.map +1 -0
  82. package/dist/async/wait.typeCheck.d.ts +6 -0
  83. package/dist/async/wait.typeCheck.d.ts.map +1 -0
  84. package/dist/async/wait.typeCheck.js +104 -0
  85. package/dist/async/wait.typeCheck.js.map +1 -0
  86. package/dist/atom.d.ts +46 -0
  87. package/dist/atom.d.ts.map +1 -0
  88. package/dist/atom.js +86 -0
  89. package/dist/atom.js.map +1 -0
  90. package/dist/atom.test.d.ts +2 -0
  91. package/dist/atom.test.d.ts.map +1 -0
  92. package/dist/atom.test.js +75 -0
  93. package/dist/atom.test.js.map +1 -0
  94. package/dist/batch.d.ts +15 -0
  95. package/dist/batch.d.ts.map +1 -0
  96. package/dist/batch.js +45 -0
  97. package/dist/batch.js.map +1 -0
  98. package/dist/defer.d.ts +56 -0
  99. package/dist/defer.d.ts.map +1 -0
  100. package/dist/defer.js +49 -0
  101. package/dist/defer.js.map +1 -0
  102. package/dist/effect.d.ts +91 -0
  103. package/dist/effect.d.ts.map +1 -0
  104. package/dist/effect.js +311 -0
  105. package/dist/effect.js.map +1 -0
  106. package/dist/effect.test.d.ts +2 -0
  107. package/dist/effect.test.d.ts.map +1 -0
  108. package/dist/effect.test.js +123 -0
  109. package/dist/effect.test.js.map +1 -0
  110. package/dist/emitter.d.ts +129 -0
  111. package/dist/emitter.d.ts.map +1 -0
  112. package/dist/emitter.js +164 -0
  113. package/dist/emitter.js.map +1 -0
  114. package/dist/emitter.test.d.ts +2 -0
  115. package/dist/emitter.test.d.ts.map +1 -0
  116. package/dist/emitter.test.js +259 -0
  117. package/dist/emitter.test.js.map +1 -0
  118. package/dist/equality.d.ts +66 -0
  119. package/dist/equality.d.ts.map +1 -0
  120. package/dist/equality.js +145 -0
  121. package/dist/equality.js.map +1 -0
  122. package/dist/event.d.ts +18 -0
  123. package/dist/event.d.ts.map +1 -0
  124. package/dist/event.js +166 -0
  125. package/dist/event.js.map +1 -0
  126. package/dist/event.test.d.ts +2 -0
  127. package/dist/event.test.d.ts.map +1 -0
  128. package/dist/event.test.js +167 -0
  129. package/dist/event.test.js.map +1 -0
  130. package/dist/hooks.d.ts +152 -0
  131. package/dist/hooks.d.ts.map +1 -0
  132. package/dist/hooks.js +122 -0
  133. package/dist/hooks.js.map +1 -0
  134. package/dist/hooks.test.d.ts +2 -0
  135. package/dist/hooks.test.d.ts.map +1 -0
  136. package/dist/hooks.test.js +99 -0
  137. package/dist/hooks.test.js.map +1 -0
  138. package/dist/index.d.ts +33 -0
  139. package/dist/index.d.ts.map +1 -0
  140. package/dist/index.js +35 -0
  141. package/dist/index.js.map +1 -0
  142. package/dist/isPromiseLike.d.ts +10 -0
  143. package/dist/isPromiseLike.d.ts.map +1 -0
  144. package/dist/isPromiseLike.js +15 -0
  145. package/dist/isPromiseLike.js.map +1 -0
  146. package/dist/pick.d.ts +22 -0
  147. package/dist/pick.d.ts.map +1 -0
  148. package/dist/pick.js +46 -0
  149. package/dist/pick.js.map +1 -0
  150. package/dist/react/index.d.ts +8 -0
  151. package/dist/react/index.d.ts.map +1 -0
  152. package/dist/react/index.js +8 -0
  153. package/dist/react/index.js.map +1 -0
  154. package/dist/react/useRx.d.ts +14 -0
  155. package/dist/react/useRx.d.ts.map +1 -0
  156. package/dist/react/useRx.js +110 -0
  157. package/dist/react/useRx.js.map +1 -0
  158. package/dist/react/useRx.test.d.ts +2 -0
  159. package/dist/react/useRx.test.d.ts.map +1 -0
  160. package/dist/react/useRx.test.js +457 -0
  161. package/dist/react/useRx.test.js.map +1 -0
  162. package/dist/strictModeTest.d.ts +11 -0
  163. package/dist/strictModeTest.d.ts.map +1 -0
  164. package/dist/strictModeTest.js +41 -0
  165. package/dist/strictModeTest.js.map +1 -0
  166. package/dist/types.d.ts +606 -0
  167. package/dist/types.d.ts.map +1 -0
  168. package/dist/types.js +5 -0
  169. package/dist/types.js.map +1 -0
  170. package/dist/untrack.d.ts +14 -0
  171. package/dist/untrack.d.ts.map +1 -0
  172. package/dist/untrack.js +17 -0
  173. package/dist/untrack.js.map +1 -0
  174. package/dist/utils/withUse.d.ts +10 -0
  175. package/dist/utils/withUse.d.ts.map +1 -0
  176. package/dist/utils/withUse.js +21 -0
  177. package/dist/utils/withUse.js.map +1 -0
  178. package/dist/utils/withUse.test.d.ts +2 -0
  179. package/dist/utils/withUse.test.d.ts.map +1 -0
  180. package/dist/utils/withUse.test.js +233 -0
  181. package/dist/utils/withUse.test.js.map +1 -0
  182. package/dist/utils.d.ts +7 -0
  183. package/dist/utils.d.ts.map +1 -0
  184. package/dist/utils.js +7 -0
  185. package/dist/utils.js.map +1 -0
  186. package/dist/utils.test.d.ts +2 -0
  187. package/dist/utils.test.d.ts.map +1 -0
  188. package/dist/utils.test.js +119 -0
  189. package/dist/utils.test.js.map +1 -0
  190. package/package.json +64 -0
@@ -0,0 +1,123 @@
1
+ import { describe, it, expect, vi } from "vitest";
2
+ import { atom } from "./atom";
3
+ import { effect } from "./effect";
4
+ describe("effect", () => {
5
+ it("should run immediately on creation", () => {
6
+ const fn = vi.fn();
7
+ effect(fn);
8
+ expect(fn).toHaveBeenCalledTimes(1);
9
+ });
10
+ it("should track dependencies and re-run on change", () => {
11
+ const [count, setCount] = atom(0);
12
+ const fn = vi.fn(() => {
13
+ count(); // track dependency
14
+ });
15
+ effect(fn);
16
+ expect(fn).toHaveBeenCalledTimes(1);
17
+ setCount(1);
18
+ expect(fn).toHaveBeenCalledTimes(2);
19
+ });
20
+ it("should provide context with nth counter", () => {
21
+ const [count, setCount] = atom(0);
22
+ const nthValues = [];
23
+ effect((ctx) => {
24
+ nthValues.push(ctx.nth);
25
+ count();
26
+ });
27
+ setCount(1);
28
+ setCount(2);
29
+ expect(nthValues).toEqual([0, 1, 2]);
30
+ });
31
+ it("should provide name from options in context", () => {
32
+ let receivedName;
33
+ effect((ctx) => {
34
+ receivedName = ctx.key;
35
+ }, { key: "testEffect" });
36
+ expect(receivedName).toBe("testEffect");
37
+ });
38
+ it("should have undefined name when not provided", () => {
39
+ let receivedName = "initial";
40
+ effect((ctx) => {
41
+ receivedName = ctx.key;
42
+ });
43
+ expect(receivedName).toBeUndefined();
44
+ });
45
+ it("should dispose when dispose function is called", () => {
46
+ const [count, setCount] = atom(0);
47
+ const fn = vi.fn(() => {
48
+ count();
49
+ });
50
+ const dispose = effect(fn);
51
+ expect(fn).toHaveBeenCalledTimes(1);
52
+ dispose();
53
+ setCount(1);
54
+ expect(fn).toHaveBeenCalledTimes(1); // not called again
55
+ });
56
+ it("should handle errors with onError handler", () => {
57
+ const errorHandler = vi.fn();
58
+ const error = new Error("test error");
59
+ effect((ctx) => {
60
+ ctx.onError(errorHandler);
61
+ throw error;
62
+ });
63
+ expect(errorHandler).toHaveBeenCalledWith(error);
64
+ });
65
+ it("should rethrow errors if no onError handler", () => {
66
+ const error = new Error("test error");
67
+ expect(() => {
68
+ effect(() => {
69
+ throw error;
70
+ });
71
+ }).toThrow(error);
72
+ });
73
+ it("should support use() for plugins", () => {
74
+ const plugin = vi.fn((ctx) => {
75
+ return ctx.nth * 2;
76
+ });
77
+ effect((ctx) => {
78
+ const result = ctx.use(plugin);
79
+ expect(result).toBe(0);
80
+ });
81
+ });
82
+ it("should throw error if effect returns a Promise", () => {
83
+ // Using async function (returns Promise)
84
+ expect(() => {
85
+ effect(async () => {
86
+ await Promise.resolve();
87
+ });
88
+ }).toThrow("effect() received an async function");
89
+ });
90
+ it("should throw error if effect returns a Promise-like object", () => {
91
+ // Returning a thenable object
92
+ expect(() => {
93
+ effect(() => {
94
+ return { then: () => { } };
95
+ });
96
+ }).toThrow("effect() received an async function");
97
+ });
98
+ it("should throw error if effect returns a plain Promise", () => {
99
+ // Explicitly returning a Promise
100
+ expect(() => {
101
+ effect(() => {
102
+ return Promise.resolve("value");
103
+ });
104
+ }).toThrow("effect() received an async function");
105
+ });
106
+ it("should not throw for effects that return undefined", () => {
107
+ // Normal effect that returns nothing
108
+ expect(() => {
109
+ effect(() => {
110
+ // do something
111
+ });
112
+ }).not.toThrow();
113
+ });
114
+ it("should not throw for effects that return non-Promise values", () => {
115
+ // Effect that returns a value (unusual but allowed)
116
+ expect(() => {
117
+ effect(() => {
118
+ return "some value";
119
+ });
120
+ }).not.toThrow();
121
+ });
122
+ });
123
+ //# sourceMappingURL=effect.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"effect.test.js","sourceRoot":"","sources":["../src/effect.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAC;AAClD,OAAO,EAAE,IAAI,EAAE,MAAM,QAAQ,CAAC;AAC9B,OAAO,EAAE,MAAM,EAAE,MAAM,UAAU,CAAC;AAElC,QAAQ,CAAC,QAAQ,EAAE,GAAG,EAAE;IACtB,EAAE,CAAC,oCAAoC,EAAE,GAAG,EAAE;QAC5C,MAAM,EAAE,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC;QACnB,MAAM,CAAC,EAAE,CAAC,CAAC;QACX,MAAM,CAAC,EAAE,CAAC,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;IACtC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gDAAgD,EAAE,GAAG,EAAE;QACxD,MAAM,CAAC,KAAK,EAAE,QAAQ,CAAC,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC;QAClC,MAAM,EAAE,GAAG,EAAE,CAAC,EAAE,CAAC,GAAG,EAAE;YACpB,KAAK,EAAE,CAAC,CAAC,mBAAmB;QAC9B,CAAC,CAAC,CAAC;QAEH,MAAM,CAAC,EAAE,CAAC,CAAC;QACX,MAAM,CAAC,EAAE,CAAC,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;QAEpC,QAAQ,CAAC,CAAC,CAAC,CAAC;QACZ,MAAM,CAAC,EAAE,CAAC,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;IACtC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,yCAAyC,EAAE,GAAG,EAAE;QACjD,MAAM,CAAC,KAAK,EAAE,QAAQ,CAAC,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC;QAClC,MAAM,SAAS,GAAa,EAAE,CAAC;QAE/B,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE;YACb,SAAS,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;YACxB,KAAK,EAAE,CAAC;QACV,CAAC,CAAC,CAAC;QAEH,QAAQ,CAAC,CAAC,CAAC,CAAC;QACZ,QAAQ,CAAC,CAAC,CAAC,CAAC;QAEZ,MAAM,CAAC,SAAS,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;IACvC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6CAA6C,EAAE,GAAG,EAAE;QACrD,IAAI,YAAgC,CAAC;QAErC,MAAM,CACJ,CAAC,GAAG,EAAE,EAAE;YACN,YAAY,GAAG,GAAG,CAAC,GAAG,CAAC;QACzB,CAAC,EACD,EAAE,GAAG,EAAE,YAAY,EAAE,CACtB,CAAC;QAEF,MAAM,CAAC,YAAY,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;IAC1C,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,8CAA8C,EAAE,GAAG,EAAE;QACtD,IAAI,YAAY,GAAuB,SAAS,CAAC;QAEjD,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE;YACb,YAAY,GAAG,GAAG,CAAC,GAAG,CAAC;QACzB,CAAC,CAAC,CAAC;QAEH,MAAM,CAAC,YAAY,CAAC,CAAC,aAAa,EAAE,CAAC;IACvC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gDAAgD,EAAE,GAAG,EAAE;QACxD,MAAM,CAAC,KAAK,EAAE,QAAQ,CAAC,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC;QAClC,MAAM,EAAE,GAAG,EAAE,CAAC,EAAE,CAAC,GAAG,EAAE;YACpB,KAAK,EAAE,CAAC;QACV,CAAC,CAAC,CAAC;QAEH,MAAM,OAAO,GAAG,MAAM,CAAC,EAAE,CAAC,CAAC;QAC3B,MAAM,CAAC,EAAE,CAAC,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;QAEpC,OAAO,EAAE,CAAC;QACV,QAAQ,CAAC,CAAC,CAAC,CAAC;QACZ,MAAM,CAAC,EAAE,CAAC,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,CAAC,mBAAmB;IAC1D,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,2CAA2C,EAAE,GAAG,EAAE;QACnD,MAAM,YAAY,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC;QAC7B,MAAM,KAAK,GAAG,IAAI,KAAK,CAAC,YAAY,CAAC,CAAC;QAEtC,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE;YACb,GAAG,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC;YAC1B,MAAM,KAAK,CAAC;QACd,CAAC,CAAC,CAAC;QAEH,MAAM,CAAC,YAAY,CAAC,CAAC,oBAAoB,CAAC,KAAK,CAAC,CAAC;IACnD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6CAA6C,EAAE,GAAG,EAAE;QACrD,MAAM,KAAK,GAAG,IAAI,KAAK,CAAC,YAAY,CAAC,CAAC;QAEtC,MAAM,CAAC,GAAG,EAAE;YACV,MAAM,CAAC,GAAG,EAAE;gBACV,MAAM,KAAK,CAAC;YACd,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;IACpB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,kCAAkC,EAAE,GAAG,EAAE;QAC1C,MAAM,MAAM,GAAG,EAAE,CAAC,EAAE,CAAC,CAAC,GAAG,EAAE,EAAE;YAC3B,OAAO,GAAG,CAAC,GAAG,GAAG,CAAC,CAAC;QACrB,CAAC,CAAC,CAAC;QAEH,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE;YACb,MAAM,MAAM,GAAG,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;YAC/B,MAAM,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QACzB,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gDAAgD,EAAE,GAAG,EAAE;QACxD,yCAAyC;QACzC,MAAM,CAAC,GAAG,EAAE;YACV,MAAM,CAAC,KAAK,IAAI,EAAE;gBAChB,MAAM,OAAO,CAAC,OAAO,EAAE,CAAC;YAC1B,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC,OAAO,CAAC,qCAAqC,CAAC,CAAC;IACpD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,4DAA4D,EAAE,GAAG,EAAE;QACpE,8BAA8B;QAC9B,MAAM,CAAC,GAAG,EAAE;YACV,MAAM,CAAC,GAAG,EAAE;gBACV,OAAO,EAAE,IAAI,EAAE,GAAG,EAAE,GAAE,CAAC,EAAS,CAAC;YACnC,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC,OAAO,CAAC,qCAAqC,CAAC,CAAC;IACpD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,sDAAsD,EAAE,GAAG,EAAE;QAC9D,iCAAiC;QACjC,MAAM,CAAC,GAAG,EAAE;YACV,MAAM,CAAC,GAAG,EAAE;gBACV,OAAO,OAAO,CAAC,OAAO,CAAC,OAAO,CAAQ,CAAC;YACzC,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC,OAAO,CAAC,qCAAqC,CAAC,CAAC;IACpD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,oDAAoD,EAAE,GAAG,EAAE;QAC5D,qCAAqC;QACrC,MAAM,CAAC,GAAG,EAAE;YACV,MAAM,CAAC,GAAG,EAAE;gBACV,eAAe;YACjB,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,EAAE,CAAC;IACnB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6DAA6D,EAAE,GAAG,EAAE;QACrE,oDAAoD;QACpD,MAAM,CAAC,GAAG,EAAE;YACV,MAAM,CAAC,GAAG,EAAE;gBACV,OAAO,YAAmB,CAAC;YAC7B,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,EAAE,CAAC;IACnB,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -0,0 +1,129 @@
1
+ import { Listener, SingleOrMultipleListeners } from "./types";
2
+ /**
3
+ * Event emitter interface for pub/sub pattern.
4
+ *
5
+ * @template T - The type of payload emitted to listeners (defaults to void)
6
+ */
7
+ export interface Emitter<T = void> {
8
+ /**
9
+ * Subscribe to events with one or more listeners.
10
+ *
11
+ * @param listeners - Single listener or array of listeners
12
+ * @returns Unsubscribe function (idempotent - safe to call multiple times)
13
+ */
14
+ on(listeners: SingleOrMultipleListeners<T>): VoidFunction;
15
+ /**
16
+ * Subscribe with a mapping function that filters and transforms events.
17
+ *
18
+ * The map function receives the emitted value and returns either:
19
+ * - `{ value: TValue }` - Listener is called with the transformed value
20
+ * - `undefined` - Listener is NOT called (event filtered out)
21
+ *
22
+ * @template TValue - The transformed value type passed to listeners
23
+ * @param map - Transform function that can filter (return undefined) or map values
24
+ * @param listeners - Single listener or array of listeners for transformed values
25
+ * @returns Unsubscribe function
26
+ *
27
+ * @example Filter and transform
28
+ * ```ts
29
+ * const emitter = emitter<{ type: string; data: number }>();
30
+ *
31
+ * // Only listen to 'success' events, extract just the data
32
+ * emitter.on(
33
+ * (event) => event.type === 'success' ? { value: event.data } : undefined,
34
+ * (data) => console.log('Success data:', data)
35
+ * );
36
+ * ```
37
+ */
38
+ on<TValue>(map: (value: T) => {
39
+ value: TValue;
40
+ } | undefined, listeners: SingleOrMultipleListeners<TValue>): VoidFunction;
41
+ /**
42
+ * Emit an event to all registered listeners.
43
+ *
44
+ * @param payload - The value to pass to all listeners
45
+ */
46
+ emit(payload: T): void;
47
+ /**
48
+ * Emit an event to all registered listeners in LIFO (reverse) order.
49
+ * Useful for cleanup scenarios where resources should be released
50
+ * in reverse order of acquisition.
51
+ *
52
+ * @param payload - The value to pass to all listeners
53
+ */
54
+ emitLifo(payload: T): void;
55
+ /**
56
+ * Remove all registered listeners.
57
+ */
58
+ clear(): void;
59
+ /**
60
+ * Emit an event to all listeners, then clear all listeners.
61
+ * Useful for one-time events like disposal.
62
+ *
63
+ * @param payload - The value to pass to all listeners
64
+ */
65
+ emitAndClear(payload: T): void;
66
+ /**
67
+ * Emit an event to all listeners in LIFO (reverse) order, then clear.
68
+ * Useful for cleanup scenarios where resources should be released
69
+ * in reverse order of acquisition.
70
+ *
71
+ * @param payload - The value to pass to all listeners
72
+ */
73
+ emitAndClearLifo(payload: T): void;
74
+ /**
75
+ * Emit to all listeners, clear, and "settle" the emitter.
76
+ *
77
+ * After settling:
78
+ * - Any new `on()` call immediately invokes the listener with the settled payload
79
+ * - Returns a no-op unsubscribe function
80
+ * - `emit()` and `emitAndClear()` become no-ops
81
+ *
82
+ * Useful for one-time events where late subscribers should still receive the value
83
+ * (similar to Promise behavior).
84
+ *
85
+ * @param payload - The final value to pass to all listeners
86
+ */
87
+ settle(payload: T): void;
88
+ /** Number of registered listeners */
89
+ readonly size: number;
90
+ /** Whether the emitter has been settled */
91
+ readonly settled: boolean;
92
+ }
93
+ /**
94
+ * Creates an event emitter for managing and notifying listeners.
95
+ *
96
+ * An emitter provides a simple pub/sub pattern for managing event listeners.
97
+ * It's used internally by signals and effects to manage subscriptions and notifications.
98
+ *
99
+ * Features:
100
+ * - Add listeners that will be notified when events are emitted
101
+ * - Emit events to all registered listeners
102
+ * - Remove listeners via unsubscribe functions
103
+ * - Clear all listeners at once
104
+ * - Safe to call unsubscribe multiple times (idempotent)
105
+ *
106
+ * @template T - The type of payload that will be emitted to listeners (defaults to void)
107
+ * @returns An emitter object with add, emit, and clear methods
108
+ *
109
+ * @example
110
+ * ```ts
111
+ * const eventEmitter = emitter<string>();
112
+ *
113
+ * // Subscribe to events
114
+ * const unsubscribe = eventEmitter.on((message) => {
115
+ * console.log('Received:', message);
116
+ * });
117
+ *
118
+ * // Emit an event
119
+ * eventEmitter.emit('Hello'); // Logs: "Received: Hello"
120
+ *
121
+ * // Unsubscribe
122
+ * unsubscribe();
123
+ *
124
+ * // Clear all listeners
125
+ * eventEmitter.clear();
126
+ * ```
127
+ */
128
+ export declare function emitter<T = void>(initialListeners?: Listener<T>[]): Emitter<T>;
129
+ //# sourceMappingURL=emitter.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"emitter.d.ts","sourceRoot":"","sources":["../src/emitter.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,yBAAyB,EAAE,MAAM,SAAS,CAAC;AAE9D;;;;GAIG;AACH,MAAM,WAAW,OAAO,CAAC,CAAC,GAAG,IAAI;IAC/B;;;;;OAKG;IACH,EAAE,CAAC,SAAS,EAAE,yBAAyB,CAAC,CAAC,CAAC,GAAG,YAAY,CAAC;IAE1D;;;;;;;;;;;;;;;;;;;;;;OAsBG;IACH,EAAE,CAAC,MAAM,EACP,GAAG,EAAE,CAAC,KAAK,EAAE,CAAC,KAAK;QAAE,KAAK,EAAE,MAAM,CAAA;KAAE,GAAG,SAAS,EAChD,SAAS,EAAE,yBAAyB,CAAC,MAAM,CAAC,GAC3C,YAAY,CAAC;IAEhB;;;;OAIG;IACH,IAAI,CAAC,OAAO,EAAE,CAAC,GAAG,IAAI,CAAC;IAEvB;;;;;;OAMG;IACH,QAAQ,CAAC,OAAO,EAAE,CAAC,GAAG,IAAI,CAAC;IAE3B;;OAEG;IACH,KAAK,IAAI,IAAI,CAAC;IAEd;;;;;OAKG;IACH,YAAY,CAAC,OAAO,EAAE,CAAC,GAAG,IAAI,CAAC;IAE/B;;;;;;OAMG;IACH,gBAAgB,CAAC,OAAO,EAAE,CAAC,GAAG,IAAI,CAAC;IAEnC;;;;;;;;;;;;OAYG;IACH,MAAM,CAAC,OAAO,EAAE,CAAC,GAAG,IAAI,CAAC;IAEzB,qCAAqC;IACrC,QAAQ,CAAC,IAAI,EAAE,MAAM,CAAC;IAEtB,2CAA2C;IAC3C,QAAQ,CAAC,OAAO,EAAE,OAAO,CAAC;CAC3B;AA4ID;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAkCG;AACH,wBAAgB,OAAO,CAAC,CAAC,GAAG,IAAI,EAC9B,gBAAgB,CAAC,EAAE,QAAQ,CAAC,CAAC,CAAC,EAAE,GAC/B,OAAO,CAAC,CAAC,CAAC,CAEZ"}
@@ -0,0 +1,164 @@
1
+ const noop = () => { };
2
+ /**
3
+ * Class-based emitter implementation for better V8 optimization.
4
+ * All instances share methods via prototype.
5
+ */
6
+ class EmitterImpl {
7
+ /** Set of registered listeners */
8
+ _listeners;
9
+ /** Settled payload (if settled) */
10
+ _settledPayload = undefined;
11
+ /** Whether the emitter has been settled */
12
+ _isSettled = false;
13
+ /** Bound version of on() for passing as callback */
14
+ on;
15
+ constructor(initialListeners) {
16
+ this._listeners = new Set(initialListeners);
17
+ // Bind 'on' to preserve 'this' context when passed as callback
18
+ this.on = this._on.bind(this);
19
+ }
20
+ get size() {
21
+ return this._listeners.size;
22
+ }
23
+ get settled() {
24
+ return this._isSettled;
25
+ }
26
+ _on(listenersOrMap, mappedListeners) {
27
+ let newListeners;
28
+ if (mappedListeners === undefined) {
29
+ // Simple form: on(listeners)
30
+ newListeners = Array.isArray(listenersOrMap)
31
+ ? listenersOrMap
32
+ : [listenersOrMap];
33
+ }
34
+ else {
35
+ // Mapped form: on(map, listeners)
36
+ const map = listenersOrMap;
37
+ const sourceListeners = Array.isArray(mappedListeners)
38
+ ? mappedListeners
39
+ : [mappedListeners];
40
+ newListeners = [
41
+ (value) => {
42
+ const mappedValue = map(value);
43
+ if (mappedValue) {
44
+ for (let i = 0; i < sourceListeners.length; i++) {
45
+ sourceListeners[i](mappedValue.value);
46
+ }
47
+ }
48
+ },
49
+ ];
50
+ }
51
+ // If settled, call listeners immediately and return no-op
52
+ if (this._isSettled) {
53
+ const payload = this._settledPayload;
54
+ for (let i = 0; i < newListeners.length; i++) {
55
+ newListeners[i](payload);
56
+ }
57
+ return noop;
58
+ }
59
+ const listeners = this._listeners;
60
+ for (let i = 0; i < newListeners.length; i++) {
61
+ listeners.add(newListeners[i]);
62
+ }
63
+ return () => {
64
+ for (let i = 0; i < newListeners.length; i++) {
65
+ listeners.delete(newListeners[i]);
66
+ }
67
+ };
68
+ }
69
+ emit(payload) {
70
+ if (this._isSettled)
71
+ return;
72
+ this._doEmit(payload, false, false);
73
+ }
74
+ emitLifo(payload) {
75
+ if (this._isSettled)
76
+ return;
77
+ this._doEmit(payload, false, true);
78
+ }
79
+ clear() {
80
+ this._listeners.clear();
81
+ }
82
+ emitAndClear(payload) {
83
+ if (this._isSettled)
84
+ return;
85
+ this._doEmit(payload, true, false);
86
+ }
87
+ emitAndClearLifo(payload) {
88
+ if (this._isSettled)
89
+ return;
90
+ this._doEmit(payload, true, true);
91
+ }
92
+ settle(payload) {
93
+ if (this._isSettled)
94
+ return;
95
+ this._settledPayload = payload;
96
+ this._isSettled = true;
97
+ this._doEmit(payload, true, false);
98
+ }
99
+ /**
100
+ * Internal emit implementation.
101
+ * Creates snapshot to handle modifications during iteration.
102
+ */
103
+ _doEmit(payload, clear, lifo) {
104
+ const listeners = this._listeners;
105
+ const size = listeners.size;
106
+ if (size === 0)
107
+ return;
108
+ // Create snapshot - necessary because Set.forEach includes items added during iteration
109
+ const copy = Array.from(listeners);
110
+ if (clear) {
111
+ listeners.clear();
112
+ }
113
+ // Use traditional for loop for maximum performance
114
+ if (lifo) {
115
+ for (let i = size - 1; i >= 0; i--) {
116
+ copy[i](payload);
117
+ }
118
+ }
119
+ else {
120
+ for (let i = 0; i < size; i++) {
121
+ copy[i](payload);
122
+ }
123
+ }
124
+ }
125
+ }
126
+ /**
127
+ * Creates an event emitter for managing and notifying listeners.
128
+ *
129
+ * An emitter provides a simple pub/sub pattern for managing event listeners.
130
+ * It's used internally by signals and effects to manage subscriptions and notifications.
131
+ *
132
+ * Features:
133
+ * - Add listeners that will be notified when events are emitted
134
+ * - Emit events to all registered listeners
135
+ * - Remove listeners via unsubscribe functions
136
+ * - Clear all listeners at once
137
+ * - Safe to call unsubscribe multiple times (idempotent)
138
+ *
139
+ * @template T - The type of payload that will be emitted to listeners (defaults to void)
140
+ * @returns An emitter object with add, emit, and clear methods
141
+ *
142
+ * @example
143
+ * ```ts
144
+ * const eventEmitter = emitter<string>();
145
+ *
146
+ * // Subscribe to events
147
+ * const unsubscribe = eventEmitter.on((message) => {
148
+ * console.log('Received:', message);
149
+ * });
150
+ *
151
+ * // Emit an event
152
+ * eventEmitter.emit('Hello'); // Logs: "Received: Hello"
153
+ *
154
+ * // Unsubscribe
155
+ * unsubscribe();
156
+ *
157
+ * // Clear all listeners
158
+ * eventEmitter.clear();
159
+ * ```
160
+ */
161
+ export function emitter(initialListeners) {
162
+ return new EmitterImpl(initialListeners);
163
+ }
164
+ //# sourceMappingURL=emitter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"emitter.js","sourceRoot":"","sources":["../src/emitter.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=emitter.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"emitter.test.d.ts","sourceRoot":"","sources":["../src/emitter.test.ts"],"names":[],"mappings":""}