@praxisjs/core 0.4.0 → 0.4.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/CHANGELOG.md +10 -0
- package/dist/__tests__/batch.test.d.ts +2 -0
- package/dist/__tests__/batch.test.d.ts.map +1 -0
- package/dist/__tests__/batch.test.js +26 -0
- package/dist/__tests__/batch.test.js.map +1 -0
- package/dist/__tests__/component.test.d.ts +2 -0
- package/dist/__tests__/component.test.d.ts.map +1 -0
- package/dist/__tests__/component.test.js +48 -0
- package/dist/__tests__/component.test.js.map +1 -0
- package/dist/__tests__/computed.test.d.ts +2 -0
- package/dist/__tests__/computed.test.d.ts.map +1 -0
- package/dist/__tests__/computed.test.js +79 -0
- package/dist/__tests__/computed.test.js.map +1 -0
- package/dist/__tests__/effect.test.d.ts +2 -0
- package/dist/__tests__/effect.test.d.ts.map +1 -0
- package/dist/__tests__/effect.test.js +49 -0
- package/dist/__tests__/effect.test.js.map +1 -0
- package/dist/__tests__/peek.test.d.ts +2 -0
- package/dist/__tests__/peek.test.d.ts.map +1 -0
- package/dist/__tests__/peek.test.js +26 -0
- package/dist/__tests__/peek.test.js.map +1 -0
- package/dist/__tests__/persisted-signal.test.d.ts +2 -0
- package/dist/__tests__/persisted-signal.test.d.ts.map +1 -0
- package/dist/__tests__/persisted-signal.test.js +74 -0
- package/dist/__tests__/persisted-signal.test.js.map +1 -0
- package/dist/__tests__/reactive.test.d.ts +2 -0
- package/dist/__tests__/reactive.test.d.ts.map +1 -0
- package/dist/__tests__/reactive.test.js +181 -0
- package/dist/__tests__/reactive.test.js.map +1 -0
- package/dist/__tests__/resource.test.d.ts +2 -0
- package/dist/__tests__/resource.test.d.ts.map +1 -0
- package/dist/__tests__/resource.test.js +84 -0
- package/dist/__tests__/resource.test.js.map +1 -0
- package/dist/__tests__/signal.test.d.ts +2 -0
- package/dist/__tests__/signal.test.d.ts.map +1 -0
- package/dist/__tests__/signal.test.js +64 -0
- package/dist/__tests__/signal.test.js.map +1 -0
- package/dist/__tests__/track.test.d.ts +2 -0
- package/dist/__tests__/track.test.d.ts.map +1 -0
- package/dist/__tests__/track.test.js +54 -0
- package/dist/__tests__/track.test.js.map +1 -0
- package/dist/reactive/reactive.d.ts.map +1 -1
- package/dist/reactive/reactive.js +8 -1
- package/dist/reactive/reactive.js.map +1 -1
- package/dist/signal/persisted.d.ts.map +1 -1
- package/dist/signal/persisted.js +10 -12
- package/dist/signal/persisted.js.map +1 -1
- package/package.json +1 -1
- package/src/__tests__/batch.test.ts +29 -0
- package/src/__tests__/component.test.ts +58 -0
- package/src/__tests__/computed.test.ts +89 -0
- package/src/__tests__/effect.test.ts +54 -0
- package/src/__tests__/peek.test.ts +31 -0
- package/src/__tests__/persisted-signal.test.ts +87 -0
- package/src/__tests__/reactive.test.ts +210 -0
- package/src/__tests__/resource.test.ts +97 -0
- package/src/__tests__/signal.test.ts +74 -0
- package/src/__tests__/track.test.ts +61 -0
- package/src/reactive/reactive.ts +9 -1
- package/src/signal/persisted.ts +12 -14
|
@@ -0,0 +1,84 @@
|
|
|
1
|
+
import { describe, it, expect, vi } from "vitest";
|
|
2
|
+
import { resource, createResource } from "../async/resource";
|
|
3
|
+
import { signal } from "../signal/signal";
|
|
4
|
+
describe("resource", () => {
|
|
5
|
+
it("starts as pending and resolves to success", async () => {
|
|
6
|
+
const r = resource(() => Promise.resolve("hello"));
|
|
7
|
+
expect(r.status()).toBe("pending");
|
|
8
|
+
await vi.waitFor(() => r.status() === "success");
|
|
9
|
+
expect(r.data()).toBe("hello");
|
|
10
|
+
expect(r.pending()).toBe(false);
|
|
11
|
+
expect(r.error()).toBeNull();
|
|
12
|
+
});
|
|
13
|
+
it("transitions to error on rejection", async () => {
|
|
14
|
+
const r = resource(() => Promise.reject(new Error("oops")));
|
|
15
|
+
await new Promise((res) => setTimeout(res, 0));
|
|
16
|
+
expect(r.status()).toBe("error");
|
|
17
|
+
expect(r.error().message).toBe("oops");
|
|
18
|
+
expect(r.data()).toBeNull();
|
|
19
|
+
});
|
|
20
|
+
it("wraps non-Error rejections in Error", async () => {
|
|
21
|
+
const r = resource(() => Promise.reject("plain string"));
|
|
22
|
+
await new Promise((res) => setTimeout(res, 0));
|
|
23
|
+
expect(r.status()).toBe("error");
|
|
24
|
+
expect(r.error()).toBeInstanceOf(Error);
|
|
25
|
+
});
|
|
26
|
+
it("immediate=false does not fetch on creation", () => {
|
|
27
|
+
const fetcher = vi.fn(() => Promise.resolve(1));
|
|
28
|
+
const r = resource(fetcher, { immediate: false });
|
|
29
|
+
expect(fetcher).not.toHaveBeenCalled();
|
|
30
|
+
expect(r.status()).toBe("idle");
|
|
31
|
+
});
|
|
32
|
+
it("refetch() re-runs the fetcher", async () => {
|
|
33
|
+
let count = 0;
|
|
34
|
+
const r = resource(() => Promise.resolve(++count));
|
|
35
|
+
await vi.waitFor(() => r.status() === "success");
|
|
36
|
+
expect(r.data()).toBe(1);
|
|
37
|
+
r.refetch();
|
|
38
|
+
await vi.waitFor(() => r.data() === 2);
|
|
39
|
+
expect(r.data()).toBe(2);
|
|
40
|
+
});
|
|
41
|
+
it("cancel() sets status back to idle and ignores in-flight result", async () => {
|
|
42
|
+
let resolve;
|
|
43
|
+
const r = resource(() => new Promise((res) => { resolve = res; }));
|
|
44
|
+
r.cancel();
|
|
45
|
+
expect(r.status()).toBe("idle");
|
|
46
|
+
resolve(99);
|
|
47
|
+
await new Promise((r) => setTimeout(r, 10));
|
|
48
|
+
expect(r.data()).toBeNull(); // stale result ignored
|
|
49
|
+
});
|
|
50
|
+
it("mutate() sets data directly", async () => {
|
|
51
|
+
const r = resource(() => Promise.resolve(1));
|
|
52
|
+
await vi.waitFor(() => r.status() === "success");
|
|
53
|
+
r.mutate(999);
|
|
54
|
+
expect(r.data()).toBe(999);
|
|
55
|
+
expect(r.status()).toBe("success");
|
|
56
|
+
expect(r.error()).toBeNull();
|
|
57
|
+
});
|
|
58
|
+
it("initialData is visible when immediate=false", () => {
|
|
59
|
+
const r = resource(() => new Promise(() => { }), { initialData: "cached", immediate: false });
|
|
60
|
+
expect(r.data()).toBe("cached");
|
|
61
|
+
expect(r.status()).toBe("idle");
|
|
62
|
+
});
|
|
63
|
+
it("keepPreviousData=true preserves old data during refetch", async () => {
|
|
64
|
+
let call = 0;
|
|
65
|
+
const r = resource(() => Promise.resolve(++call), { keepPreviousData: true });
|
|
66
|
+
await vi.waitFor(() => r.data() === 1);
|
|
67
|
+
r.refetch();
|
|
68
|
+
// data should still be 1 while pending
|
|
69
|
+
expect(r.data()).toBe(1);
|
|
70
|
+
await vi.waitFor(() => r.data() === 2);
|
|
71
|
+
});
|
|
72
|
+
});
|
|
73
|
+
describe("createResource", () => {
|
|
74
|
+
it("re-fetches when the param signal changes", async () => {
|
|
75
|
+
const id = signal(1);
|
|
76
|
+
const fetcher = vi.fn((n) => Promise.resolve(n * 10));
|
|
77
|
+
const r = createResource(id, fetcher);
|
|
78
|
+
await vi.waitFor(() => r.data() === 10);
|
|
79
|
+
id.set(2);
|
|
80
|
+
await vi.waitFor(() => r.data() === 20);
|
|
81
|
+
expect(fetcher).toHaveBeenCalledTimes(2);
|
|
82
|
+
});
|
|
83
|
+
});
|
|
84
|
+
//# sourceMappingURL=resource.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"resource.test.js","sourceRoot":"","sources":["../../src/__tests__/resource.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"signal.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/signal.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,64 @@
|
|
|
1
|
+
import { describe, it, expect } from "vitest";
|
|
2
|
+
import { signal } from "../signal/signal";
|
|
3
|
+
describe("signal", () => {
|
|
4
|
+
it("returns initial value", () => {
|
|
5
|
+
const s = signal(42);
|
|
6
|
+
expect(s()).toBe(42);
|
|
7
|
+
});
|
|
8
|
+
it("updates value with set", () => {
|
|
9
|
+
const s = signal(0);
|
|
10
|
+
s.set(10);
|
|
11
|
+
expect(s()).toBe(10);
|
|
12
|
+
});
|
|
13
|
+
it("updates value with update fn", () => {
|
|
14
|
+
const s = signal(5);
|
|
15
|
+
s.update((v) => v * 2);
|
|
16
|
+
expect(s()).toBe(10);
|
|
17
|
+
});
|
|
18
|
+
it("skips update when value is identical", () => {
|
|
19
|
+
const s = signal(1);
|
|
20
|
+
const calls = [];
|
|
21
|
+
s.subscribe((v) => calls.push(v));
|
|
22
|
+
const before = calls.length;
|
|
23
|
+
s.set(1); // same value
|
|
24
|
+
expect(calls.length).toBe(before);
|
|
25
|
+
});
|
|
26
|
+
it("subscribe fires immediately with current value", () => {
|
|
27
|
+
const s = signal("hello");
|
|
28
|
+
const received = [];
|
|
29
|
+
s.subscribe((v) => received.push(v));
|
|
30
|
+
expect(received).toEqual(["hello"]);
|
|
31
|
+
});
|
|
32
|
+
it("subscribe fires on subsequent updates", () => {
|
|
33
|
+
const s = signal(0);
|
|
34
|
+
const received = [];
|
|
35
|
+
s.subscribe((v) => received.push(v));
|
|
36
|
+
s.set(1);
|
|
37
|
+
s.set(2);
|
|
38
|
+
expect(received).toEqual([0, 1, 2]);
|
|
39
|
+
});
|
|
40
|
+
it("unsubscribe stops receiving updates", () => {
|
|
41
|
+
const s = signal(0);
|
|
42
|
+
const received = [];
|
|
43
|
+
const unsub = s.subscribe((v) => received.push(v));
|
|
44
|
+
unsub();
|
|
45
|
+
s.set(99);
|
|
46
|
+
expect(received).toEqual([0]);
|
|
47
|
+
});
|
|
48
|
+
it("marks __isSignal = true", () => {
|
|
49
|
+
const s = signal(0);
|
|
50
|
+
expect(s.__isSignal).toBe(true);
|
|
51
|
+
});
|
|
52
|
+
it("supports object values", () => {
|
|
53
|
+
const s = signal({ name: "Alice" });
|
|
54
|
+
s.set({ name: "Bob" });
|
|
55
|
+
expect(s().name).toBe("Bob");
|
|
56
|
+
});
|
|
57
|
+
it("supports null and undefined", () => {
|
|
58
|
+
const s = signal(null);
|
|
59
|
+
expect(s()).toBeNull();
|
|
60
|
+
s.set(5);
|
|
61
|
+
expect(s()).toBe(5);
|
|
62
|
+
});
|
|
63
|
+
});
|
|
64
|
+
//# sourceMappingURL=signal.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"signal.test.js","sourceRoot":"","sources":["../../src/__tests__/signal.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"track.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/track.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
import { describe, it, expect, vi } from "vitest";
|
|
2
|
+
import { track, runEffect, activeEffect } from "../signal/effect";
|
|
3
|
+
import { signal } from "../signal/signal";
|
|
4
|
+
describe("track", () => {
|
|
5
|
+
it("runs the effect immediately", () => {
|
|
6
|
+
const fn = vi.fn();
|
|
7
|
+
track(fn);
|
|
8
|
+
expect(fn).toHaveBeenCalledOnce();
|
|
9
|
+
});
|
|
10
|
+
it("sets activeEffect during execution, resets after", () => {
|
|
11
|
+
let during = "not-set";
|
|
12
|
+
track(() => {
|
|
13
|
+
during = activeEffect;
|
|
14
|
+
});
|
|
15
|
+
expect(during).toBeTypeOf("function");
|
|
16
|
+
expect(activeEffect).toBeNull();
|
|
17
|
+
});
|
|
18
|
+
it("supports nested track calls — restores outer effect", () => {
|
|
19
|
+
const outer = vi.fn();
|
|
20
|
+
const inner = vi.fn();
|
|
21
|
+
let outerActive;
|
|
22
|
+
let innerActive;
|
|
23
|
+
track(() => {
|
|
24
|
+
outer();
|
|
25
|
+
outerActive = activeEffect;
|
|
26
|
+
track(() => {
|
|
27
|
+
inner();
|
|
28
|
+
innerActive = activeEffect;
|
|
29
|
+
});
|
|
30
|
+
// after inner track, activeEffect should be back to outer
|
|
31
|
+
expect(activeEffect).toBe(outerActive);
|
|
32
|
+
});
|
|
33
|
+
expect(outer).toHaveBeenCalledOnce();
|
|
34
|
+
expect(inner).toHaveBeenCalledOnce();
|
|
35
|
+
expect(innerActive).not.toBe(outerActive);
|
|
36
|
+
});
|
|
37
|
+
it("subscribes the tracked effect to signals read inside it", () => {
|
|
38
|
+
const s = signal(1);
|
|
39
|
+
const values = [];
|
|
40
|
+
track(() => values.push(s()));
|
|
41
|
+
s.set(2);
|
|
42
|
+
expect(values).toEqual([1, 2]);
|
|
43
|
+
});
|
|
44
|
+
});
|
|
45
|
+
describe("runEffect", () => {
|
|
46
|
+
it("sets activeEffect to the given value", () => {
|
|
47
|
+
const fn = () => { };
|
|
48
|
+
runEffect(fn);
|
|
49
|
+
expect(activeEffect).toBe(fn);
|
|
50
|
+
runEffect(null);
|
|
51
|
+
expect(activeEffect).toBeNull();
|
|
52
|
+
});
|
|
53
|
+
});
|
|
54
|
+
//# sourceMappingURL=track.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"track.test.js","sourceRoot":"","sources":["../../src/__tests__/track.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAC;AAElD,OAAO,EAAE,KAAK,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,kBAAkB,CAAC;AAClE,OAAO,EAAE,MAAM,EAAE,MAAM,kBAAkB,CAAC;AAE1C,QAAQ,CAAC,OAAO,EAAE,GAAG,EAAE;IACrB,EAAE,CAAC,6BAA6B,EAAE,GAAG,EAAE;QACrC,MAAM,EAAE,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC;QACnB,KAAK,CAAC,EAAE,CAAC,CAAC;QACV,MAAM,CAAC,EAAE,CAAC,CAAC,oBAAoB,EAAE,CAAC;IACpC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,kDAAkD,EAAE,GAAG,EAAE;QAC1D,IAAI,MAAM,GAAY,SAAS,CAAC;QAChC,KAAK,CAAC,GAAG,EAAE;YACT,MAAM,GAAG,YAAY,CAAC;QACxB,CAAC,CAAC,CAAC;QACH,MAAM,CAAC,MAAM,CAAC,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC;QACtC,MAAM,CAAC,YAAY,CAAC,CAAC,QAAQ,EAAE,CAAC;IAClC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,qDAAqD,EAAE,GAAG,EAAE;QAC7D,MAAM,KAAK,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC;QACtB,MAAM,KAAK,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC;QACtB,IAAI,WAAoB,CAAC;QACzB,IAAI,WAAoB,CAAC;QAEzB,KAAK,CAAC,GAAG,EAAE;YACT,KAAK,EAAE,CAAC;YACR,WAAW,GAAG,YAAY,CAAC;YAC3B,KAAK,CAAC,GAAG,EAAE;gBACT,KAAK,EAAE,CAAC;gBACR,WAAW,GAAG,YAAY,CAAC;YAC7B,CAAC,CAAC,CAAC;YACH,0DAA0D;YAC1D,MAAM,CAAC,YAAY,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QACzC,CAAC,CAAC,CAAC;QAEH,MAAM,CAAC,KAAK,CAAC,CAAC,oBAAoB,EAAE,CAAC;QACrC,MAAM,CAAC,KAAK,CAAC,CAAC,oBAAoB,EAAE,CAAC;QACrC,MAAM,CAAC,WAAW,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;IAC5C,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,yDAAyD,EAAE,GAAG,EAAE;QACjE,MAAM,CAAC,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;QACpB,MAAM,MAAM,GAAa,EAAE,CAAC;QAC5B,KAAK,CAAC,GAAG,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;QAC9B,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;QACT,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;IACjC,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,WAAW,EAAE,GAAG,EAAE;IACzB,EAAE,CAAC,sCAAsC,EAAE,GAAG,EAAE;QAC9C,MAAM,EAAE,GAAG,GAAG,EAAE,GAAE,CAAC,CAAC;QACpB,SAAS,CAAC,EAAE,CAAC,CAAC;QACd,MAAM,CAAC,YAAY,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAC9B,SAAS,CAAC,IAAI,CAAC,CAAC;QAChB,MAAM,CAAC,YAAY,CAAC,CAAC,QAAQ,EAAE,CAAC;IAClC,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"reactive.d.ts","sourceRoot":"","sources":["../../src/reactive/reactive.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,kBAAkB,CAAC;AAKzD,wBAAgB,IAAI,CAAC,CAAC,EACpB,MAAM,EAAE,MAAM,CAAC,CAAC,CAAC,GAAG,QAAQ,CAAC,CAAC,CAAC,EAC/B,EAAE,EAAE,CAAC,KAAK,EAAE,WAAW,CAAC,CAAC,CAAC,KAAK,IAAI,
|
|
1
|
+
{"version":3,"file":"reactive.d.ts","sourceRoot":"","sources":["../../src/reactive/reactive.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,kBAAkB,CAAC;AAKzD,wBAAgB,IAAI,CAAC,CAAC,EACpB,MAAM,EAAE,MAAM,CAAC,CAAC,CAAC,GAAG,QAAQ,CAAC,CAAC,CAAC,EAC/B,EAAE,EAAE,CAAC,KAAK,EAAE,WAAW,CAAC,CAAC,CAAC,KAAK,IAAI,cAsBpC;AAED,wBAAgB,KAAK,CAAC,CAAC,EAAE,MAAM,EAAE,MAAM,CAAC,CAAC,CAAC,GAAG,QAAQ,CAAC,CAAC,CAAC,2BAQvD;AAED,wBAAgB,SAAS,CAAC,CAAC,EAAE,MAAM,EAAE,MAAM,CAAC,CAAC,CAAC,GAAG,QAAQ,CAAC,CAAC,CAAC,EAAE,EAAE,EAAE,MAAM,aAcvE"}
|
|
@@ -2,14 +2,21 @@ import { signal } from "../signal";
|
|
|
2
2
|
import { effect } from "../signal/effect";
|
|
3
3
|
export function when(source, fn) {
|
|
4
4
|
let disposed = false;
|
|
5
|
+
const ref = { cancel: undefined };
|
|
5
6
|
const stop = effect(() => {
|
|
6
7
|
const value = source();
|
|
7
8
|
if (!value || disposed)
|
|
8
9
|
return;
|
|
9
10
|
disposed = true;
|
|
10
|
-
|
|
11
|
+
ref.cancel?.();
|
|
11
12
|
fn(value);
|
|
12
13
|
});
|
|
14
|
+
ref.cancel = stop;
|
|
15
|
+
// If source was truthy on the first synchronous run, ref.cancel?.() above
|
|
16
|
+
// was a no-op because it hadn't been assigned yet. Cancel the effect now.
|
|
17
|
+
// eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
|
|
18
|
+
if (disposed)
|
|
19
|
+
stop();
|
|
13
20
|
return stop;
|
|
14
21
|
}
|
|
15
22
|
export function until(source) {
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"reactive.js","sourceRoot":"","sources":["../../src/reactive/reactive.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,MAAM,EAAE,MAAM,WAAW,CAAC;AACnC,OAAO,EAAE,MAAM,EAAE,MAAM,kBAAkB,CAAC;AAE1C,MAAM,UAAU,IAAI,CAClB,MAA+B,EAC/B,EAAmC;IAEnC,IAAI,QAAQ,GAAG,KAAK,CAAC;
|
|
1
|
+
{"version":3,"file":"reactive.js","sourceRoot":"","sources":["../../src/reactive/reactive.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,MAAM,EAAE,MAAM,WAAW,CAAC;AACnC,OAAO,EAAE,MAAM,EAAE,MAAM,kBAAkB,CAAC;AAE1C,MAAM,UAAU,IAAI,CAClB,MAA+B,EAC/B,EAAmC;IAEnC,IAAI,QAAQ,GAAG,KAAK,CAAC;IACrB,MAAM,GAAG,GAAG,EAAE,MAAM,EAAE,SAAqC,EAAE,CAAC;IAE9D,MAAM,IAAI,GAAG,MAAM,CAAC,GAAG,EAAE;QACvB,MAAM,KAAK,GAAG,MAAM,EAAE,CAAC;QACvB,IAAI,CAAC,KAAK,IAAI,QAAQ;YAAE,OAAO;QAE/B,QAAQ,GAAG,IAAI,CAAC;QAChB,GAAG,CAAC,MAAM,EAAE,EAAE,CAAC;QACf,EAAE,CAAC,KAAK,CAAC,CAAC;IACZ,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,MAAM,GAAG,IAAI,CAAC;IAElB,0EAA0E;IAC1E,0EAA0E;IAC1E,uEAAuE;IACvE,IAAI,QAAQ;QAAE,IAAI,EAAE,CAAC;IAErB,OAAO,IAAI,CAAC;AACd,CAAC;AAED,MAAM,UAAU,KAAK,CAAI,MAA+B;IACtD,OAAO,IAAI,OAAO,CAAiB,CAAC,OAAO,EAAE,EAAE;QAC7C,MAAM,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC,KAAK,EAAE,EAAE;YAClC,OAAO,CAAC,KAAK,CAAC,CAAC;QACjB,CAAC,CAAC,CAAC;QAEH,KAAK,IAAI,CAAC;IACZ,CAAC,CAAC,CAAC;AACL,CAAC;AAED,MAAM,UAAU,SAAS,CAAI,MAA+B,EAAE,EAAU;IACtE,MAAM,OAAO,GAAG,MAAM,CAAI,MAAM,EAAE,CAAC,CAAC;IACpC,IAAI,OAAkD,CAAC;IAEvD,MAAM,CAAC,GAAG,EAAE;QACV,MAAM,KAAK,GAAG,MAAM,EAAE,CAAC;QACvB,IAAI,OAAO;YAAE,YAAY,CAAC,OAAO,CAAC,CAAC;QACnC,OAAO,GAAG,UAAU,CAAC,GAAG,EAAE;YACxB,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;YACnB,OAAO,GAAG,SAAS,CAAC;QACtB,CAAC,EAAE,EAAE,CAAC,CAAC;IACT,CAAC,CAAC,CAAC;IAEH,OAAO,OAAO,CAAC;AACjB,CAAC"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"persisted.d.ts","sourceRoot":"","sources":["../../src/signal/persisted.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,kBAAkB,CAAC;AAI/C,MAAM,WAAW,sBAAsB,CAAC,CAAC;IACvC,SAAS,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC,KAAK,MAAM,CAAC;IACjC,WAAW,CAAC,EAAE,CAAC,KAAK,EAAE,MAAM,KAAK,CAAC,CAAC;IACnC,QAAQ,CAAC,EAAE,OAAO,CAAC;CACpB;AAED,wBAAgB,eAAe,CAAC,CAAC,EAC/B,GAAG,EAAE,MAAM,EACX,YAAY,EAAE,CAAC,EACf,OAAO,GAAE,sBAAsB,CAAC,CAAC,CAAM,
|
|
1
|
+
{"version":3,"file":"persisted.d.ts","sourceRoot":"","sources":["../../src/signal/persisted.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,kBAAkB,CAAC;AAI/C,MAAM,WAAW,sBAAsB,CAAC,CAAC;IACvC,SAAS,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC,KAAK,MAAM,CAAC;IACjC,WAAW,CAAC,EAAE,CAAC,KAAK,EAAE,MAAM,KAAK,CAAC,CAAC;IACnC,QAAQ,CAAC,EAAE,OAAO,CAAC;CACpB;AAED,wBAAgB,eAAe,CAAC,CAAC,EAC/B,GAAG,EAAE,MAAM,EACX,YAAY,EAAE,CAAC,EACf,OAAO,GAAE,sBAAsB,CAAC,CAAC,CAAM,aAwExC"}
|
package/dist/signal/persisted.js
CHANGED
|
@@ -39,19 +39,17 @@ export function persistedSignal(key, initialValue, options = {}) {
|
|
|
39
39
|
}
|
|
40
40
|
if (syncTabs) {
|
|
41
41
|
window.addEventListener("storage", (event) => {
|
|
42
|
-
if (event.key
|
|
42
|
+
if (event.key !== key || event.storageArea !== localStorage)
|
|
43
43
|
return;
|
|
44
|
-
{
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
inner.set(initialValue);
|
|
54
|
-
}
|
|
44
|
+
try {
|
|
45
|
+
const newValue = event.newValue
|
|
46
|
+
? deserialize(event.newValue)
|
|
47
|
+
: initialValue;
|
|
48
|
+
inner.set(newValue);
|
|
49
|
+
}
|
|
50
|
+
catch (e) {
|
|
51
|
+
console.warn(`Failed to deserialize value for key "${key}" from storage event:`, e);
|
|
52
|
+
inner.set(initialValue);
|
|
55
53
|
}
|
|
56
54
|
});
|
|
57
55
|
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"persisted.js","sourceRoot":"","sources":["../../src/signal/persisted.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,MAAM,EAAE,MAAM,UAAU,CAAC;AAQlC,MAAM,UAAU,eAAe,CAC7B,GAAW,EACX,YAAe,EACf,UAAqC,EAAE;IAEvC,MAAM,EACJ,SAAS,GAAG,IAAI,CAAC,SAAS,EAC1B,WAAW,GAAG,IAAI,CAAC,KAA6B,EAChD,QAAQ,GAAG,IAAI,GAChB,GAAG,OAAO,CAAC;IAEZ,SAAS,cAAc;QACrB,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,YAAY,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;YACzC,OAAO,MAAM,CAAC,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC;QACrD,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,CAAC,IAAI,CAAC,wCAAwC,GAAG,IAAI,EAAE,CAAC,CAAC,CAAC;YACjE,OAAO,YAAY,CAAC;QACtB,CAAC;IACH,CAAC;IAED,SAAS,cAAc,CAAC,KAAQ;QAC9B,IAAI,CAAC;YACH,IAAI,KAAK,KAAK,SAAS,IAAI,KAAK,KAAK,IAAI,EAAE,CAAC;gBAC1C,YAAY,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC;YAC/B,CAAC;iBAAM,CAAC;gBACN,YAAY,CAAC,OAAO,CAAC,GAAG,EAAE,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC;YAC9C,CAAC;QACH,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,CAAC,IAAI,CAAC,sCAAsC,GAAG,IAAI,EAAE,CAAC,CAAC,CAAC;QACjE,CAAC;IACH,CAAC;IAED,MAAM,KAAK,GAAG,MAAM,CAAI,cAAc,EAAE,CAAC,CAAC;IAE1C,SAAS,IAAI;QACX,OAAO,KAAK,EAAE,CAAC;IACjB,CAAC;IAED,SAAS,GAAG,CAAC,KAAQ;QACnB,cAAc,CAAC,KAAK,CAAC,CAAC;QACtB,KAAK,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;IACnB,CAAC;IAED,SAAS,MAAM,CAAC,EAAkB;QAChC,MAAM,QAAQ,GAAG,EAAE,CAAC,KAAK,EAAE,CAAC,CAAC;QAC7B,cAAc,CAAC,QAAQ,CAAC,CAAC;QACzB,KAAK,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;IACtB,CAAC;IAED,IAAI,QAAQ,EAAE,CAAC;QACb,MAAM,CAAC,gBAAgB,CAAC,SAAS,EAAE,CAAC,KAAK,EAAE,EAAE;YAC3C,IAAI,KAAK,CAAC,GAAG,KAAK,GAAG,IAAI,KAAK,CAAC,WAAW,KAAK,YAAY;gBAAE,OAAO;YACpE,
|
|
1
|
+
{"version":3,"file":"persisted.js","sourceRoot":"","sources":["../../src/signal/persisted.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,MAAM,EAAE,MAAM,UAAU,CAAC;AAQlC,MAAM,UAAU,eAAe,CAC7B,GAAW,EACX,YAAe,EACf,UAAqC,EAAE;IAEvC,MAAM,EACJ,SAAS,GAAG,IAAI,CAAC,SAAS,EAC1B,WAAW,GAAG,IAAI,CAAC,KAA6B,EAChD,QAAQ,GAAG,IAAI,GAChB,GAAG,OAAO,CAAC;IAEZ,SAAS,cAAc;QACrB,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,YAAY,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;YACzC,OAAO,MAAM,CAAC,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC;QACrD,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,CAAC,IAAI,CAAC,wCAAwC,GAAG,IAAI,EAAE,CAAC,CAAC,CAAC;YACjE,OAAO,YAAY,CAAC;QACtB,CAAC;IACH,CAAC;IAED,SAAS,cAAc,CAAC,KAAQ;QAC9B,IAAI,CAAC;YACH,IAAI,KAAK,KAAK,SAAS,IAAI,KAAK,KAAK,IAAI,EAAE,CAAC;gBAC1C,YAAY,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC;YAC/B,CAAC;iBAAM,CAAC;gBACN,YAAY,CAAC,OAAO,CAAC,GAAG,EAAE,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC;YAC9C,CAAC;QACH,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,CAAC,IAAI,CAAC,sCAAsC,GAAG,IAAI,EAAE,CAAC,CAAC,CAAC;QACjE,CAAC;IACH,CAAC;IAED,MAAM,KAAK,GAAG,MAAM,CAAI,cAAc,EAAE,CAAC,CAAC;IAE1C,SAAS,IAAI;QACX,OAAO,KAAK,EAAE,CAAC;IACjB,CAAC;IAED,SAAS,GAAG,CAAC,KAAQ;QACnB,cAAc,CAAC,KAAK,CAAC,CAAC;QACtB,KAAK,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;IACnB,CAAC;IAED,SAAS,MAAM,CAAC,EAAkB;QAChC,MAAM,QAAQ,GAAG,EAAE,CAAC,KAAK,EAAE,CAAC,CAAC;QAC7B,cAAc,CAAC,QAAQ,CAAC,CAAC;QACzB,KAAK,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;IACtB,CAAC;IAED,IAAI,QAAQ,EAAE,CAAC;QACb,MAAM,CAAC,gBAAgB,CAAC,SAAS,EAAE,CAAC,KAAK,EAAE,EAAE;YAC3C,IAAI,KAAK,CAAC,GAAG,KAAK,GAAG,IAAI,KAAK,CAAC,WAAW,KAAK,YAAY;gBAAE,OAAO;YACpE,IAAI,CAAC;gBACH,MAAM,QAAQ,GAAG,KAAK,CAAC,QAAQ;oBAC7B,CAAC,CAAC,WAAW,CAAC,KAAK,CAAC,QAAQ,CAAC;oBAC7B,CAAC,CAAC,YAAY,CAAC;gBACjB,KAAK,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;YACtB,CAAC;YAAC,OAAO,CAAC,EAAE,CAAC;gBACX,OAAO,CAAC,IAAI,CACV,wCAAwC,GAAG,uBAAuB,EAClE,CAAC,CACF,CAAC;gBACF,KAAK,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;YAC1B,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED,MAAM,MAAM,GAAG,IAAiB,CAAC;IACjC,MAAM,CAAC,GAAG,GAAG,GAAG,CAAC;IACjB,MAAM,CAAC,MAAM,GAAG,MAAM,CAAC;IACvB,MAAM,CAAC,SAAS,GAAG,KAAK,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC/C,MAAM,CAAC,UAAU,GAAG,IAAI,CAAC;IAEzB,OAAO,MAAM,CAAC;AAChB,CAAC"}
|
package/package.json
CHANGED
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
import { describe, it, expect } from "vitest";
|
|
2
|
+
|
|
3
|
+
import { batch } from "../signal/batch";
|
|
4
|
+
import { signal } from "../signal/signal";
|
|
5
|
+
|
|
6
|
+
describe("batch", () => {
|
|
7
|
+
it("executes the wrapped function", () => {
|
|
8
|
+
const s = signal(0);
|
|
9
|
+
batch(() => {
|
|
10
|
+
s.set(5);
|
|
11
|
+
s.set(10);
|
|
12
|
+
});
|
|
13
|
+
expect(s()).toBe(10);
|
|
14
|
+
});
|
|
15
|
+
|
|
16
|
+
it("does not throw when no effects are tracking", () => {
|
|
17
|
+
const s = signal(0);
|
|
18
|
+
expect(() => { batch(() => { s.set(1); }); }).not.toThrow();
|
|
19
|
+
expect(s()).toBe(1);
|
|
20
|
+
});
|
|
21
|
+
|
|
22
|
+
it("propagates errors thrown inside the batch", () => {
|
|
23
|
+
expect(() =>
|
|
24
|
+
{ batch(() => {
|
|
25
|
+
throw new Error("inside batch");
|
|
26
|
+
}); },
|
|
27
|
+
).toThrow("inside batch");
|
|
28
|
+
});
|
|
29
|
+
});
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
import { describe, it, expect } from "vitest";
|
|
2
|
+
|
|
3
|
+
import { RootComponent } from "../component/base";
|
|
4
|
+
import { StatefulComponent } from "../component/stateful";
|
|
5
|
+
|
|
6
|
+
class ConcreteRoot extends RootComponent<{ name: string }> {
|
|
7
|
+
render() { return null; }
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
class ConcreteStateful extends StatefulComponent {
|
|
11
|
+
render() { return null; }
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
describe("RootComponent", () => {
|
|
15
|
+
it("stores props on _rawProps", () => {
|
|
16
|
+
const c = new ConcreteRoot({ name: "Alice" });
|
|
17
|
+
expect(c._rawProps.name).toBe("Alice");
|
|
18
|
+
});
|
|
19
|
+
|
|
20
|
+
it("exposes props via .props getter", () => {
|
|
21
|
+
const c = new ConcreteRoot({ name: "Bob" });
|
|
22
|
+
expect(c.props.name).toBe("Bob");
|
|
23
|
+
});
|
|
24
|
+
|
|
25
|
+
it("_mounted starts as false", () => {
|
|
26
|
+
const c = new ConcreteRoot({ name: "" });
|
|
27
|
+
expect(c._mounted).toBe(false);
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
it("_anchor starts as undefined", () => {
|
|
31
|
+
const c = new ConcreteRoot({ name: "" });
|
|
32
|
+
expect(c._anchor).toBeUndefined();
|
|
33
|
+
});
|
|
34
|
+
|
|
35
|
+
it("defaults to empty object when no props given", () => {
|
|
36
|
+
const c = new ConcreteRoot();
|
|
37
|
+
expect(c._rawProps).toEqual({});
|
|
38
|
+
});
|
|
39
|
+
});
|
|
40
|
+
|
|
41
|
+
describe("StatefulComponent", () => {
|
|
42
|
+
it("_defaults starts as empty object", () => {
|
|
43
|
+
const c = new ConcreteStateful();
|
|
44
|
+
expect(c._defaults).toEqual({});
|
|
45
|
+
});
|
|
46
|
+
|
|
47
|
+
it("_stateDirty starts as false", () => {
|
|
48
|
+
const c = new ConcreteStateful();
|
|
49
|
+
expect(c._stateDirty).toBe(false);
|
|
50
|
+
});
|
|
51
|
+
|
|
52
|
+
it("_setProps replaces _rawProps contents", () => {
|
|
53
|
+
const c = new ConcreteStateful({ x: 1 });
|
|
54
|
+
c._setProps({ y: 2 });
|
|
55
|
+
expect((c._rawProps).x).toBeUndefined();
|
|
56
|
+
expect((c._rawProps).y).toBe(2);
|
|
57
|
+
});
|
|
58
|
+
});
|
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
import { describe, it, expect, vi } from "vitest";
|
|
2
|
+
|
|
3
|
+
import { computed } from "../signal/computed";
|
|
4
|
+
import { signal } from "../signal/signal";
|
|
5
|
+
|
|
6
|
+
describe("computed", () => {
|
|
7
|
+
it("derives value from signal", () => {
|
|
8
|
+
const s = signal(4);
|
|
9
|
+
const doubled = computed(() => s() * 2);
|
|
10
|
+
expect(doubled()).toBe(8);
|
|
11
|
+
});
|
|
12
|
+
|
|
13
|
+
it("updates when source signal changes", () => {
|
|
14
|
+
const s = signal(3);
|
|
15
|
+
const triple = computed(() => s() * 3);
|
|
16
|
+
s.set(5);
|
|
17
|
+
expect(triple()).toBe(15);
|
|
18
|
+
});
|
|
19
|
+
|
|
20
|
+
it("is lazy — does not recompute until accessed", () => {
|
|
21
|
+
const s = signal(1);
|
|
22
|
+
const fn = vi.fn(() => s() + 1);
|
|
23
|
+
const c = computed(fn);
|
|
24
|
+
expect(fn).not.toHaveBeenCalled();
|
|
25
|
+
c();
|
|
26
|
+
expect(fn).toHaveBeenCalledOnce();
|
|
27
|
+
});
|
|
28
|
+
|
|
29
|
+
it("caches value between reads when source has not changed", () => {
|
|
30
|
+
const s = signal(2);
|
|
31
|
+
const fn = vi.fn(() => s() * 10);
|
|
32
|
+
const c = computed(fn);
|
|
33
|
+
c();
|
|
34
|
+
c();
|
|
35
|
+
expect(fn).toHaveBeenCalledOnce();
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
it("recomputes after source change", () => {
|
|
39
|
+
const s = signal(1);
|
|
40
|
+
const fn = vi.fn(() => s() + 100);
|
|
41
|
+
const c = computed(fn);
|
|
42
|
+
c();
|
|
43
|
+
s.set(2);
|
|
44
|
+
c();
|
|
45
|
+
expect(fn).toHaveBeenCalledTimes(2);
|
|
46
|
+
});
|
|
47
|
+
|
|
48
|
+
it("marks __isComputed = true", () => {
|
|
49
|
+
const c = computed(() => 42);
|
|
50
|
+
expect(c.__isComputed).toBe(true);
|
|
51
|
+
});
|
|
52
|
+
|
|
53
|
+
it("chained computed values update correctly", () => {
|
|
54
|
+
const s = signal(1);
|
|
55
|
+
const doubled = computed(() => s() * 2);
|
|
56
|
+
const quadrupled = computed(() => doubled() * 2);
|
|
57
|
+
expect(quadrupled()).toBe(4);
|
|
58
|
+
s.set(3);
|
|
59
|
+
expect(quadrupled()).toBe(12);
|
|
60
|
+
});
|
|
61
|
+
|
|
62
|
+
it("subscribe fires immediately with computed value", () => {
|
|
63
|
+
const s = signal(5);
|
|
64
|
+
const c = computed(() => s() + 1);
|
|
65
|
+
const received: number[] = [];
|
|
66
|
+
c.subscribe((v) => received.push(v));
|
|
67
|
+
expect(received).toEqual([6]);
|
|
68
|
+
});
|
|
69
|
+
|
|
70
|
+
it("subscribe fires when source changes", () => {
|
|
71
|
+
const s = signal(0);
|
|
72
|
+
const c = computed(() => s() * 3);
|
|
73
|
+
const received: number[] = [];
|
|
74
|
+
c.subscribe((v) => received.push(v));
|
|
75
|
+
s.set(2);
|
|
76
|
+
s.set(4);
|
|
77
|
+
expect(received).toEqual([0, 6, 12]);
|
|
78
|
+
});
|
|
79
|
+
|
|
80
|
+
it("unsubscribe stops receiving updates", () => {
|
|
81
|
+
const s = signal(0);
|
|
82
|
+
const c = computed(() => s() + 1);
|
|
83
|
+
const received: number[] = [];
|
|
84
|
+
const unsub = c.subscribe((v) => received.push(v));
|
|
85
|
+
unsub();
|
|
86
|
+
s.set(10);
|
|
87
|
+
expect(received).toEqual([1]);
|
|
88
|
+
});
|
|
89
|
+
});
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
import { describe, it, expect, vi } from "vitest";
|
|
2
|
+
|
|
3
|
+
import { effect } from "../signal/effect";
|
|
4
|
+
import { signal } from "../signal/signal";
|
|
5
|
+
|
|
6
|
+
describe("effect", () => {
|
|
7
|
+
it("runs immediately on creation", () => {
|
|
8
|
+
const ran = vi.fn();
|
|
9
|
+
effect(ran);
|
|
10
|
+
expect(ran).toHaveBeenCalledOnce();
|
|
11
|
+
});
|
|
12
|
+
|
|
13
|
+
it("re-runs when a tracked signal changes", () => {
|
|
14
|
+
const s = signal(1);
|
|
15
|
+
const values: number[] = [];
|
|
16
|
+
effect(() => {
|
|
17
|
+
values.push(s());
|
|
18
|
+
});
|
|
19
|
+
s.set(2);
|
|
20
|
+
s.set(3);
|
|
21
|
+
expect(values).toEqual([1, 2, 3]);
|
|
22
|
+
});
|
|
23
|
+
|
|
24
|
+
it("stop() calls the cleanup function", () => {
|
|
25
|
+
const cleanupFn = vi.fn();
|
|
26
|
+
const stop = effect(() => cleanupFn);
|
|
27
|
+
stop();
|
|
28
|
+
expect(cleanupFn).toHaveBeenCalledOnce();
|
|
29
|
+
});
|
|
30
|
+
|
|
31
|
+
it("calls cleanup function before re-run", () => {
|
|
32
|
+
const s = signal(0);
|
|
33
|
+
const order: string[] = [];
|
|
34
|
+
effect(() => {
|
|
35
|
+
const v = s();
|
|
36
|
+
order.push(`run:${v}`);
|
|
37
|
+
return () => order.push(`cleanup`);
|
|
38
|
+
});
|
|
39
|
+
s.set(1);
|
|
40
|
+
expect(order).toEqual(["run:0", "cleanup", "run:1"]);
|
|
41
|
+
});
|
|
42
|
+
|
|
43
|
+
it("tracks multiple signals", () => {
|
|
44
|
+
const a = signal(1);
|
|
45
|
+
const b = signal(10);
|
|
46
|
+
const sums: number[] = [];
|
|
47
|
+
effect(() => {
|
|
48
|
+
sums.push(a() + b());
|
|
49
|
+
});
|
|
50
|
+
a.set(2);
|
|
51
|
+
b.set(20);
|
|
52
|
+
expect(sums).toEqual([11, 12, 22]);
|
|
53
|
+
});
|
|
54
|
+
});
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
import { describe, it, expect } from "vitest";
|
|
2
|
+
|
|
3
|
+
import { computed } from "../signal/computed";
|
|
4
|
+
import { effect } from "../signal/effect";
|
|
5
|
+
import { peek } from "../signal/peek";
|
|
6
|
+
import { signal } from "../signal/signal";
|
|
7
|
+
|
|
8
|
+
describe("peek", () => {
|
|
9
|
+
it("returns the current value without tracking", () => {
|
|
10
|
+
const s = signal(42);
|
|
11
|
+
expect(peek(s)).toBe(42);
|
|
12
|
+
});
|
|
13
|
+
|
|
14
|
+
it("does not subscribe the active effect when peeked", () => {
|
|
15
|
+
const s = signal(1);
|
|
16
|
+
const runs: number[] = [];
|
|
17
|
+
|
|
18
|
+
effect(() => {
|
|
19
|
+
runs.push(peek(s)); // peek — should not subscribe
|
|
20
|
+
});
|
|
21
|
+
|
|
22
|
+
s.set(99); // should NOT trigger re-run
|
|
23
|
+
expect(runs).toEqual([1]);
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
it("works with computed values", () => {
|
|
27
|
+
const s = signal(5);
|
|
28
|
+
const c = computed(() => s() * 2);
|
|
29
|
+
expect(peek(c)).toBe(10);
|
|
30
|
+
});
|
|
31
|
+
});
|