// @vitest-environment jsdom import { act } from "react"; import type { ReactNode } from "react"; import { createRoot } from "react-dom/client"; import { afterEach, beforeEach, describe, expect, it, vi } from "vitest"; import { SidebarSection } from "./SidebarSection"; import { Plus } from "lucide-react"; const sidebarState = vi.hoisted(() => ({ isMobile: false, })); vi.mock("@/lib/router", () => ({ Link: ({ children, to, ...props }: { children: ReactNode; to: string }) => ( {children} ), })); vi.mock("../context/SidebarContext", () => ({ useSidebar: () => sidebarState, })); // eslint-disable-next-line @typescript-eslint/no-explicit-any (globalThis as any).IS_REACT_ACT_ENVIRONMENT = true; if (!globalThis.PointerEvent) { // eslint-disable-next-line @typescript-eslint/no-explicit-any (globalThis as any).PointerEvent = MouseEvent; } async function flushReact() { await act(async () => { await Promise.resolve(); await new Promise((resolve) => window.setTimeout(resolve, 0)); }); } async function openSectionMenu(container: HTMLElement) { const trigger = container.querySelector('button[aria-label="Projects section actions"]'); expect(trigger).not.toBeNull(); await act(async () => { trigger?.dispatchEvent(new PointerEvent("pointerdown", { bubbles: true, button: 0 })); trigger?.dispatchEvent(new MouseEvent("click", { bubbles: true })); }); await flushReact(); } describe("SidebarSection", () => { let container: HTMLDivElement; let root: ReturnType | null; beforeEach(() => { sidebarState.isMobile = false; container = document.createElement("div"); document.body.appendChild(container); root = null; }); afterEach(async () => { const currentRoot = root; if (currentRoot) { await act(async () => { currentRoot.unmount(); }); } container.remove(); document.body.innerHTML = ""; vi.clearAllMocks(); }); it("keeps static and collapsible section labels on the same text column", async () => { const currentRoot = createRoot(container); root = currentRoot; await act(async () => { currentRoot.render(
Issues Projects
, ); }); await flushReact(); const workLabel = Array.from(container.querySelectorAll("span")) .find((element) => element.textContent === "Work"); const projectsLabel = Array.from(container.querySelectorAll("span")) .find((element) => element.textContent === "Projects"); expect(workLabel?.parentElement?.textContent).toBe("Work"); expect(projectsLabel?.parentElement?.textContent).toBe("Projects"); expect(projectsLabel?.parentElement?.querySelector("svg")).toBeNull(); expect(container.querySelector('button[aria-label="Collapse Projects"] svg')).toBeTruthy(); }); it("keeps collapse on the caret and opens the menu from the heading", async () => { const onOpenChange = vi.fn(); const currentRoot = createRoot(container); root = currentRoot; await act(async () => { currentRoot.render( Projects , ); }); await flushReact(); await openSectionMenu(container); expect(onOpenChange).not.toHaveBeenCalled(); expect(document.body.textContent).toContain("Browse projects"); const caret = container.querySelector('button[aria-label="Collapse Projects"]'); await act(async () => { caret?.dispatchEvent(new MouseEvent("click", { bubbles: true })); }); expect(onOpenChange).toHaveBeenCalledWith(false); }); it("does not apply hover background classes to static section labels", async () => { const currentRoot = createRoot(container); root = currentRoot; await act(async () => { currentRoot.render( Issues , ); }); await flushReact(); const workLabel = Array.from(container.querySelectorAll("span")) .find((element) => element.textContent === "Work"); const staticLabelControl = workLabel?.parentElement; expect(staticLabelControl?.tagName).toBe("DIV"); expect(staticLabelControl?.getAttribute("class")).not.toContain("hover:bg-accent/50"); expect(staticLabelControl?.getAttribute("class")).not.toContain("focus-visible:bg-accent/50"); }); it("keeps the header action outside the label menu hit area", async () => { const onAction = vi.fn(); const currentRoot = createRoot(container); root = currentRoot; await act(async () => { currentRoot.render( Projects , ); }); await flushReact(); const sectionMenuTrigger = container.querySelector('button[aria-label="Projects section actions"]'); const newProjectButton = container.querySelector('button[aria-label="New project"]'); expect(sectionMenuTrigger?.textContent).toContain("Projects"); expect(sectionMenuTrigger?.querySelector("svg")).toBeNull(); expect(sectionMenuTrigger?.getAttribute("class")).toContain("hover:bg-accent/50"); expect(newProjectButton).toBeTruthy(); await act(async () => { newProjectButton?.dispatchEvent(new MouseEvent("click", { bubbles: true })); }); await flushReact(); expect(onAction).toHaveBeenCalledTimes(1); expect(document.body.textContent).not.toContain("Browse projects"); await openSectionMenu(container); expect(document.body.textContent).toContain("Browse projects"); }); it("renders configured menu actions and radio choices", async () => { const onAction = vi.fn(); const onRadioValueChange = vi.fn(); const currentRoot = createRoot(container); root = currentRoot; await act(async () => { currentRoot.render( Projects , ); }); await flushReact(); await openSectionMenu(container); const newProjectItem = Array.from(document.body.querySelectorAll('[data-slot="dropdown-menu-item"]')) .find((element) => element.textContent?.includes("New project")); expect(newProjectItem).toBeTruthy(); const browseLink = Array.from(document.body.querySelectorAll("a")) .find((element) => element.textContent?.includes("Browse projects")); expect(browseLink?.getAttribute("href")).toBe("/projects"); const alphabeticalItem = Array.from(document.body.querySelectorAll('[data-slot="dropdown-menu-radio-item"]')) .find((element) => element.textContent?.includes("Alphabetical")); expect(alphabeticalItem).toBeTruthy(); await act(async () => { alphabeticalItem?.dispatchEvent(new MouseEvent("click", { bubbles: true })); }); expect(onRadioValueChange).toHaveBeenCalledWith("alphabetical"); await openSectionMenu(container); const reopenedNewProjectItem = Array.from(document.body.querySelectorAll('[data-slot="dropdown-menu-item"]')) .find((element) => element.textContent?.includes("New project")); await act(async () => { reopenedNewProjectItem?.dispatchEvent(new MouseEvent("click", { bubbles: true })); }); expect(onAction).toHaveBeenCalledTimes(1); }); it("keeps section header controls visible on mobile", async () => { sidebarState.isMobile = true; const currentRoot = createRoot(container); root = currentRoot; await act(async () => { currentRoot.render( Projects , ); }); await flushReact(); const projectsLabel = Array.from(container.querySelectorAll("span")) .find((element) => element.textContent === "Projects"); const caret = container.querySelector('button[aria-label="Expand Projects"] svg'); const action = container.querySelector('button[aria-label="New project"]'); expect(caret?.getAttribute("class")).toContain("opacity-100"); expect(caret?.getAttribute("class")).not.toContain("opacity-0"); expect(projectsLabel?.parentElement?.textContent).toBe("Projects"); expect(action?.getAttribute("class")).toContain("opacity-100"); expect(action?.getAttribute("class")).not.toContain("opacity-0"); }); });