Emdash source with visual editor image upload fix
Fixes: 1. media.ts: wrap placeholder generation in try-catch 2. toolbar.ts: check r.ok, display error message in popover
This commit is contained in:
126
packages/admin/tests/components/SectionEditor.test.tsx
Normal file
126
packages/admin/tests/components/SectionEditor.test.tsx
Normal file
@@ -0,0 +1,126 @@
|
||||
import { Toasty } from "@cloudflare/kumo";
|
||||
import { QueryClient, QueryClientProvider } from "@tanstack/react-query";
|
||||
import * as React from "react";
|
||||
import { describe, it, expect, vi, beforeEach } from "vitest";
|
||||
|
||||
import type { Section } from "../../src/lib/api";
|
||||
import { render } from "../utils/render.tsx";
|
||||
|
||||
// Capture props passed to PortableTextEditor so the test can invoke the
|
||||
// block-sidebar callbacks the way the image node view does at runtime.
|
||||
const portableTextProps: { current: Record<string, unknown> | null } = { current: null };
|
||||
|
||||
vi.mock("../../src/components/PortableTextEditor", () => ({
|
||||
PortableTextEditor: (props: Record<string, unknown>) => {
|
||||
portableTextProps.current = props;
|
||||
return <div data-testid="portable-text-editor" />;
|
||||
},
|
||||
}));
|
||||
|
||||
vi.mock("../../src/components/MediaPickerModal", () => ({
|
||||
MediaPickerModal: () => null,
|
||||
}));
|
||||
|
||||
vi.mock("@tanstack/react-router", async () => {
|
||||
const actual = await vi.importActual("@tanstack/react-router");
|
||||
return {
|
||||
...(actual as Record<string, unknown>),
|
||||
Link: ({ children, to, ...props }: { children: React.ReactNode; to?: string }) => (
|
||||
<a href={String(to ?? "")} {...props}>
|
||||
{children}
|
||||
</a>
|
||||
),
|
||||
useParams: () => ({ slug: "footer" }),
|
||||
useNavigate: () => vi.fn(),
|
||||
};
|
||||
});
|
||||
|
||||
const mockFetchSection = vi.fn<() => Promise<Section>>();
|
||||
const mockUpdateSection = vi.fn();
|
||||
|
||||
vi.mock("../../src/lib/api", async () => {
|
||||
const actual = await vi.importActual("../../src/lib/api");
|
||||
return {
|
||||
...(actual as Record<string, unknown>),
|
||||
fetchSection: (...args: unknown[]) => mockFetchSection(...(args as [])),
|
||||
updateSection: (...args: unknown[]) => mockUpdateSection(...(args as [])),
|
||||
};
|
||||
});
|
||||
|
||||
// Import after mocks so the module under test picks up the mocked deps.
|
||||
const { SectionEditor } = await import("../../src/components/SectionEditor");
|
||||
|
||||
function makeSection(overrides: Partial<Section> = {}): Section {
|
||||
return {
|
||||
id: "sec_footer",
|
||||
slug: "footer",
|
||||
title: "Footer",
|
||||
description: "All page footer",
|
||||
keywords: [],
|
||||
content: [],
|
||||
source: "user",
|
||||
createdAt: "2025-01-01T00:00:00Z",
|
||||
updatedAt: "2025-01-02T00:00:00Z",
|
||||
...overrides,
|
||||
};
|
||||
}
|
||||
|
||||
function Wrapper({ children }: { children: React.ReactNode }) {
|
||||
const qc = new QueryClient({
|
||||
defaultOptions: { queries: { retry: false }, mutations: { retry: false } },
|
||||
});
|
||||
return (
|
||||
<QueryClientProvider client={qc}>
|
||||
<Toasty>{children}</Toasty>
|
||||
</QueryClientProvider>
|
||||
);
|
||||
}
|
||||
|
||||
describe("SectionEditor", () => {
|
||||
beforeEach(() => {
|
||||
vi.clearAllMocks();
|
||||
portableTextProps.current = null;
|
||||
mockFetchSection.mockResolvedValue(makeSection());
|
||||
});
|
||||
|
||||
it("opens the image settings panel when a block requests the sidebar", async () => {
|
||||
const screen = await render(<SectionEditor />, { wrapper: Wrapper });
|
||||
|
||||
// Editor must mount (and capture its props) before we can simulate.
|
||||
await expect.element(screen.getByTestId("portable-text-editor")).toBeInTheDocument();
|
||||
|
||||
const onBlockSidebarOpen = portableTextProps.current?.onBlockSidebarOpen as
|
||||
| ((panel: unknown) => void)
|
||||
| undefined;
|
||||
const onBlockSidebarClose = portableTextProps.current?.onBlockSidebarClose as
|
||||
| (() => void)
|
||||
| undefined;
|
||||
|
||||
// Both callbacks must be wired — without them, clicking the image-settings
|
||||
// icon in the editor is a silent no-op (#845).
|
||||
expect(typeof onBlockSidebarOpen).toBe("function");
|
||||
expect(typeof onBlockSidebarClose).toBe("function");
|
||||
|
||||
// Simulate the image node view asking for sidebar space, the same shape
|
||||
// it sends from ImageNode.openSidebar(). expect.element below polls until
|
||||
// React flushes the state update, so we don't need an explicit act wrapper.
|
||||
onBlockSidebarOpen!({
|
||||
type: "image",
|
||||
attrs: {
|
||||
src: "https://example.com/logo.png",
|
||||
alt: "Logo",
|
||||
mediaId: "media-1",
|
||||
width: 400,
|
||||
height: 200,
|
||||
},
|
||||
onUpdate: vi.fn(),
|
||||
onReplace: vi.fn(),
|
||||
onDelete: vi.fn(),
|
||||
onClose: vi.fn(),
|
||||
});
|
||||
|
||||
// The Image Settings panel should now be rendered in the sidebar slot.
|
||||
// The panel renders the image preview using the src we passed.
|
||||
await expect.element(screen.getByRole("img", { name: "Logo" })).toBeInTheDocument();
|
||||
});
|
||||
});
|
||||
Reference in New Issue
Block a user