Mats Bosson 8f075f1792 feat: add 12 components — Tooltip, Popover, HoverCard, Alert, Badge,
Skeleton, Breadcrumbs, Link, Button, Image, Meter, NumberField
Floating components: Tooltip (hover/focus), Popover (click, with focus
trap and dismiss), HoverCard (hover with safe area).
Simple components: Alert (role=alert), Badge (role=status), Skeleton
(loading placeholder with data attributes).
Navigation: Breadcrumbs (nav>ol>li with separators), Link (accessible
anchor with disabled), Button (with disabled click suppression).
Data/Form: Image (Img+Fallback with loading status), Meter (like
Progress for known ranges), NumberField (spinbutton with inc/dec).
302 tests across 46 files, typecheck clean, build produces 176 files.
2026-03-29 19:34:13 +07:00

78 lines
2.4 KiB
TypeScript

import { fireEvent, render, screen } from "@solidjs/testing-library";
import { describe, expect, it } from "vitest";
import { Popover } from "../../../src/components/popover/index";
describe("Popover", () => {
it("content has role=dialog when open", () => {
render(() => (
<Popover defaultOpen>
<Popover.Trigger>Open</Popover.Trigger>
<Popover.Content>Content</Popover.Content>
</Popover>
));
expect(screen.getByRole("dialog")).toBeTruthy();
});
it("trigger has correct ARIA", () => {
render(() => (
<Popover>
<Popover.Trigger>Open</Popover.Trigger>
<Popover.Content>Content</Popover.Content>
</Popover>
));
const trigger = screen.getByText("Open");
expect(trigger.getAttribute("aria-haspopup")).toBe("dialog");
expect(trigger.getAttribute("aria-expanded")).toBe("false");
});
it("click trigger opens", () => {
render(() => (
<Popover>
<Popover.Trigger>Open</Popover.Trigger>
<Popover.Content>Content</Popover.Content>
</Popover>
));
fireEvent.click(screen.getByText("Open"));
expect(screen.getByRole("dialog")).toBeTruthy();
});
it("Escape closes", () => {
render(() => (
<Popover defaultOpen>
<Popover.Trigger>Open</Popover.Trigger>
<Popover.Content>Content</Popover.Content>
</Popover>
));
fireEvent.keyDown(document, { key: "Escape" });
expect(screen.queryByRole("dialog")).toBeNull();
});
it("Close button closes", () => {
render(() => (
<Popover defaultOpen>
<Popover.Trigger>Open</Popover.Trigger>
<Popover.Content>
<Popover.Close>X</Popover.Close>
Content
</Popover.Content>
</Popover>
));
fireEvent.click(screen.getByText("X"));
expect(screen.queryByRole("dialog")).toBeNull();
});
it("controlled mode", () => {
render(() => (
<Popover open={true} onOpenChange={() => {}}>
<Popover.Trigger>Open</Popover.Trigger>
<Popover.Content>Content</Popover.Content>
</Popover>
));
expect(screen.getByRole("dialog")).toBeTruthy();
});
it("content is positioned", () => {
render(() => (
<Popover defaultOpen>
<Popover.Trigger>Open</Popover.Trigger>
<Popover.Content data-testid="content">Content</Popover.Content>
</Popover>
));
expect(screen.getByTestId("content").style.position).toBeTruthy();
});
});