first commit
This commit is contained in:
77
packages/admin/tests/lib/hooks.test.tsx
Normal file
77
packages/admin/tests/lib/hooks.test.tsx
Normal file
@@ -0,0 +1,77 @@
|
||||
import { userEvent } from "@vitest/browser/context";
|
||||
import * as React from "react";
|
||||
import { describe, it, expect, vi } from "vitest";
|
||||
import { render } from "vitest-browser-react";
|
||||
|
||||
import { useStableCallback } from "../../src/lib/hooks";
|
||||
|
||||
/**
|
||||
* Test component that attaches a keydown listener using useStableCallback.
|
||||
* Verifies that re-renders with new callback identities don't cause
|
||||
* listener churn, and the latest callback is always invoked.
|
||||
*/
|
||||
function KeydownListener({ onEscape }: { onEscape: () => void }) {
|
||||
const stableOnEscape = useStableCallback(onEscape);
|
||||
|
||||
React.useEffect(() => {
|
||||
const handler = (e: KeyboardEvent) => {
|
||||
if (e.key === "Escape") stableOnEscape();
|
||||
};
|
||||
document.addEventListener("keydown", handler);
|
||||
return () => document.removeEventListener("keydown", handler);
|
||||
}, [stableOnEscape]);
|
||||
|
||||
return <div data-testid="listener">Listening</div>;
|
||||
}
|
||||
|
||||
describe("useStableCallback", () => {
|
||||
it("calls the latest callback after re-render with new identity", async () => {
|
||||
const first = vi.fn();
|
||||
const second = vi.fn();
|
||||
|
||||
const screen = await render(<KeydownListener onEscape={first} />);
|
||||
await expect.element(screen.getByTestId("listener")).toBeInTheDocument();
|
||||
|
||||
// First callback should work
|
||||
await userEvent.keyboard("{Escape}");
|
||||
expect(first).toHaveBeenCalledTimes(1);
|
||||
|
||||
// Re-render with a new callback identity
|
||||
await screen.rerender(<KeydownListener onEscape={second} />);
|
||||
|
||||
// Second callback should be called, not the first
|
||||
await userEvent.keyboard("{Escape}");
|
||||
expect(second).toHaveBeenCalledTimes(1);
|
||||
expect(first).toHaveBeenCalledTimes(1); // still just the one from before
|
||||
});
|
||||
|
||||
it("does not add/remove listeners on re-render", async () => {
|
||||
const addSpy = vi.spyOn(document, "addEventListener");
|
||||
const removeSpy = vi.spyOn(document, "removeEventListener");
|
||||
|
||||
const screen = await render(<KeydownListener onEscape={vi.fn()} />);
|
||||
await expect.element(screen.getByTestId("listener")).toBeInTheDocument();
|
||||
|
||||
const addCountAfterMount = addSpy.mock.calls.filter(([type]) => type === "keydown").length;
|
||||
const removeCountAfterMount = removeSpy.mock.calls.filter(
|
||||
([type]) => type === "keydown",
|
||||
).length;
|
||||
|
||||
// Re-render 3 times with different callback identities
|
||||
for (let i = 0; i < 3; i++) {
|
||||
await screen.rerender(<KeydownListener onEscape={vi.fn()} />);
|
||||
}
|
||||
|
||||
const addCountAfterRerenders = addSpy.mock.calls.filter(([type]) => type === "keydown").length;
|
||||
const removeCountAfterRerenders = removeSpy.mock.calls.filter(
|
||||
([type]) => type === "keydown",
|
||||
).length;
|
||||
|
||||
// No new addEventListener/removeEventListener calls for keydown
|
||||
expect(addCountAfterRerenders).toBe(addCountAfterMount);
|
||||
expect(removeCountAfterRerenders).toBe(removeCountAfterMount);
|
||||
|
||||
addSpy.mockRestore();
|
||||
removeSpy.mockRestore();
|
||||
});
|
||||
});
|
||||
Reference in New Issue
Block a user