import { describe, it, expect } from "vitest"; import { deepSignal, getDeepSignalRootId, peek } from "../deepSignal"; import { watch, __traverseCount, __resetTraverseCount, traverse, } from "../watch"; import { effect } from "../core"; describe("watch advanced", () => { it("basic patch watcher fires on deep mutations", async () => { const st = deepSignal({ a: { b: { c: 1 } } }); let batches: number = 0; watch(st, ({ patches }) => { if (patches.length) batches++; }); st.a.b.c = 2; st.a.b = { c: 3 } as any; await Promise.resolve(); expect(batches).toBeGreaterThan(0); }); // multi-source value mode removed; patch-only now - skip equivalent // getter source value mode removed in patch-only watcher it("watch once option still stops after first batch", async () => { const st = deepSignal({ a: 1 }); let count = 0; watch( st, () => { count++; }, { once: true, immediate: true } ); st.a = 2; st.a = 3; await Promise.resolve(); expect(count).toBe(1); }); // observe value mode removed; observe is alias of watch }); describe("patches & root ids", () => { it("root ids are unique", () => { const a = deepSignal({}); const b = deepSignal({}); expect(getDeepSignalRootId(a)).not.toBe(getDeepSignalRootId(b)); }); // legacy watchPatches API removed; patch mode only valid for deepSignal roots it("watch throws on non-deepSignal input", () => { expect(() => watch({} as any, () => {})).toThrow(); }); it("Map unsupported does not emit patches", async () => { const m = new Map(); const st = deepSignal({ m }); const patches: any[] = []; const { stopListening: stop } = watch(st, ({ patches: batch }) => patches.push(batch) ); m.set("a", 1); await Promise.resolve(); await Promise.resolve(); expect(patches.length).toBe(0); stop(); }); }); describe("tier3: Set iteration variants", () => { it("entries() iteration proxies nested mutation", async () => { const st = deepSignal({ s: new Set() }); st.s.add({ id: "eEnt", inner: { v: 1 } }); const paths: string[] = []; const { stopListening: stop } = watch(st, ({ patches }) => paths.push(...patches.map((pp: any) => pp.path.join("."))) ); for (const [val] of st.s.entries()) { (val as any).inner.v; } // ensure proxy for (const [val] of st.s.entries()) { (val as any).inner.v = 2; } await Promise.resolve(); await Promise.resolve(); expect(paths.some((p) => p.endsWith("eEnt.inner.v"))).toBe(true); stop(); }); it("forEach iteration proxies nested mutation", async () => { const st = deepSignal({ s: new Set() }); st.s.add({ id: "fe1", data: { n: 1 } }); const { stopListening: stop } = watch(st, () => {}); st.s.forEach((e) => (e as any).data.n); // access st.s.forEach((e) => { (e as any).data.n = 2; }); await Promise.resolve(); await Promise.resolve(); stop(); }); it("keys() iteration returns proxies", async () => { const st = deepSignal({ s: new Set() }); st.s.add({ id: "k1", foo: { x: 1 } }); const { stopListening: stop } = watch(st, () => {}); for (const e of st.s.keys()) { (e as any).foo.x = 2; } await Promise.resolve(); await Promise.resolve(); stop(); }); }); describe("tier3: peek behavior", () => { it("peek does not create reactive dependency on property", async () => { const st = deepSignal({ a: 1 }); let runs = 0; effect(() => { runs++; peek(st, "a"); }); expect(runs).toBe(1); st.a = 2; // Flush microtasks await Promise.resolve(); await Promise.resolve(); expect(runs).toBe(1); // no rerun }); }); describe("tier3: traverse helper direct calls (symbols & sets)", () => { it("traverse counts and respects depth param", () => { __resetTraverseCount(); const obj: any = { a: { b: { c: 1 } } }; traverse(obj, 1); const shallowCount = __traverseCount; __resetTraverseCount(); traverse(obj, 3); const deepCount = __traverseCount; expect(deepCount).toBeGreaterThan(shallowCount); }); });