Implement share and export components

This commit is contained in:
adilallo
2026-04-29 22:27:46 -06:00
parent a31a36c926
commit a37a72c71d
58 changed files with 3153 additions and 117 deletions
+40
View File
@@ -0,0 +1,40 @@
import { describe, it, expect, vi } from "vitest";
import userEvent from "@testing-library/user-event";
import { renderWithProviders as render, screen } from "../../utils/test-utils";
import "@testing-library/jest-dom/vitest";
import ListItem from "../../../app/components/layout/ListItem";
import Popover from "../../../app/components/modals/Popover";
describe("Popover (export menu)", () => {
it("exposes a menu landmark with localized label", () => {
render(
<Popover id="export-menu" menuAriaLabel="Export format">
<ListItem
showDivider
leadingIcon="markdown_copy"
label="Download Markdown"
onClick={vi.fn()}
/>
</Popover>,
);
expect(screen.getByRole("menu", { name: "Export format" })).toBeTruthy();
expect(screen.getByRole("menuitem", { name: "Download Markdown" })).toBeTruthy();
});
it("invokes handler when list item clicked", async () => {
const user = userEvent.setup();
const onCsv = vi.fn();
render(
<Popover id="popover-csv" menuAriaLabel="Pick format">
<ListItem
showDivider={false}
leadingIcon="csv"
label="Download CSV"
onClick={onCsv}
/>
</Popover>,
);
await user.click(screen.getByRole("menuitem", { name: "Download CSV" }));
expect(onCsv).toHaveBeenCalledTimes(1);
});
});
+80
View File
@@ -0,0 +1,80 @@
import { describe, it, expect, vi } from "vitest";
import userEvent from "@testing-library/user-event";
import { renderWithProviders as render, screen } from "../../utils/test-utils";
import "@testing-library/jest-dom/vitest";
import Share from "../../../app/components/modals/Share";
const noopHandlers = {
onCopyLink: vi.fn(),
onEmailShare: vi.fn(),
onSignalShare: vi.fn(),
onSlackShare: vi.fn(),
onDiscordShare: vi.fn(),
};
describe("Share modal", () => {
it("does not render dialog when closed", () => {
render(
<Share isOpen={false} onClose={vi.fn()} {...noopHandlers} />,
);
expect(screen.queryByRole("dialog")).not.toBeInTheDocument();
});
it("renders localized heading and copy link action when open", async () => {
const user = userEvent.setup();
const onCopyLink = vi.fn();
render(
<Share
isOpen={true}
onClose={vi.fn()}
{...noopHandlers}
onCopyLink={onCopyLink}
/>,
);
expect(screen.getByRole("dialog")).toBeInTheDocument();
expect(
screen.getByRole("heading", { level: 1, name: /Share this CommunityRule/ }),
).toBeInTheDocument();
await user.click(screen.getByRole("button", { name: "Copy link" }));
expect(onCopyLink).toHaveBeenCalledTimes(1);
});
it("invokes channel handlers for Signal, Slack, and Discord", async () => {
const user = userEvent.setup();
const onSignalShare = vi.fn();
const onSlackShare = vi.fn();
const onDiscordShare = vi.fn();
render(
<Share
isOpen={true}
onClose={vi.fn()}
{...noopHandlers}
onSignalShare={onSignalShare}
onSlackShare={onSlackShare}
onDiscordShare={onDiscordShare}
/>,
);
await user.click(screen.getByRole("button", { name: "Signal" }));
await user.click(screen.getByRole("button", { name: "Slack" }));
await user.click(screen.getByRole("button", { name: "Discord" }));
expect(onSignalShare).toHaveBeenCalledTimes(1);
expect(onSlackShare).toHaveBeenCalledTimes(1);
expect(onDiscordShare).toHaveBeenCalledTimes(1);
});
it("calls onClose when Done is clicked", async () => {
const user = userEvent.setup();
const onClose = vi.fn();
render(<Share isOpen={true} onClose={onClose} {...noopHandlers} />);
await user.click(screen.getByRole("button", { name: "Done" }));
expect(onClose).toHaveBeenCalledTimes(1);
});
it("calls onClose when header overflow (more) is activated, matching modal chrome parity", async () => {
const user = userEvent.setup();
const onClose = vi.fn();
render(<Share isOpen={true} onClose={onClose} {...noopHandlers} />);
await user.click(screen.getByRole("button", { name: "More share options" }));
expect(onClose).toHaveBeenCalledTimes(1);
});
});